自定义类型——位段,枚举,共用体(联合体)

简介: ✅<1>主页:C语言的前男友📃<2>知识讲解:结构体,声明,变量初始化,结构体内存对齐🔥<3>创作者:C语言的前男友☂️<4>开发环境:Visual Studio 2022💬<5>前言:今天继续把剩下的自定义类型拉出来说一说。

🍍一.位段

🌽(1)位段的声明

位段的声明和结构体有些相似,但是有两个不同:


1.位段的成员必须是  int ,unsigned int  ,或 signed int 。


2.位段的成员名后边有一个买冒号和一个数字。


例如:

struct A
{
    int _a:2;
    int _b:5;
    int _c:10;
    int _d:30;
};

这里的 A 就是一个位段,那这里的位段A的大小又是多大呢?


🏡(2)位段的内存分配

struct S
{
    char a:3;
    char b:4;
    char c:5;
    char d:4;
};
struct S s = {0};
    s.a = 10;
    s.b = 12;
    s.c = 3;
    s.d = 4;

如何分配的:


1. 位段的成员可以是 int unsigned int signed int 或者是 char (属于整形家族)类型

2. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的。

3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段。



🍐(3) 位段的跨平台问题

1. int 位段被当成有符号数还是无符号数是不确定的。

2. 位段中最大位的数目不能确定。(16位机器最大16,32位机器最大32,写成27,在16位机

器会出问题。

3. 位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。

4. 当一个结构包含两个位段,第二个位段成员比较大,无法容纳于第一个位段剩余的位时,是

舍弃剩余的位还是利用,这是不确定的。


总结:

跟结构相比,位段可以达到同样的效果,但是可以很好的节省空间,但是有跨平台的问题存在。


🍓二.枚举

(1) 枚举类型的定义

enum Day//星期
{
    Mon,
    Tues,
    Wed,
    Thur,
    Fri,
    Sat,
    Sun
};
enum Sex//性别
{
    MALE,
    FEMALE,
    SECRET
};
enum Color//颜色
{
    RED,
    GREEN,
    BLUE
};


以上定义的 enum Day , enum Sex , enum Color 都是枚举类型。 { } 中的内容是枚举类型的可能取值,也叫 枚举常量 。


这些可能取值都是有值的,默认从 0 开始,一次递增 1,当然在定义的时候也可以赋初值


例如:


a8cdaa9cc00443ab96262e0a16da4499.png

3cb14c0e71c04cb9af86ac97d3ea188c.png


🍉  (2)枚举的优点

我们可以使用 #define 定义常量,为什么非要使用枚举?枚举的优点:


1. 增加代码的可读性和可维护性

2. 和#define定义的标识符比较枚举有类型检查,更加严谨。

3. 防止了命名污染(封装)

4. 便于调试

5. 使用方便,一次可以定义多个常量


🍚三.联合体(共用体)

🍎(1)联合类型的定义

联合也是一种特殊的自定义类型这种类型定义的变量也包含一系列的成员,特征是这些成员公用同一块空间(所以联合也叫共用体)。联合体(公用体)使用关键字 union来声明的。


例如:

//联合类型的声明
union Un
{
    char c;
    int i;
};

那么这样的一个联合体(共用体)占用多少空间呢?


🍊(2)联合体(共用体)的特点

联合的成员是共用同一块内存空间的,这样一个联合变量的大小,至少是最大成员的大小(因为联

合至少得有能力保存最大的那个成员)。


我们来验证一下联合体(共用体)共用一块空间的特点。

union Un
{
    int i;
    char c;
};
int main()
{
    union Un un;
    // 下面输出的结果是一样的吗?
    printf("%d\n", &(un.i));
    printf("%d\n", &(un.c));
    return 0;
}


fd3d20ad9be34296bdedb643ac8bd695.png


这里不难看出联合体的两个成员,起始地址是相同的,也就意味着共用一块空间。


🍒(3)借助联合体判断当前机器的大小端问题

       但我们在有两个成员一个是( int )另一个是( char )的联合体中,给整形的成员变量存入一个数,再用字符型去取出来。也就可以来判断出大小端的存储顺序了。


代码:

union Un
{
    int i;
    char c;
};
int main()
{
    union Un un;
    un.i = 0x11223344;
    if (un.c == 11)
    {
        printf("大端\n");
    }
    else
    {
        printf("小端\n");
    }
    return 0;
}


8b1a4e9bfd744ad6abfbb43aa945af0d.png


🍭(4)联合体(共用体)的内存分配

1.联合的大小至少是最大成员的大小。

2.当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。

union Un1
{
    char c[5];
    int i;
};
union Un2
{
    short c[7];
    int i;
};
int main()
{
    //下面输出的结果是什么?
    printf("%d\n", sizeof(union Un1));
    printf("%d\n", sizeof(union Un2));
    return 0;
}





🎸 最后:

莫道官忙身老大,即无年少逐春心。



3.jpg


相关文章
|
3月前
|
负载均衡 算法 调度
基于遗传算法的新的异构分布式系统任务调度算法研究(Matlab代码实现)
基于遗传算法的新的异构分布式系统任务调度算法研究(Matlab代码实现)
209 11
|
3月前
|
机器学习/深度学习 算法 安全
【强化学习应用(八)】基于Q-learning的无人机物流路径规划研究(Python代码实现)
【强化学习应用(八)】基于Q-learning的无人机物流路径规划研究(Python代码实现)
252 6
|
3月前
|
存储 人工智能 数据可视化
规则引擎在医疗实际中的解决方案有哪些?
Together规则引擎通过医疗保健功能集(HFS)实现医疗逻辑自动化,提升互操作性与临床决策效率。它支持可视化建模,集成临床路径与决策支持系统,助力医疗机构优化流程、降低成本,并确保遵循最佳实践与行业标准,适用于医疗管理、临床决策及产品开发等场景。
|
固态存储 iOS开发 MacOS
第三方SSD问题引起电脑频繁重启问题IONVMeController.cpp:5499
第三方SSD问题引起电脑频繁重启问题IONVMeController.cpp:5499
388 1
|
人工智能 小程序 前端开发
小程序二手商城|使用Springboot+vue+微信小程序开发校园二手商城系统
本项目基于Springboot+vue+微信小程序实现了一个校园二手物品商城交易系统。系统的后台使用springboot+mybatis开发实现,后台管理页面使用Vue+ElementUI开发实现,用户端基于微信小程序开发实现。前端用户使用微信登录小程序后,可以在线浏览二手商品,并在线购买下单和评论等,同时自己也可以发布相应的二手商品,并管理自己的订单信息和销售信息。管理员登录后台管理系统可以管理人员、商品分类、商品、订单等相关信息。具体见下面展示。
394 0
|
Oracle 关系型数据库 Java
java操作多数据源将oracle数据同步达梦数据库
java操作多数据源将oracle数据同步达梦数据库
|
存储 Android开发
android Jetpack—ViewModel使用方法和详细原理解析
android Jetpack—ViewModel使用方法和详细原理解析
1250 0
android Jetpack—ViewModel使用方法和详细原理解析
|
JavaScript 前端开发
ESLint—— Failed to load config “standard“ to extend from
ESLint—— Failed to load config “standard“ to extend from
487 0
|
存储 算法 Linux
深入理解Linux内存管理brk 和 sbrk 与以及使用C++ list实现内存分配器
深入理解Linux内存管理brk 和 sbrk 与以及使用C++ list实现内存分配器
1035 0
|
消息中间件 关系型数据库 MySQL
2021年最新Flink读写Kafka数据——Flink数据写入Kafka+从Kafka存入Mysql(二)
2021年最新Flink读写Kafka数据——Flink数据写入Kafka+从Kafka存入Mysql(二)
346 0