目录

1、概念和定义

2、实例


在做嵌入式开发的时候,我们经常会遇到这样的代码:

struct
{unsigned int widthValidated : 1;unsigned int heightValidated : 1;
} status;

这样定义结构体变量是什么意思呢?

主要原因是:有些信息在存储时,只需占几个或一个二进制位(bit),并不需要占用一个完整的字节。例如,在存放一个开关量时,只有0和1两种状态,用一位二进位即可。为了节省存储空间,并使处理简便,C语言提供了一种数据结构,称为“位域”或“位段”。

1、概念和定义

位域:是把一个字节中的二进位划分为几个不同的区域,并说明每个区域的位数。每个域有一个域名,允许在程序中按域名进行操作。 这样就可以把几个不同的对象用一个字节的二进制位域来表示。

位域定义与结构定义相仿,其形式为:

struct 位域结构名
{位域列表};

其中位域列表的形式为:

type [member_name] : width ;

下面是有关位域中变量元素的描述:

位域的使用和结构体成员的使用相同,其一般形式为:

位域变量名.位域名
位域变量名->位域名

位域最大的作用就是节省存储空间,在本质上就是一种结构类型,不过其成员是按二进位分配的。例如以下案例:

#include <stdio.h>
#include <string.h>/* 定义简单的结构 */
struct
{unsigned int widthValidated;unsigned int heightValidated;
} status1;/* 定义位域结构 */
struct
{unsigned int widthValidated : 1;unsigned int heightValidated : 1;
} status2;int main( )
{printf( "Memory size occupied by status1 : %d\n", sizeof(status1));printf( "Memory size occupied by status2 : %d\n", sizeof(status2));return 0;
}

代码被编译和执行时,它会产生下列结果:

Memory size occupied by status1 : 8
Memory size occupied by status2 : 4

结构体status1是由正常的两个unsigned int类型变量组成,占用内存是8字节,结构体status2也是unsigned int类型变量,但是它仅使用了一个unsigned int类型内存的前2bit,实际上还有30bit没使用,所以占用内存是4字节。

对于位域的定义有以下几点说明:

  • 一个位域存储在同一个字节中,如一个字节所剩空间不够存放另一位域时,则会从下一单元起存放该位域。也可以有意使某位域从下一单元开始。例如:
struct bs{unsigned a:4;unsigned  :4;    /* 空域 */unsigned b:4;    /* 从下一单元开始存放 */unsigned c:4
}

在这个位域定义中,a 占第一字节的 4 位,后 4 位填 0 表示不使用,b 从第二字节开始,占用 4 位,c 占用 4 位。

  • 位域的宽度不能超过它所依附的数据类型的长度,成员变量都是有类型的,这个类型限制了成员变量的最大长度,: 后面的数字不能超过这个长度。

  • 位域可以是无名位域,这时它只用来作填充或调整位置。无名的位域是不能使用的。例如:

struct k{int a:1;int  :2;    /* 该 2 位不能使用 */int b:3;int c:2;
};
  • 当相邻成员的类型相同时,如果它们的位宽之和小于类型的 sizeof 大小,那么后面的成员紧邻前一个成员存储,直到不能容纳为止;如果它们的位宽之和大于类型的 sizeof 大小,那么后面的成员将从新的存储单元开始,其偏移量为类型大小的整数倍。例如:
#include <stdio.h>int main(){struct bs{unsigned m: 6;unsigned n: 12;unsigned p: 4;};printf("%d\n", sizeof(struct bs));return 0;
}

运行结果:

4

m、n、p 的类型都是 unsigned int,sizeof 的结果为 4 个字节(Byte),也即 32 个位(Bit)。m、n、p 的位宽之和为 6+12+4 = 22,小于 32,所以它们会挨着存储,中间没有缝隙。

  • 当相邻成员的类型不同时,不同的编译器有不同的实现方案,GCC 会压缩存储,而 VC/VS 不会。例如:
#include <stdio.h>int main(){struct bs{unsigned m: 12;unsigned char ch: 4;unsigned p: 4;};printf("%d\n", sizeof(struct bs));return 0;
}

在 GCC 下的运行结果为 4,三个成员挨着存储;在 VC/VS 下的运行结果为 12,三个成员按照各自的类型存储(与不指定位宽时的存储方式相同)。

  • 如果成员之间穿插着非位域成员,那么不会进行压缩。 例如:
struct bs{unsigned m: 12;unsigned ch;unsigned p: 4;
};

在各个编译器下 sizeof 的结果都是 12。

注意:位域成员往往不占用完整的字节,有时候也不处于字节的开头位置,因此使用&获取位域成员的地址是没有意义的,C语言也禁止这样做。地址是字节(Byte)的编号,而不是位(bit)的编号。

2、实例

通过一个示例,加深对位域的理解和应用:

#include <stdio.h>
#include <string.h>struct
{unsigned int age : 3;
} Age;int main( )
{Age.age = 4;printf( "Sizeof( Age ) : %d\n", sizeof(Age) );printf( "Age.age : %d\n", Age.age );Age.age = 7;printf( "Age.age : %d\n", Age.age );Age.age = 8; // 二进制表示为 1000 有四位,超出printf( "Age.age : %d\n", Age.age );return 0;
}

