【C语言】自定义类型——枚举和联合体

简介: 在C语言中,有三个自定义类型——结构体,枚举,联合,自定义类型出现是为了解决内置类型无法解决的问题。例如人这个对象,如果要描述人这样一个复杂对象,就不只是一个简简单单的int、char、double类型的数据能描述的,这时候就需要我们使用自定义类型来描述。

前言


在C语言中,有三个自定义类型——结构体,枚举,联合,自定义类型出现是为了解决内置类型无法解决的问题。例如人这个对象,如果要描述人这样一个复杂对象,就不只是一个简简单单的int、char、double类型的数据能描述的,这时候就需要我们使用自定义类型来描述。

(PS:内置类型是指任何语言在设计初期定义的类型,如c语言中的int, double, char… 它也是在一种语言中最基本的类型,与编译器编译出的代码具有重大关系,值得一提的是,不同语言也拥有不同的内置类型,但是所有内置类型的定义都与计算机的运算方式相关。)

对于结构体来说,我们应该都不陌生,结构体很重要,内容也很多,我之前写过一篇关于结构体的文章,不过那篇文章写的都是结构体的一些基础内容,过两天会写一篇进阶的,所以现在在就先不讲结构体了,我们先认识一下另外两个自定义类型。


枚举

枚举的定义


枚举类型的定义要使用enum关键字,举个例子,如果我要用枚举常量才定义一下三原色,也就是红蓝绿(RGB)。请看下面这段代码:

enum Color

{

//枚举常量

RED,

GREEN,

BLUE

};

