`
7090
  • 浏览: 281677 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

(转)C/C++ 结构体的一个高级特性 ―― 指定成员的位数

 
阅读更多
原文:http://blog.donews.com/21cnbao/archive/2006/10/07/1054807.aspx

在大多数情况下,我们一般这样定义结构体:
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;
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的结构。


分享到:
评论

相关推荐

    C_C++结构体的一个高级特性―指定成员的位数

    ### C/C++结构体的高级特性:指定成员的位数 在C/C++语言中,结构体是一种复合数据类型,允许我们将多个不同类型的变量组织在一起,形成一个整体。这种特性非常强大,尤其当涉及到硬件接口或者需要精细控制内存布局...

    嵌入式C/C++语言开发精华文章集锦(嵌入式C/C++开发者必看,很有价值的文章)

    C/C++结构体的一个高级特性――指定成员的位数 C/C++中的近指令、远指针和巨指针 从两道经典试题谈 C/C++中联合体(union)的使用. 基于 ARM 的嵌入式Linux 移植真实体验 基于 ARM 的嵌入式Linux 移植真实体验(1)...

    C 语言嵌入式系统编程修炼

    C/C++结构体的一个高级特性――指定成员的位数 C/C++中的近指令、远指针和巨指针 从两道经典试题谈 C/C++中联合体(union)的使用 基于 ARM 的嵌入式 Linux 移植真实体验 深入浅出 Linux 设备驱动编程

    嵌入式C/C++精华

    11. **C/C++结构体的一个高级特性——指定成员的位数** - **位字段**:在某些情况下,可能需要在结构体中定义位字段,以节省空间。这在嵌入式系统中尤为重要。 12. **C/C++中的近指令、远指针和巨指针** - **指针...

    c++结构体--简单的学生信息管理

    本cpp文件对于初学c++结构体的学生有用!你正在学结构体,值得一看!

    高质量C++_C编程指南

    - **C/C++结构体的高级特性**:介绍了一种高级特性——指定成员的位数,这是一种在嵌入式编程中特别有用的特性,可以有效地利用内存空间。 - **C/C++中的近指令、远指针和巨指针**:探讨了不同类型的指针及其应用...

    嵌入式C/C++语言精华文章集锦

    C/C++结构体的一个高级特性――指定成员的位数 在某些情况下,为了节省内存空间,可以通过指定结构体成员的位数来优化数据结构。 **关键知识点:** - **位字段的概念**:介绍了位字段的概念及其在C/C++中的使用...

    嵌入式C/C++语言精华文章集锦(免费下载)

    C/C++结构体的一个高级特性――指定成员的位数.......................................................78 C/C++中的近指令、远指针和巨指针......................................................................

    嵌入式C/C++语言精华

    `extern "C"`是一个C++的关键字,用于告诉编译器按照C语言的方式处理链接,这在C和C++混合编程时尤为重要。由于C++支持函数重载,编译器会对函数名进行修饰,以区分具有相同名字但参数不同的函数,这一过程称为名称...

    嵌入式 C C++语言精华文章集锦.pdf

    C C++结构体的一个高级特性――指定成员的位数 78 C C++中的近指令 远指针和巨指针 80 从两道经典试题谈 C C++中联合体(union)的使用 81 基于 ARM 的嵌入式 Linux 移植真实体验 83 基于 ARM 的嵌入式 Linux 移植...

    嵌入式C/C++编程精华文档

    如结构体中指定成员的位数,或近指针、远指针和巨指针的概念,也是嵌入式程序员需要掌握的知识点。 在嵌入式Linux移植方面,程序员需要熟悉Linux内核模块的加载和卸载,字符设备驱动程序的编写,设备驱动中的并发...

    嵌入式C与C++语言精华文章集锦

    C/C++结构体的一个高级特性——指定成员的位数 - **位字段**:C/C++允许开发者指定结构体成员占用的位数,这对于节省内存和精确控制数据布局非常有用。 #### 12. C/C++中的近指令、远指针和巨指针 - **指针类型*...

    C/C++笔试题

    `volatile` 是一个关键字,在 C/C++ 中用来修饰变量。当一个变量被声明为 `volatile` 类型时,编译器不会对它进行优化(例如,将它保存在寄存器中而不是内存中),以确保每次对该变量的读取都从内存中获取最新的值。...

    腾讯c/c++笔试题

    给出一个结构体,求其大小。 **给出的结构体**: ```c++ struct st { int i; short s; char c; }; ``` **解析**: 结构体中的成员变量可能会由于内存对齐的原因而出现填充。默认情况下,编译器通常会按成员类型...

    嵌入式C语言精华文章集锦

    11. **C/C++结构体的一个高级特性――指定成员的位数** - 讲解如何在结构体中定义具有特定位数的成员。 - 探讨这一特性的应用场景。 12. **C/C++中的近指令、远指针和巨指针** - 定义并解释这些术语的含义。 - ...

    基于C/C++的数据结构ADT演示系统(全操作)

    在C/C++编程中,ADT通过结构体和类来实现,提供了一种封装数据和相关操作的方式。 本项目"基于C/C++的数据结构ADT演示系统(全操作)"涵盖了多种基本和重要的数据结构及其操作,包括: 1. **线性表**:线性表是一...

    SHA C/C++算法实现,仅供参考和学习。

    这个压缩包包含了一个C/C++语言实现的SHA算法,对于理解SHA的工作原理和编程实现具有一定的参考价值。 SHA家族包括SHA-0、SHA-1、SHA-256、SHA-384、SHA-512等,其中SHA-1和SHA-256是最常见的。这些算法基于消息...

Global site tag (gtag.js) - Google Analytics