当上面的代码被编译时,它会带有警告,当上面的代码被执行时,它会产生下列结果:

Sizeof( Age ) : 4
Age.age : 4
Age.age : 7
Age.age : 0

当执行到Age.age = 8;时,二进制表示为:1000 有四位,超出位域,所以会提示警告。

聊一聊C语言位域/位段相关推荐

  1. (转)C结构体之位域(位段)

    转载自C结构体之位域(位段) 有些信息在存储时,并不需要占用一个完整的字节, 而只需占几个或一个二进制位.例如在存放一个开关量时,只有0和1 两种状态, 用一位二进位即可.为了节省存储空间,并使处理简 ...

  2. 位在c语言中用什么定义,C语言中位段的详细介绍

    C语言中位段的详细介绍 位段(bit-field)是以位为单位来定义结构体(或联合体)中的成员变量所占的空间.含有位段的结构体(联合体)称为位段结构.采用位段结构既能够节省空间,又方便于操作.以下是百 ...

  3. 符号位处理方式 c语言,C语言位域解析符号位扩展规则

    从一个例子说起: int main(void){ union{ int i; struct{ char a : ; char b : ; char c : ; }bits; }num; printf( ...

  4. 聊一聊Go语言的error处理

    前言 Go语言的错误处理是一个常见的操作,经常可以见到一个函数返回错误类型(error),后续通过if err != nil来判断错误以及错误类型.这一次尝试通过Go内置的error接口,聊一聊Go语 ...

  5. C语言位域(位段)详解

    有些数据在存储时并不需要占用一个完整的字节,只需要占用一个或几个二进制位即可.例如开关只有通电和断电两种状态,用 0 和 1 表示足以,也就是用一个二进位.正是基于这种考虑,C语言又提供了一种叫做位域 ...

  6. 纠缠不清的C语言位域(位段)详解

    位域是什么? 有些数据在存储时并不需要占用一个完整的字节,只需要占用一个或几个二进制位即可.例如开关只有通电和断电两种状态,用 0 和 1 表示足以,也就是用一个二进位.正是基于这种考虑,C语言又提供 ...

  7. 新唐c语言怎么计算指数运算,C语言位域精解

    有些信息在存储时,并不需要占用一个完整的字节, 而只需占几个或一个二进制位.例如在存放一个开关量时,只有0和1 两种状态,用一位二进位即可.为了节省存储空间,并使处理简便,C语言又提供了一种数据结构, ...

  8. C++ bit field 位域/位段

    " 位域 " 或 " 位段 "(Bit field)为一种数据结构,可以把数据以位的形式紧凑的储存,并允许程序员对此结构的位进行操作.这种数据结构的一个好处是它 ...

  9. C语言 | 位域的使用详解

    1024G 嵌入式资源大放送!包括但不限于C/C++.单片机.Linux等.关注微信公众号[嵌入式大杂烩],回复1024,即可免费获取! 位域的概念 有些数据在存储时并不需要占用一个完整的字节,只需要 ...

最新文章

  1. 使用C++的Socket实现从客户端到服务端,服务端到客户端传输文件
  2. Windows下开源缺陷跟踪系统mantis安装指南(续)-
  3. Python基础类型之元组
  4. C++的iostream标准库介绍
  5. [mmu/cache]-Cache Type Register(CTR)寄存器介绍-InProgress
  6. Pycharm远程连接Linux服务器
  7. MapReduce进阶:多路径输入输出
  8. OC的项目网址(自己编写的项目)
  9. [网络安全自学篇] 四十二.DNS欺骗和钓鱼网站原理详解及防御机理
  10. rfid射频前端的主要组成部分有_第4章 RFID的射频前端(simple).ppt
  11. 两难!先更新数据库再删缓存?还是先删缓存再更新数据库?
  12. python列表赋值 连续整数_Python_03_字符串_数据类型_for循环_列表操作
  13. java字符串拼接_Java 8中字符串拼接新姿势:StringJoiner
  14. 浅谈百度司南大数据企业的风向标
  15. msiafterburner并行配置不正确_dubbo常用配置及使用场景
  16. win7 64位如何安装sql2005
  17. matlab 产生任意概率密度的联合分布
  18. 几种并行计算模型的区别(BSP LogP PRAM)
  19. 【美学集】色彩之冷暖色
  20. matlab maps 指北针和比例尺,R 地图绘制-比例尺与指北针

热门文章

  1. 《千字文细谈》2021神级程序员都在用什么工具?-09-02
  2. 贝壳找房二手房信息爬虫
  3. 0x00007FFE81272FE1 (ucrtbased.dll) (Project1.exe 中)处有未经处理的异常: 0xC0000005: 读取位置 0x0000000000000000 时发
  4. 电感感应电压公式v(t)=L*di/dt的推导
  5. 外网下载速度过慢问题解决办法
  6. java获取一年的周数、单周开始时间与结束时间、一年所有周开始时间与结束时间、月开始时间与结束时间
  7. Vue3+Vite+TS后台项目 ~ 10.商品管理
  8. 武汉大学 遥感院 数据结构实习
  9. 《大唐卫星网络电视》
  10. 51nod 1556 计算(默慈金数)