- 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/
我有这个代码: System.err.print("number of terms = "); System.out.println(allTerms.size()); System.err
我有以下问题:在操作系统是 Linux 的情况下和在操作系统是 MacOs 的情况下,我必须执行不同的操作。 所以我创建了以下 Ant 脚本目标: /u
我正在调用 system("bash ../tools/bashScript\"This is an argument!\"&"),然后我正在调用 close(socketFD) 直接在 system
使用最初生成的随机元素来约束随机数组的连续元素是否有效。 例如:我想生成一组 10 个 addr、size 对来模拟典型的内存分配例程并具有如下类: class abc; rand bit[5:0
我正在创建一个必须使用system(const char*)函数来完成一些“繁重工作”的应用程序,并且我需要能够为用户提供粗略的进度百分比。例如,如果操作系统正在为您移动文件,它会为您提供一个进度条,
我即将编写一些项目经理、开发人员和业务分析师会使用的标准/指南和模板。目标是更好地理解正在开发或已经开发的解决方案。 其中一部分是提供有关记录解决方案的标准/指南。例如。记录解决/满足业务案例/用户需
在开发使用压缩磁盘索引或磁盘文件的应用程序时,其中部分索引或文件被重复访问(为了论证,让我们说一些类似于 Zipfian 分布的东西),我想知道什么时候足够/更好地依赖操作系统级缓存(例如,Debia
我们编写了一个 powershell 脚本,用于处理来自内部系统的图像并将其发送到另一个系统。现在,业务的另一部分希望加入其中,对数据进行自己的处理,并将其推送到另一个系统。打听了一下,公司周围有几个
我正在尝试朗姆酒我的应用程序,但我收到以下错误:System.Web.HttpUnhandledException:引发了“System.Web.HttpUnhandledException”类型的异
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,
所以我在其他程序中没有收到此错误,但我在这个程序中收到了它。 这个程序是一个我没有收到错误的示例。 #include int main() { system("pause"); } // en
我在 c# System.URI.FormatExption 中遇到问题 为了清楚起见,我使用的是 Segseuil 的 Matlab 方法,并且它返回一个图片路径 result。我想为其他用户保存此
我正在尝试像这样设置文本框的背景色: txtCompanyName.BackColor = Drawing.Color.WhiteSmoke; 它不喜欢它,因为它要我在前面添加系统,例如: txtCo
请帮助我解决 System.StackOverflowException我想用 .aspx 将记录写入数据库我使用 4 层架构来实现这一切都正常但是当我编译页面然后它显示要插入数据的字段时,当我将数据
我使用了一些通常由系统调用的API。 因此,我将 android:sharedUserId="android.uid.system" 添加到 manifest.xml, 并使用来自 GIT 的 And
我正在尝试创建一个小型应用程序,它需要对/system 文件夹进行读/写访问(它正在尝试删除一个文件,并创建一个新文件来代替它)。我可以使用 adb 毫无问题地重新挂载该文件夹,如果我这样做,我的应用
我想从没有 su 的系统 priv-app 将/system 重新挂载为 RW。如何以编程方式执行此操作?只会用 Runtime.getruntime().exec() 执行一个 shell 命令吗
我正在尝试制作一个带有登录系统的程序我对此很陌生,但我已经连续工作 8 个小时试图解决这个问题。这是我得到的错误代码 + ServerVersion 'con.ServerVersion' threw
当我“构建并运行”Code::Blocks 中的程序时,它运行得非常好!但是当我从“/bin”文件夹手动运行它时,当它试图用 system() 调用“temp.bat”时,它会重置。这是为什么?它没有
我想使用 system/pipe 命令来执行具有特殊字符的命令。下面是示例代码。通过系统/管道执行命令后,它通过改变特殊字符来改变命令。我很惊讶地看到系统命令正在更改作为命令传递的文本。 run(ch
我是一名优秀的程序员,十分优秀!