Color就被定义成了枚举类型,{}中的内容是枚举类型的可能取值,也就是枚举常量。(补充:对于#define和吗枚举定义的常量 一般把变量名写成大写)。


枚举的使用


如果要用枚举类型创建变量就要使用enum Color,enum Color就代表着枚举类型,而创建好的变量的值就只能是{}中的内容。如下:

enum Color a = RED;

enum Color b = GREEN;

enum Color c = BLUE;

枚举常量是不能改变的。但是这些枚举常量都是有值的,默认从0开始,一次递增1,当然在定义的时候也可以赋初值。给大家解释一下,看下面这段代码:

printf("%d\n", RED);

printf("%d\n", GREEN);

printf("%d\n", BLUE);

23.png

我想看到这大家应该明白这是什么意思了,枚举常量是有值的,数值从0开始依次加1。如果要改变初始的值,只需要给第一个枚举常量赋值就行。例如:

enum Color

{

//枚举常量

RED = 2,

GREEN,

BLUE

};

像这样,再去打印RED,GREEN和BLUE,值就是2,3,4。


枚举的优点


对于枚举,我们可以使用 #define 定义常量,为什么非要使用枚举?

枚举的优点:

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

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

防止了命名污染(封装)

便于调试

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

联合(共用体)


在进行某些算法的C语言编程的时候,需要使几种不同类型的变量存放到同一段内存单元中。也就是使用覆盖技术,几个变量互相覆盖。这种几个不同的变量共同占用一段内存的结构,在C语言中,被称作“共用体”类型结构,简称共用体,也叫联合体。


联合体的定义


联合体的定义与结构体比较相似,要先声明联合体,要使用union这个关键字,声明方式如下:

union MyUnion

union MyUnion

{

int a;

char b;

};

联合体的定义如下:

//union 类型名 变量名

union MyUnion un;

这样就创建好了un这个联合体类型。

联合体的特点

先看这段代码以及运行结果:

union MyUnion

{

int a;//4

char b;//1

};

int main()

{

union MyUnion un;

printf("%d\n", sizeof(un));

printf("%u\n", &(un));

printf("%u\n", &(un.a));

printf("%u\n", &(un.b));

return 0;

}

24.png

我们可以看到un的大小是4不是5,而且un,un.a和un.b的地址是一样的。为什么?

下面我给大家一一解释:

我们在创建变量时,编译器就会给我们开辟一些空间,因此我创建un变量时,编译器就已经给我开辟空间了,那么我在打印un地址时,打印的就是un的首地址。在64位环境下,int是4个字节,char是一个字节,它们的地址和un是一样的,说明int是从un的首地址开始向后占用4个字节的空间,char也是从un的首地址开始的。因此打印出来的un.a和un.b的地址是相同的。它们公用同一部分的内存空间,这就是联合体。


联合体大小的计算


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

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

对于第一条,上面的例子证明过了,接下来重点来看第二条。看一下下面这段代码:

union un

{

char c[5];

int i;

};

int main()

{

printf("%d\n", sizeof(union un));

return 0;

}

对于这段代码,最后输出的结果是什么?5?

答案是8.

25.png

为什么?

对于un里面有一个长度为5的字符数组c和一个整型i。我们可以把它看成5个字符变量和一个整型变量。那么在计算联合体大小的时候,对齐数就是int类型,也就是4个字节,而不是5个字节。

因此我们计算联合体大小,考虑以多少字节为对齐数时,就只需要看数据类型就可以了,然后再看多少最大对齐数的整数倍能放下最大的变量就可以了。


总结:


自定义类型中枚举和联合体的难度不是很大,也就联合体大小的计算有点难度。大家只要记住计算联合体大小的两个特点就可以了。(感谢您的观看,如有错误,欢迎指正!感谢!)

相关文章
|
2月前
|
存储 编译器 C语言
【C语言篇】自定义类型:联合体和枚举详细介绍
像结构体⼀样,联合体也是由⼀个或者多个成员构成,这些成员可以不同的类型。
|
2月前
|
存储 安全 编译器
C语言自定义类型
C语言自定义类型
29 10
|
4月前
|
C语言
枚举(C语言)
枚举(C语言)
|
4月前
|
编译器 C语言
C语言枚举:深入探索下标默认值、自定义值及部分自定义情况
C语言枚举:深入探索下标默认值、自定义值及部分自定义情况
|
4月前
|
存储 编译器 C语言
C语言的联合体:一种节省内存的数据结构
C语言的联合体:一种节省内存的数据结构
|
4月前
|
编译器 C语言 C++
【海贼王编程冒险 - C语言海上篇】自定义类型:结构体,枚举,联合怎样定义?如何使用?
【海贼王编程冒险 - C语言海上篇】自定义类型:结构体,枚举,联合怎样定义?如何使用?
28 0
|
21天前
|
存储 Serverless C语言
【C语言基础考研向】11 gets函数与puts函数及str系列字符串操作函数
本文介绍了C语言中的`gets`和`puts`函数,`gets`用于从标准输入读取字符串直至换行符,并自动添加字符串结束标志`\0`。`puts`则用于向标准输出打印字符串并自动换行。此外,文章还详细讲解了`str`系列字符串操作函数,包括统计字符串长度的`strlen`、复制字符串的`strcpy`、比较字符串的`strcmp`以及拼接字符串的`strcat`。通过示例代码展示了这些函数的具体应用及注意事项。
|
24天前
|
存储 C语言
C语言程序设计核心详解 第十章:位运算和c语言文件操作详解_文件操作函数
本文详细介绍了C语言中的位运算和文件操作。位运算包括按位与、或、异或、取反、左移和右移等六种运算符及其复合赋值运算符,每种运算符的功能和应用场景都有具体说明。文件操作部分则涵盖了文件的概念、分类、文件类型指针、文件的打开与关闭、读写操作及当前读写位置的调整等内容,提供了丰富的示例帮助理解。通过对本文的学习,读者可以全面掌握C语言中的位运算和文件处理技术。
|
24天前
|
存储 C语言
C语言程序设计核心详解 第七章 函数和预编译命令
本章介绍C语言中的函数定义与使用,以及预编译命令。主要内容包括函数的定义格式、调用方式和示例分析。C程序结构分为`main()`单框架或多子函数框架。函数不能嵌套定义但可互相调用。变量具有类型、作用范围和存储类别三种属性,其中作用范围分为局部和全局。预编译命令包括文件包含和宏定义,宏定义分为无参和带参两种形式。此外,还介绍了变量的存储类别及其特点。通过实例详细解析了函数调用过程及宏定义的应用。
|
1月前
|
Linux C语言
C语言 多进程编程(三)信号处理方式和自定义处理函数
本文详细介绍了Linux系统中进程间通信的关键机制——信号。首先解释了信号作为一种异步通知机制的特点及其主要来源,接着列举了常见的信号类型及其定义。文章进一步探讨了信号的处理流程和Linux中处理信号的方式,包括忽略信号、捕捉信号以及执行默认操作。此外,通过具体示例演示了如何创建子进程并通过信号进行控制。最后,讲解了如何通过`signal`函数自定义信号处理函数,并提供了完整的示例代码,展示了父子进程之间通过信号进行通信的过程。