我编译了代码,
#include <stdio.h>

struct s {
    int a : 6;
    _Bool b : 1;
    _Bool c : 1;
    _Bool d : 1;
    _Bool e : 1;
    _Bool f : 1;
    _Bool g : 1;
    int h : 12;
};

void main(void) {
    printf("%d\n",sizeof(struct s));
}

输出有点出乎意料.

12

如C11草案所述,

… If enough space remains,a bit-field that immediately follows another bit-field in a structure shall be packed into adjacent bits of the same unit…

因为我使用了32位编译器,所以我预计它会适合4个字节.具体来说,我使用了gcc(tdm-1)5.1.0.这是违反标准的吗?

编辑:

将所有_Bools替换为int按预期工作…我不知道为什么……

编辑:

在gcc 5.4.0中,代码按预期工作.这个问题的关键点是为什么尾随的_Bools和int不适合第一个.我想我没有做太多关于实现的假设(除了int至少4个字节,这是可以接受的),我在这里谈论C标准的C保证行为.因此,我不能同意下面的一些评论.

解决方法

这些是位字段. “预期”输出的方式并不多,因为这些标准很难指定.此外,编译器往往对它们的支持很差.

首先,您引用的完整部分(6.7.2.1/11)说:

An implementation may allocate any addressable storage unit large
enough to hold a bitfield. If enough space remains,a bit-field that
immediately follows another bit-field in a structure shall be packed
into adjacent bits of the same unit. If insufficient space remains,
whether a bit-field that does not fit is put into the next unit or
overlaps adjacent units is implementation-defined. The order of
allocation of bit-fields within a unit (high-order to low-order or
low-order to high-order) is implementation-defined. The alignment of
the addressable storage unit is unspecified.

所有这一切都意味着你几乎不能对这些位如何在内存中结束做出任何假设.你不可能知道编译器将如何排列对齐,你不可能知道位的位顺序,你无法知道签名,你无法知道endianess.

至于if int和_Bool将合并到同一个“存储单元”……好吧,为什么会这样?它们是不兼容的类型. C标准没有提到当你有两个不相容类型的相邻位字段时会发生什么 – 它可以接受主观解释.我怀疑会有各种类型的别名问题.

因此,编译器完全可以将所有这些放在单独的“存储单元”中.如果您相邻地放置相同类型的项目,我会希望它合并它们或引用的部分没有任何意义.例如,我希望从以下大小8:

struct s {
    int a : 6;
    int h : 12;
    _Bool b : 1;
    _Bool c : 1;
    _Bool d : 1;
    _Bool e : 1;
    _Bool f : 1;
    _Bool g : 1;
};

现在,如果你想要确定性的行为,你应该做什么,可移植代码是将位字段抛出窗口并使用逐位运算符.它们具有100%的确定性和便携性.

假设你实际上不想要一些神秘的签名号码字段,原始代码提示,那么:

#define a UINT32_C(0xFC000000)
#define b (1u << 18)
#define c (1u << 17)
#define d (1u << 16)
#define e (1u << 15)
#define f (1u << 14)
#define g (1u << 13)
#define h UINT32_C(0x00000FFF)

typedef uint32_t special_thing;

然后设计setter / getter函数或宏来设置/获取这个32位块的数据.写得正确,你甚至可以使这样的代码与endianess无关.

位字段的意外行为的更多相关文章

  1. Bitmask在PHP中进行设置?

    位和位掩码是我一直在努力理解一段时间的东西,但是我想学习如何使用它们进行PHP中的设置和设置.我终于找到了一个声称要这样做的课程,我可以告诉我,这似乎是有效的,但我不知道这是否是最好的方法.我将以下面的示例代码发布类文件,以便正常运行.请,如果你有经验,告诉我是否可以改进,性能或任何其他.我真的想学习这个,我一直在阅读,但是对我来说这是一个困难的事情.班上…

  2. 哪一个位字段是最重要的一点?

    我正在使用VisualStudio2008为WindowsXP/Vista/7编写C应用程序.我的一些结构使用一个位字段,如示例所示.该领域的哪一端是最重要的一点?

  3. 在32位机器上使用__int8_t对一些小值进行无用的优化工作?

    我的函数有一些值

  4. 使用C类中的位字段的错误配置属性

    我正在尝试使用C中的位字段来实现特定的类大小,但由于某种原因它比我预期的要大.问题是,一个32位(4字节)的类报告5个字节.下面的示例类:如果切换了four_bit_field和eight_bit_field位置,则sizeof返回适当的大小,4个字节.我相信它可能是一个内存排列问题.那么,有人知道这种行为背后的原因吗?

  5. 为什么位域的类型会影响包含结构的大小?

    )是,为什么gcc这样做?gcc的作者是否认为声明的位字段类型必须影响包含结构的大小和对齐?它们在这个假设中是否正确?我错过了C标准本身的要求吗?

  6. 位字段的意外行为

    我编译了代码,输出有点出乎意料.如C11草案所述,…因为我使用了32位编译器,所以我预计它会适合4个字节.具体来说,我使用了gcc5.1.0.这是违反标准的吗?我不知道为什么……

  7. C中的位域内存管理

    >如何为我预期的行为编写代码?

  8. c – 为什么禁止对bitfields的非const引用?

    C11中的第9.6/3节非常清楚:“非常量参考不应与位域绑定”.这个禁令背后的动机是什么?我明白,不可能直接将引用绑定到位域.但是如果我宣布这样的话,为什么我不能这样说?

  9. 有没有办法控制C中的结构体之间的填充(包括位域)?

    我正在解析网络数据流,我想知道是否有任何方法可以将数据流直接映射到数据结构.例如,我想如下定义RTP协议的数据结构.并以这种方式使用它.但是,由于C编译器将在成员之间插入填充,是否有任何方法来控制,以便在成员之间不添加填充?

  10. c – 比掩码位和手工提取数据更有效(计算)位字段?

    我有许多小块数据,我希望能够将其推送到一个更大的数据类型中.假设,假设这是一个日期和时间.显而易见的方法是通过像这样的位字段.现在让我们假装这个东西是有序的,这样首先声明的东西比后面声明的东西具有更高的重要性,所以如果我用变量的第一个字母表示位,它看起来像:最后,让我假装我只是声明一个unsignedlong并使用mask将其拆分并进行相同的操作.这是我的问题:在计算机需要做什么方面,以下是访问分钟,小时等等的方法吗?

