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

简介: ✅<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


相关文章
|
6月前
|
存储 C语言
自定义类型:联合体和枚举
联合体类型,联合体的大小计算,联合体的实际应用,练习:判断大小端,枚举类型,枚举类型的声明:枚举类型的优点,枚举类型的实际案例:
|
1月前
|
存储 编译器 C语言
自定义类型(二)结构体位段,联合体,枚举
本文介绍了C++中结构体的默认对齐数修改,位段的声明和使用,联合体的概念及其实际应用,以及枚举类型的用途。通过实例展示了如何优化内存使用和提高代码可读性。
16 1
|
6月前
|
存储 编译器 Linux
自定义类型——结构体、联合体、枚举
自定义类型——结构体、联合体、枚举
|
6月前
|
编译器 测试技术 C语言
详解自定义类型:枚举与联合体!
详解自定义类型:枚举与联合体!
|
6月前
|
存储 开发框架 .NET
自定义类型:联合体和枚举类型(联合体与结构体的区别)
自定义类型:联合体和枚举类型(联合体与结构体的区别)
|
6月前
|
存储 编译器 C语言
自定义类型:结构体(自引用、内存对齐、位段(位域))
自定义类型:结构体(自引用、内存对齐、位段(位域))
|
编译器
自定义类型联合体(上)
自定义类型联合体
44 0
自定义类型联合体(上)
|
存储 编译器
自定义类型联合体(下)
自定义类型联合体
36 0
自定义类型联合体(下)
|
11月前
|
编译器 C++
结构体、枚举、位段、联合体详解
结构体、枚举、位段、联合体详解
69 0
|
11月前
|
存储 编译器 Linux
C语言自定义类型详解 位段+(联合体,枚举)
C语言自定义类型详解 位段+(联合体,枚举)