位段和联合

来源:互联网 发布:普通disco视频软件 编辑:程序博客网 时间:2024/05/17 22:21

位段:

有些信息在存储时,并不需要占用一个完整的字节, 而只需占几个或一个二进制位。

例如在存放一个开关量时,只有0两种状态, 用一位二进位即可。

为了节省存储空间并使处理简便,C语言又提供了一种数据结构,称为“位域”或“位段”。所谓“位域”是把一个字节中的二进位划分为几个不同的区域并说明每个区域的位数。每个域有一个域名,允许在程序中按域名进行操作 这样就可以把几个不同的对象用一个字节的二进制位域来表示。

 

 如果结构体中含有位域(bit-field),那么VC中准则是:

  1) 如果相邻位域字段的类型相同且其位宽之和小于声明数据类型的sizeof大小,则后面的字段将紧邻前一个字段存储,直到不能容纳为止;

  2) 如果相邻位域字段的类型相同但其位宽之和大于类型的sizeof大小(如:char 的位域长度不能超过8int的字节长度不能超过32),则后面的字段将从新的存储单元开始,其偏移量为其类型大小的整数倍;

  3) 如果相邻的位域字段的类型不同,则各编译器的具体实现有差异,VC6采取不压缩方式(不同位域字段存放在不同的位域类型字节中),Dev-C++GCC都采取压缩方式;

系统会先为结构体成员按照对齐方式分配空间和填塞(padding,然后对变量进行位域操作

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

////位段

//#include<stdio.h>
//#include<stdlib.h>
//
//typedef struct
//{
// unsigned int one : 1;
// unsigned int two : 3;
// unsigned int three : 7;
// unsigned int four : 5;
// unsigned int : 2;
// unsigned int five : 8;
// unsigned int six : 8;
//}demo_type;
//
//int main()
//{
// demo_type s = { 1, 5, 513, 17, 129, 0x81 };
// printf("sizeof demo_type=%u\n", sizeof(demo_type));
// printf("values:s=%u,%u,%u,%u,%u,%u\n",\
// s.one, s.two,s.three, s.four, \
// s.five, s.six);
// system("pause");
// return 0;
//}

代码运行结果:



联合:

union,中文名“联合体、共用体”,在某种程度上类似结构体struct的一种数据结构,共用体(union)和结构体(struct)同样可以包含很多种数据类型和变量。

不过区别也挺明显:

结构体(struct)中所有变量是“共存”的——优点是“有容乃大”,全面;缺点是struct内存空间的分配是粗放的,不管用不用,全分配。

而联合体(union)中是各变量是“互斥”的——缺点就是不够“包容”;但优点是内存使用更为精细灵活,也节省了内存空间。


////联合
//#include<stdio.h>
//#include<stdlib.h>
//
//int main()
//{
// struct 
// {
// char name[10];
// int age;
// char job;
// union
// {
// int class;
// char office[10];
// }depa;
// }body[2];
// int i = 0;
// for (i = 0; i < 2; i++)
// {
// printf("input name,age,job and department\n");
// scanf("%s %d %c", body[i].name, &body[i].age, &body[i].job);
// if (body[i].job == 's')
// scanf("%d", &body[i].depa.class);
// else
// scanf("%s", body[i].depa.office);
// }
// printf("name\tage job class/office\n");
// for (i = 0; i < 2; i++)
// {
// if (body[i].job == 's')
// printf("%s\t%3d%3c%3d\n",\
// body[i].name,\
// body[i].age,\
// body[i].job,\
// body[i].depa.class);
// else
// printf("%s\t%3d%3c%3s\n", \
// body[i].name, \
// body[i].age, \
// body[i].job, \
// body[i].depa.office);
// }
// system("pause");
// return 0;

//}


代码运行结果:


0 0
原创粉丝点击