随机推荐

  1. 从C到C#的zlib(如何将byte []转换为流并将流转换为byte [])

    我的任务是使用zlib解压缩数据包(已接收),然后使用算法从数据中生成图片好消息是我在C中有代码,但任务是在C#中完成C我正在尝试使用zlib.NET,但所有演示都有该代码进行解压缩(C#)我的问题:我不想在解压缩后保存文件,因为我必须使用C代码中显示的算法.如何将byte[]数组转换为类似于C#zlib代码中的流来解压缩数据然后如何将流转换回字节数组?

  2. 为什么C标准使用不确定的变量未定义?

    垃圾价值存储在哪里,为什么目的?解决方法由于效率原因,C选择不将变量初始化为某些自动值.为了初始化这些数据,必须添加指令.以下是一个例子:产生:虽然这段代码:产生:你可以看到,一个完整的额外的指令用来移动1到x.这对于嵌入式系统来说至关重要.

  3. 如何使用命名管道从c调用WCF方法?

    更新:通过协议here,我无法弄清楚未知的信封记录.我在网上找不到任何例子.原版的:我有以下WCF服务我输出添加5行,所以我知道服务器是否处理了请求与否.我有一个.NET客户端,我曾经测试这一切,一切正常工作预期.现在我想为这个做一个非托管的C客户端.我想出了如何得到管道的名称,并写信给它.我从here下载了协议我可以写信给管道,但我看不懂.每当我尝试读取它,我得到一个ERROR_broKEN_P

  4. “这”是否保证指向C中的对象的开始?

    我想使用fwrite将一个对象写入顺序文件.班级就像当我将一个对象写入文件时.我正在游荡,我可以使用fwrite(this,sizeof(int),2,fo)写入前两个整数.问题是:这是否保证指向对象数据的开始,即使对象的最开始可能存在虚拟表.所以上面的操作是安全的.解决方法这提供了对象的地址,这不一定是第一个成员的地址.唯一的例外是所谓的标准布局类型.从C11标准:(9.2/20)Apointe

  5. c – 编译单元之间共享的全局const对象

    当我声明并初始化一个const对象时.两个cpp文件包含此标头.和当我构建解决方案时,没有链接错误,你会得到什么如果g_Const是一个非const基本类型!PrintInUnit1()和PrintInUnit2()表明在两个编译单元中有两个独立的“g_Const”具有不同的地址,为什么?

  6. 什么是C名称查找在这里? (&amp;GCC对吗?)

    为什么在第三个变体找到func,但是在实例化的时候,原始变体中不合格查找找不到func?解决方法一般规则是,任何不在模板定义上下文中的内容只能通过ADL来获取.换句话说,正常的不合格查找仅在模板定义上下文中执行.因为在定义中间语句时没有声明func,并且func不在与ns::type相关联的命名空间中,所以代码形式不正确.

  7. c – 在输出参数中使用auto

    有没有办法在这种情况下使用auto关键字:当然,不可能知道什么类型的.因此,解决方案应该是以某种方式将它们合并为一个句子.这可用吗?解决方法看起来您希望默认初始化给定函数期望作为参数的类型的对象.您无法使用auto执行此操作,但您可以编写一个特征来提取函数所需的类型,然后使用它来声明您的变量:然后你就像这样使用它:当然,只要你重载函数,这一切都会失败.

  8. 在C中说“推动一切浮动”的确定性方式

    鉴于我更喜欢将程序中的数字保留为int或任何内容,那么使用这些数字的浮点数等效的任意算术最方便的方法是什么?说,我有我想写通过将转换放在解析的运算符树叶中,无需将表达式转化为混乱是否可以使用C风格的宏?应该用新的类和重载操作符完成吗?解决方法这是一个非常复杂的表达.更好地给它一个名字:现在当您使用整数参数调用它时,由于参数的类型为double,因此使用常规的算术转换将参数转换为double用C11lambda……

  9. objective-c – 如何获取未知大小的NSArray的第一个X元素?

    在objectiveC中,我有一个NSArray,我们称之为NSArray*largeArray,我想要获得一个新的NSArray*smallArray,只有第一个x对象…

  10. c – Setprecision是混乱

    我只是想问一下setprecision,因为我有点困惑.这里是代码:其中x=以下:方程的左边是x的值.1.105=1.10应为1.111.115=1.11应为1.121.125=1.12应为1.131.135=1.14是正确的1.145=1.15也正确但如果x是:2.115=2.12是正确的2.125=2.12应为2.13所以为什么在一定的价值是正确的,但有时是错误的?请启发我谢谢解决方法没有理由期望使用浮点系统可以正确地表示您的帖子中的任何常量.因此,一旦将它们存储在一个双变量中,那么你所拥有的确切的一

返回
顶部