C/C++ 结构体的一个高级特性 ―― 指定成员的位数
来源:互联网 发布:网络舆论负面案例 编辑:程序博客网 时间:2024/06/06 09:52
C/C++结构体的一个高级特性――指定成员的位数
在大多数情况下,我们一般这样定义结构体:
struct student
{
unsigned int sex;
unsigned int age;
};
对于一般的应用,这已经能很充分地实现数据了的“封装”。
但是,在实际工程中,往往碰到这样的情况:那就是要用一个基本类型变量中的不同的位表示不同的含义。譬如一个cpu内部的标志寄存器,假设为16 bit,而每个bit都可以表达不同的含义,有的表示结果是否为0,有的表示是否越界等等。这个时候我们用什么数据结构来表达这个寄存器呢?
答案还是结构体!
为达到此目的,我们要用到结构体的高级特性,那就是在基本成员变量的后面添加“: 数据位数”组成新的结构体:
struct xxx
{
成员1类型成员1 : 成员1位数;
成员2类型成员2 : 成员2位数;
成员3类型成员3 : 成员3位数;
};
基本的成员变量就会被拆分!这个语法在初级编程中很少用到,但是在高级程序设计中不断地被用到!例如:
struct student
{
unsigned int sex : 1;
unsigned int age : 15;
};
上述结构体中的两个成员sex和age加起来只占用了一个unsigned int的空间(假设unsigned int为16位)。
基本成员变量被拆分后,访问的方法仍然和访问没有拆分的情况是一样的,例如:
struct student sweek;
sweek.sex = MALE;//这里的MALE只能是0或1,值不能大于1
sweek.age = 20;
虽然拆分基本成员变量在语法上是得到支持的,但是并不等于我们想怎么分就怎么分,例如下面的拆分显然是不合理的:
struct student
{
unsigned int sex : 1;
unsigned int age : 12;
};
这是因为1+12 = 13,不能再组合成一个基本成员,不能组合成char、int或任何类型,这显然是不能“自圆其说”的。
在拆分基本成员变量的情况下,我们要特别注意数据的存放顺序,这还与CPU是Big endian还是Little endian来决定。Little endian和Big endian是CPU存放数据的两种不同顺序。对于整型、长整型等数据类型,Big endian认为第一个字节是最高位字节(按照从低地址到高地址的顺序存放数据的高位字节到低位字节);而Little endian则相反,它认为第一个字节是最低位字节(按照从低地址到高地址的顺序存放数据的低位字节到高位字节)。
我们定义IP包头结构体为:
struct iphdr {
#if defined(__LITTLE_ENDIAN_BITFIELD)
__u8 ihl:4,
version:4;
#elif defined (__BIG_ENDIAN_BITFIELD)
__u8 version:4,
ihl:4;
#else
#error "Please fix <asm/byteorder.h>"
#endif
__u8 tos;
__u16 tot_len;
__u16 id;
__u16 frag_off;
__u8 ttl;
__u8 protocol;
__u16 check;
__u32 saddr;
__u32 daddr;
/*The options start here. */
};
在Little endian模式下,iphdr中定义:
__u8 ihl:4,
version:4;
其存放方式为:
第1字节低4位 ihl
第1字节高4位 version (IP的版本号)
若在Big endian模式下还这样定义,则存放方式为:
第1字节低4位 version (IP的版本号)
第1字节高4位 ihl
这与实际的IP协议是不匹配的,所以在Linux内核源代码中,IP包头结构体的定义利用了宏:
#if defined(__LITTLE_ENDIAN_BITFIELD)
…
#elif defined (__BIG_ENDIAN_BITFIELD)
…
#endif
来区分两种不同的情况。
由此我们总结全文的主要观点:
(1) C/C++语言的结构体支持对其中的基本成员变量按位拆分;
(2) 拆分的位数应该是合乎逻辑的,应仍然可以组合为基本成员变量;
要特别注意拆分后的数据的存放顺序,这一点要结合具体的CPU的结构。
该文是由宋宝华处转载而来的,笔者以前从未知道结构体还可以这样用法,笔者做过尝试,再VC下用过的感受有两点
1、 结构体按位拆分时,虽然宋兄提醒不能拆分如文中红色背景显示的情况,但是本人试过,并非是不可以的,而且如果CPU支持32的话,显然文中的以16位来分配的话也是没有达到要求的。
2、 按位拆分时字节数目问题,我们先看两例
struct student1
{
unsigned char sex : 1;
unsigned int no : 5;
char age : 7;
int grade : 10;
};
struct student2
{
unsigned char sex : 1;
char age : 7;
unsigned int no : 5;
int grade : 10;
};
以上两例中虽然意思并不大,但是如果按int为2字节16位char为1字节8位来划分内存的话,那么student1占用了6字节共48位,但是实际使用了23位,另外25位没定义,而student2占用了3字节共24位,但是实际使用也是23位。这个过程,我把它总结为前后变量的类型不一致时,字节就重新分配。
3、 赋值过程中数据编码问题。还看两例
student1 ss;
ss.age = 255;
student2 st;
st.age= 191;
ss.age的值为-1,而st.age的值为63,其实255为11111111,因为是7位,所以采用截断方式,变成1111111,又因为age是有符号的变量,所以根据负数的编码规则赋值255时得到的结果就是-1。在这里采用了截断的方式,为止正确赋值时一定不能大于位数编码值。
以上是个人学习宋兄文章的小小实践小结,欢迎大家给出更理论的总结。原文请查看http://blog.donews.com/21cnbao/archive/2006/10/07/1054807.aspx
- C/C++ 结构体的一个高级特性 ―― 指定成员的位数
- C/C++结构体的一个高级特性――指定成员的位数
- C/C++ 结构体的一个高级特性 ―― 指定成员的位数
- C/C++结构体的一个高级特性――指定成员的位数
- C/C++ 结构体的一个高级特性 ―― 指定成员的位数
- C/C++结构体的一个高级特性――指定成员的位数
- C/C++结构体的一个高级特性――指定成员的位数
- 结构体的一个高级特性――指定成员的位数
- C - 指定结构体中成员变量的位数
- 指定struct成员变量的位数
- C 如何判断一个数的位数
- 2-结构体的最后一个成员的定义-C语言中的柔性数组-
- 做个笔记——C语言结构体成员赋值的一个小问题
- C的5位数
- 【C/C++】:如何获得一个float数的小数位数?
- C学习笔记 7 写出一个宏,求任意结构体成员相对于结构体首地址的偏移。
- C语言结构的成员对齐
- C高级(5)--C++函数的高级特性
- Objective C 除去字符串中的空格
- Microsoft® SQL Server® 2008 R2 Service Pack 1正式版
- Wifi模块分析
- 嵌入式linux内核与根文件系统制作
- JDBC数据库开发技术
- C/C++ 结构体的一个高级特性 ―― 指定成员的位数
- mysql服务器支持局域网客户端访问设置
- this.PreviousPage.FindControl遇到MasterPage,如何查找控件?
- zoj 2172
- Android学习之Service
- Java内部类总结(转)
- 入门学习1:Debug Certificate expired 解决办法
- ResourceBundle实现本地化和国际化
- 转载一篇文章,关于编码的,受益匪浅