联合体:
联合体是什么?
联合体也是一种自定义类型,这种类型定义的变量也包含一系列类型,特征是这些类型公用一块内存空间(所以叫联合体也叫公用体)可以理解为结构体公用一块内存。
//联合-联合体-共用体
//联合也是一种特殊的自定义类型,这种类型定义的变量包含一系列的成员
union Un
{char c;int i;
};
//
int main()
{union Un u;printf("%d\n", sizeof(u));printf("%p\n", &u);printf("%p\n", &(u.c));printf("%p\n", &(u.i));return 0;
}
联合的成员是共用同一块内存空间的,这样一个联合变量的大小至少是最大成员的大小(因为联合至少得有能力保存最大的那个成员)。联合体最好不要改动其中成员的值,因为改动一个另外也会跟着改,因为它们共用一块内存一块空间。
联合体的大小:
联合体至少也是最大类型的整数倍。
union Un
{int a;//4char arr[5];//5 1相当于你写了5个char//5不是4的整数倍,对齐后是8
};//联合体大小的计算
//联合体的大小至少是成员的大小
//当最大成员大小不是最大对齐数的整数倍时,就要对齐到最大对齐数的整数倍
int main()
{union Un u;printf("%d\n", sizeof(u));return 0;
}
char arr[5];相当于5个char类型,之后又追加了int,已经超过了4个字节,目前最宽字节为4,加起来一共是9字节,因为这是联合体类型,它们会共用内存。可是联合体有自己的对齐规则,如果超过了里面最大数据类型,就会对齐最大类型的的整数倍。所以这个结果是8。
我们再来看一个例子:
union Un
{char a[5];char b[2];
}u;
int main()
{printf("%d\n", sizeof(u));return 0;
}
联合体的应用:
我们用联合体判断当前系统是小端存储还是大端存储。
//int check_sys()
//{
// int a = 1;
// return *(char*)&a;
//}
int check_sys()
{union{char c;int i;}u;//匿名联合体类型,用一次以后不再用u.i = 1;//返回1 小端//返回0 大端return u.c;
}
int main()
{//int a = 1;int ret = check_sys();//if (1 == *(char*)&a)//{// printf("小端\n");//}//else//{// printf("大端\n");//}if (ret == 1){printf("小端\n");}else{printf("大端\n");}//int a = 0x11223344;//低地址------------->高地址//...[][11][22][33][44][][]...大端字节序存储模式//...[][44][33][22][11][][]...小端字节序存储模式//讨论一个数据放在内存中存放的字节顺序//大小端字节序问题return 0;
}
我们来看另外一种用法:
union U
{int n;//4struct S{char c1;char c2;char c3;char c4;}s;//4
};int main()
{//int n = 0x11223344;//4个字节//用联合体实现union U u = { 0 };u.n = 0x11223344;printf("%x %x %x %x\n", u.s.c1, u.s.c2, u.s.c3, u.s.c4);return 0;
}
其实还用一种实际的用途,比如一个公司搞活动,上线一个礼品兑换单,礼品兑换单中有三种商品:图书、被子、衬衫。
每一种商品都有:库存量、价格、商品类型和商品类型的相关信息。
- 图书:书名、作者、页数
- 杯子:设计
- 衬衫:设计、可选颜色、可选尺寸
如果我们直接使用结构体写出一下形式:
struct gift_list
{//这是每一个商品的共同属性int stock_number;//库存量double price;//定价int item_type;//商品类型//特殊属性char title[20];//书名char author[20];//作者int num_pages;//页数char design[30];//设计int colors;//颜色int sizes;//尺寸
};
这样创建一个结构体但是一个礼品单只能兑换一个商品,只用一个商品有一些属性就会空闲着,这样势必会造成空间的浪费。此时就可以使用联合体:
struct gift_list
{//这是每一个商品的共同属性int stock_number;//库存量double price;//定价int item_type;//商品类型//之后就是指定一个商品union un {struct{char title[20];//书名char author[20];//作者int num_pages;//页数}book;struct{char design[30];//设计}mug;struct{char design[30];//设计int colors;//颜色int sizes;//尺寸}shirt;}item;
};
此时兑换一个礼品,就开辟对应的空间,即可对空间进行合理的使用。
枚举:
什么是枚举:
枚举和define很像。顾名思义就是列举。因为现实生活中,总有一些东西是可以被一一列举的,比如星期,月份等。
//枚举的关键字
enum Sex
{//这里列举枚举enum Sex的可能取值MALE,FEMALE,SECRET
};int main()
{printf("%d\n", MALE);printf("%d\n", FEMALE);printf("%d\n", SECRET);return 0;
}
{}中的内容是枚举类型中可能取的值,也叫枚举常量。打印出出的是常数,这些可能取值都是有值的,默认从0开始,依次增加1,当然定义的时候也可以赋初值。
枚举的使用:
我们也可以对其进行赋值:
//枚举类型
enum Sex
{//枚举的可能取值-常量MALE=9,FEMALE,SECRET
};
enum Color
{RED,GREEN=3,BLUE
};
int main()
{enum Sex s = MALE;enum Color c = BLUE;//只能拿枚举的变量来对这个变量进行赋值printf("%d %d %d\n", MALE, FEMALE, SECRET);printf("%d %d %d\n", RED, GREEN, BLUE);return 0;
}
我们也可以对其逐个赋值:
//枚举的关键字
enum Sex
{//这里列举枚举enum Sex的可能取值MALE = 5,FEMALE = 8,SECRET = 10
};int main()
{printf("%d\n", MALE);printf("%d\n", FEMALE);printf("%d\n", SECRET);return 0;
}
枚举如果在函数中声明,就只能在函数中使用,其有生命域。
总结:
每个类型都有存在的意义,我们以后在生活中总会使用到它,之后我们会逐渐顿悟其具体的功能。