联合体和枚举

联合体:

联合体是什么?

       联合体也是一种自定义类型,这种类型定义的变量也包含一系列类型,特征是这些类型公用一块内存空间(所以叫联合体也叫公用体)可以理解为结构体公用一块内存。

//联合-联合体-共用体
//联合也是一种特殊的自定义类型,这种类型定义的变量包含一系列的成员
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;//4
    char 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;//4
	struct 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;
};

       此时兑换一个礼品,就开辟对应的空间,即可对空间进行合理的使用。

       我们来看几道相关练习题:

习题一:

union Un
{
	short s[7];
	int n;
};
int main()
{
  printf("%d\n", sizeof(union Un));
  return 0;
}

       注意这里最大成员是int,所以联合体内存对齐后是16字节。

习题二:

int main()
{
  union
  {
    short k;
    char i[2];
  }*s, a;
  s = &a;
  s->i[0] = 0x39;
  s->i[1] = 0x38;
  printf("%x\n", a.k);
  return 0;
}

      因为是小端存储,数据高位存储在高地址,低位储存在低地址(详细请看大端和小端存储模式-CSDN博客) ,所以以16进制打印输出为:0x3839

 

枚举: 

什么是枚举:

       枚举和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;
}

        枚举如果在函数中声明,就只能在函数中使用,其有生命域。 

总结:

       每个类型都有存在的意义,我们以后在生活中总会使用到它,之后我们会逐渐顿悟其具体的功能。

  • 27
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值