(C)位字段(bit-field)

在存储空间很宝贵的情况下,有可能需要将多个对象保存在一个机器字中,一种常用的方法是:使用类似于编译器符号表的单个二进制位标志集合,外部强加的数据格式(如设备接口等寄存器)经常需要从字的分值中读取数值.

通常采用的方法是:定义一个于相关位的位置对应的”屏蔽码”集合,
如:


    #define KEYWORD  (1<<0)  
    #define EXTRENAL (1<<2)  
    #define STATIC   (1<<3)  

或者


    enum{
        KEYWORD  = 01,  
        EXTRENAL = 02,
        STATIC   = 04
    };

这些数字必须是2的幂,这样就可以用移位运算,屏蔽运算以及补码运算进行简单的操作.比如:


    flags |= EXTEERNAL | STATIC;//置1  
    flags &= ~(EXTEERNAL | STATIC);//置0  

尽管这样的方法容易掌握,但是C语言提供了一种可以替代的方法,即直接定义和方位一个位字段的能力,不必通过以上的逻辑运算符,即位字段.通过位字段,以上的#define定义可以用以下的语句替代:


    struct {
        unsigned int is_keyword : 1;
        unsigned int is_extern : 1;
        unsigned int is_static : 1;
    }flafs;

这里定义一个变量flags,它包含3个1位的字段,冒号后的数字表示字段的宽度(用二进制位数表示),字段被声明为unsigned int,以保证它们的无符号量.
单个字段的引用方式与其他结构成员相同,例如:

flags.is_keyword,
flags.is_extern

等;字段的作用与小整数相似,同其他整数一样,字段可以出现在算数表达式中,因此,可以表示为:


    flags.is_extern = flags.is_static = 1;//置1
    flags.is_extern = flags.is_static = 0;//置0
    if(flags.is_extern == 0 && flags.is_static == 0)
        ...//用于对is_extern和is_static的测试

字段的所有属性几乎都同具体的实现有关,字段可以不命名,无名字段(只有冒号和宽度)起填充作用,特殊宽度0可以用来强制在下一边界上对齐.字段不是数组,没有地址,不能做&取地址操作.

posted @ 2016-07-07 11:57  跑马灯的忧伤  阅读(431)  评论(0编辑  收藏  举报