- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我对 limit.h 中的 CHAR_BIT 感到困惑。我读过一些文章说宏 CHAR_BIT 是为了可移植性。在代码中使用宏而不是像 8 这样的魔数(Magic Number),这是合理的。但是limits.h来自glibc-headers,它的值固定为8。如果glibc-headers安装在一个字节超过8位(比如16位)的系统上,那么编译时会出错吗? 'char' 被指定为 8 位还是 16 位?
当我在limits.h中将CHAR_BIT修改为9时,下面的代码仍然打印'8',这是怎么回事?
#include <stdio.h>
#include <limits.h>
int
main(int argc, char **argv)
{
printf("%d\n", CHAR_BIT);
return 0;
}
补充如下:我已经阅读了所有回复,但仍然不清楚。在实践中,#include <limits.h>
并使用 CHAR_BIT,我可以遵守。但那是另一回事了。这里我想知道为什么会这样,首先它是glibc/usr/include/limits.h中的固定值“8”,当那些1字节!= 8位的系统安装了glibc时会发生什么;然后我发现值“8”甚至不是代码使用的实际值,所以“8”意味着什么?如果根本没有使用该值,为什么要把“8”放在那里?
谢谢
最佳答案
深入研究系统头文件可能是一种令人畏惧且不愉快的经历。 glibc 头文件很容易在您的头脑中造成很多困惑,因为它们在某些情况下包含其他系统头文件,这些头文件会覆盖迄今为止定义的内容。
以limits.h
为例,如果你仔细阅读头文件,你会发现CHAR_BIT
的定义仅当您在没有 gcc 的情况下编译代码时才使用,因为这一行:
#define CHAR_BIT 8
位于 if
内条件上面几行:
/* If we are not using GNU CC we have to define all the symbols ourself.
Otherwise use gcc's definitions (see below). */
#if !defined __GNUC__ || __GNUC__ < 2
因此,如果您使用 gcc 编译代码(很可能是这种情况),则 CHAR_BIT
的定义将不会被使用。这就是为什么您更改它并且您的代码仍然打印旧值的原因。在头文件上向下滚动一点,您可以找到您使用 GCC 的情况:
/* Get the compiler's limits.h, which defines almost all the ISO constants.
We put this #include_next outside the double inclusion check because
it should be possible to include this file more than once and still get
the definitions from gcc's header. */
#if defined __GNUC__ && !defined _GCC_LIMITS_H_
/* `_GCC_LIMITS_H_' is what GCC's file defines. */
# include_next <limits.h>
include_next
是 GCC 的扩展。您可以在这个问题中了解它的作用:Why would one use #include_next in a project?
简短回答:它将搜索具有您指定名称的下一个头文件(本例中为 limits.h
),并且它将包含 GCC 生成的 limits.h
。 。在我的系统中,它恰好是 /usr/lib/gcc/i486-linux-gnu/4.7/include-fixed/limits.h
。
考虑以下程序:
#include <stdio.h>
#include <limits.h>
int main(void) {
printf("%d\n", CHAR_BIT);
return 0;
}
使用此程序,您可以在 gcc -E
的帮助下找到系统的路径。 ,它为包含的每个文件输出一个特殊行(请参阅 http://gcc.gnu.org/onlinedocs/cpp/Preprocessor-Output.html )
因为#include <limits.h>
位于该程序的第 2 行,我将其命名为 test.c
,运行gcc -E test.c
允许我找到正在包含的真实文件:
# 2 "test.c" 2
# 1 "/usr/lib/gcc/i486-linux-gnu/4.7/include-fixed/limits.h" 1 3 4
您可以在该文件中找到此内容:
/* Number of bits in a `char'. */
#undef CHAR_BIT
#define CHAR_BIT __CHAR_BIT__
注意 undef
指令:需要覆盖任何可能的先前定义。它是在说:“忘记 CHAR_BIT
是什么,这才是真实的”。 __CHAR_BIT__
是 gcc 预定义的常量。 GCC的在线文档是这样描述的:
__CHAR_BIT__
Defined to the number of bits used in the representation of the char data type. It exists to make the standard header given numerical limits work correctly. You should not use this macro directly; instead, include the appropriate headers.
您可以通过一个简单的程序读取它的值:
#include <stdio.h>
#include <limits.h>
int main(void) {
printf("%d\n", __CHAR_BIT__);
return 0;
}
然后运行gcc -E code.c
。请注意,您不应该直接使用它,正如 gcc 的联机帮助页提到的那样。
显然,如果你改变CHAR_BIT
里面的定义/usr/lib/gcc/i486-linux-gnu/4.7/include-fixed/limits.h
,或者您系统中的任何等效路径,您将能够在代码中看到此更改。考虑这个简单的程序:
#include <stdio.h>
#include <limits.h>
int main(void) {
printf("%d\n", CHAR_BIT);
return 0;
}
改变CHAR_BIT
gcc 中的定义 limits.h
(即 /usr/lib/gcc/i486-linux-gnu/4.7/include-fixed/limits.h
中的文件)来自 __CHAR_BIT__
to 9 将使此代码打印 9。同样,您可以在预处理发生后停止编译过程;你可以用gcc -E
来测试它.
如果您使用 gcc 以外的编译器编译代码怎么办?
好吧,那就这样吧,默认 ANSI 限制是针对标准 32 位字的。来自 ANSI C 标准第 5.2.4.2.1 段(整数类型的大小 <limits.h>
):
The values given below shall be replaced by constant expressions suitable for use in #if preprocessing directives. [...] Their implementation-defined values shall be equal or greater in magnitude (absolute value) to those shown, with the same sign.
number of bits for smallest object that is not a bit-field (byte)
CHAR_BIT 8
POSIX 要求兼容平台具有 CHAR_BIT == 8
.
当然,对于没有 CHAR_BIT == 8
的机器,glibc 的假设可能会出错。 ,但请注意,您必须处于不寻常的架构下,并且不使用 gcc 并且您的平台不兼容 POSIX。不太可能。
但是请记住,“实现定义”意味着编译器编写者选择发生的情况。因此,即使您没有使用 gcc
进行编译,您的编译器有可能有某种 __CHAR_BIT__
等效定义。即使 glibc 不会使用它,您也可以做一些研究并直接使用编译器的定义。这通常是不好的做法 - 您将编写针对特定编译器的代码。
请记住,您永远不应该弄乱系统头文件。当您使用错误且重要的常量(例如 CHAR_BIT
)编译内容时,可能会发生非常奇怪的事情。 。这样做仅用于教育目的,并始终恢复原始文件。
关于char - 修复了各种系统上的 CHAR_BIT?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19708810/
我对 limit.h 中的 CHAR_BIT 感到困惑。我读过一些文章说宏 CHAR_BIT 是为了可移植性。在代码中使用宏而不是像 8 这样的魔数(Magic Number),这是合理的。但是lim
CHAR_BIT 定义每个字符的位数。但它是一个旧的宏。类似的宏已在最近的 C++ 标准中被替换。 C++11/14/17 中是否有更现代的 CHAR_BIT 替代品? 最佳答案 非符号位数由 std
引用计算整数绝对值 (abs) 的代码而不从 http://graphics.stanford.edu/~seander/bithacks.html 分支: int v; // w
我只是 checking an answer并意识到 CHAR_BIT没有像我期望的那样由标题定义,甚至没有 #include , 在较新的 GCC 上。 我真的需要#include 只是为了获得
我有一个用 C 语言实现的位 vector ,我想避免除法和模数运算,并用(更快的)位运算替换它们。因此,我将位放入字节(字符)中,并需要索引到这些数组中,然后取出一些位。我的实现如下所示: #inc
ISO C 标准要求 CHAR_BIT 至少为 8。 POSIX 要求 CHAR_BIT 等于 8,并且(几乎?)所有使用 octets 的网络和通信标准,是否有任何当代C 实现,其中 CHAR_BI
根据 N4140(C++11 工作草案): The fundamental storage unit in the C ++ memory model is the byte. A byte is a
似乎没有任何库函数可以计算类型的位大小。 我认为这可以通过以下方式完成吗? #include template size_t Size_In_Bits(){ return sizeof(T
在 Tanenbaum 的操作系统设计和实现中,第 154 页说位图中的每个 NR_SYS_PROCS(32) 都有一个位。 并且在 minix/kernel/table.c 的末尾,有一个检查以确保
在 POSIX 基本原理中有一个说明,强制 CHAR_BIT 为 8 是为了保持与 C99 的一致性而不丢弃套接字/网络所必需的让步,但我从来没有看到冲突到底是什么的解释。有没有人有轶事或引文说明为什
我正在考虑一个非常特殊的(现在没用的)处理器,叫做 saturn . 这个 CPU 的一个奇怪特征是它的元素单位是半字节(4 位),而不是字节(8 位)。这不仅仅是装饰性的,任何指针值都以半字节表示。
我手边没有任何带有 CHAR_BIT > 8 的平台。 所以我更改了文件limits.h中的相关预处理器定义: #define CHAR_BIT 16 /* number
1.请考虑以下事项: unsigned int a, b; b = a >> ((sizeof a) * CHAR_BIT); /* or 2nd operand greater than ((siz
如果目标系统上的 CHAR_BIT == 8(大多数情况下),则很容易屏蔽掉单个字节: unsigned char lsb = foo & 0xFF; 但是,在一些系统和 C 实现中,CHAR_BIT
如果目标系统上的 CHAR_BIT == 8(大多数情况下),则很容易屏蔽掉单个字节: unsigned char lsb = foo & 0xFF; 但是,在一些系统和 C 实现中,CHAR_BIT
在 C99 的 7.19.7.1 节中,我们有: If the end-of-file indicator for the input stream pointed to by stream is n
C# 是否有类似于 C++ 的 CHAR_BIT 的东西? ? 更新: 基本上,我试图在没有分支的情况下计算 abs,这是 C++ 版本: // Compute the integer absolut
是否存在机器(或编译器),其中 sizeof(char) != 1? C99 标准 是否规定sizeof(char) 在标准合规性实现中必须正好为 1?如果有,请给我章节编号和引用。 更新:如果我有一
我是一名优秀的程序员,十分优秀!