零基础学C语言——结构体、共同体、枚举

2023-12-26 03:45

本文主要是介绍零基础学C语言——结构体、共同体、枚举,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这是一个C语言系列文章,如果是初学者的话,建议先行阅读之前的文章。笔者也会按照章节顺序发布。

结构体

在数组一文中,我们提到过,数组是一种集合,这个集合中的元素都是同一种数据类型的。下面介绍一种用于整合不同数据类型的集合结构——结构体

结构体定义的一般形式

struct 结构体名 {数据类型 成员名1;数据类型 成员名2;...
};
或
struct 结构体名 {数据类型 成员名1;数据类型 成员名2;...
} 结构体变量/数组[数组长度];

这里:

  • 结构体名和成员名都符合变量的命名规范
  • 数据类型不仅涵盖了基础数据类型,还包含了自定义类型以及下面两小节介绍的共同体和枚举
  • 采用后一种形式定义结构体同时定义结构体变量或数组时,结构体名可以省略不写
  • 定义结构体同时定义变量/数组时也可以同时给变量/数组赋初值,参见下面结构体变量赋初值方式

来看一个例子:

struct person {char name[64];unsigned long age;
};

我们定义了一个名为person的结构体类型,这个结构体中包含了两个成员,一个是字符数组name,一个是无符号长整型age。这是一个程序模拟人类的简单例子。

上面我将类型二字做了突出,是因为这个结构体定义是一个类型。举个例子,我们称自己为人类,而人类并不是一个具体的人,而是泛指一类生物。而码哥我是一个具体的人,我属于人类这个称谓所定义的范畴。等价到结构体上,这个结构体person只是一个类型(人类),而由这个类型定义的变量才是一个具体的人。

既然,我们定义的是一个人,那么人也应该有性别。不考虑特殊群体的话,性别仅分为男和女。我们可以如下定义:

struct person {char name[64];unsigned long age;unsigned long sex;
};

我们可以给sex的值做个约定:0代表男,1代表女。

但是0和1其实仅仅需要一个比特就可以表达了,我们却使用了一个8字节(64比特)的变量来表示。这是一种浪费,是否可以优化呢?

答案是肯定的。我们将使用位域来进行优化:

struct person {char name[64];unsigned long age:63;unsigned long sex:1;
};

对比一下,差别在于成员名后加了冒号和一个整数值。这是什么含义呢?

unsigned long占8字节内存,即64比特。冒号后的值表示,这个变量占这个数据类型中的比特数。即age占unsigned long这64个比特中的63个,还剩余了一个比特,而sex占了unsigned long中的1个比特。这二者组合在一起正好64个比特,也就是一个unsigned long变量的大小。此时,编译器会将这两个成员放入一个8字节内存中,一个占63比特,一个占1比特。

位域的一般形式

数据类型 成员名1:所占比特数;

其中,如果所占比特数大于其数据类型自身所占比特数,则编译会报错。

既然类型已经定义好了,我们就可以用其定义变量了。

struct person Tom;
//或赋上初值
struct person Tom = {"Tom", 18, 0};

这里,我们定义了一个类型为struct person的变量Tom。

可以看到,变量的定义形式与常规的变量定义形式一样,并且对结构体的初始化是用大括号(非块语句)包裹住的。

有了变量后,我希望对其name、age、sex成员进行访问,例如对他们进行赋值,方法如下:

Tom.name[0] = 'T';
Tom.name[1] = 'o';
Tom.name[2] = 'm';
Tom.name[3] = '\0';
Tom.age = 18;
Tom.sex = 0;//0-male 1-female

对结构体变量中的成员访问是通过成员选择运算符(.)来完成的

现在还有一个问题,这个Tom变量占多大内存呢?

结构体所占内存大小等于其所有成员大小之和加上编译器额外填充的字节。以Tom为例,其成员大小之和为64字节(name)+8字节(age+sex),一共72字节。编译器填充的字节涉及结构体对齐规则,本文不做延展,初学者暂时以快速入门为主,优化进阶类内容可以自行查阅网上资料。

既然定义了变量,而变量又是存在于内存中,那么结构体变量就有其内存地址,因此也就有结构体指针这一概念。

struct person *ptr;
//或定义同时初始化
struct person *ptr = &Tom;

可以看到,结构体指针的定义与变量指针定义形式完全一样。

记得运算符一文中有两个成员选择运算符—— . 和 ->。

在结构体变量访问其成员时,使用的是.。而在结构体指针访问其成员时,使用的是->。例如:

ptr->age = 8; //将Tom的age改为了8

我们曾在函数一文中说过,函数参数都是值传递,因此不可能传递结构体这样的集合结构给函数,所以当需要传递结构体给函数参数时,传递的是结构体指针。

void output_person_information(struct person *p)
{printf("name: %s, age: %lu, sex: %lu\n", p->name, p->age, p->sex);
}

这里,printf的%s用于输出字符数组类型变量的内容,%lu用于输出无符号长整型变量的数值。

既然结构体是一种类型,那么可否定义这个类型的数组呢?

当然可以,例如:

struct person couple[2];
//或同时赋初值
struct person couple[2] = {{"Tom", 25, 0},{"Susan", 23, 1}
};

共同体

首先给出共同体(union)定义的一般形式

union 共同体名 {数据类型 成员名1;数据类型 成员名2;...
};

或定义共同体同时定义共同体变量

union 共同体名 {数据类型 成员名1;数据类型 成员名2;...
} 共同体变量/数组[数组长度];

其中:

  • 成员名符合变量的命名规则
  • 采用后一种形式定义共同体同时定义共同体变量或数组时,共同体名可以省略不写

共同体定义的也是一种类型定义

在我们日常生活中,每一个人都会有多重身份。例如,在公司我是员工,在家长面前我是子女,在孩子面前我是长辈。共同体就像将一个人的多重身份汇总在一起。沿用结构体一节中的person结构体定义,可以给出如下共同体代码:

union identity {struct person employee;struct person child;struct person parent;
};

下面利用这个共同体类型来定义共同体变量:

union identity my_identity;

共同体变量的大小是其定义包含的成员中占内存最大的成员的大小

即,这个my_identity占用的内存大小为72字节。

再例如:

union example {int i;double r;
};
union example ex;

此时,ex的大小为8字节,即double类型的大小。


共同体的成员的使用与结构体有很大不同。结构体变量内的每个成员都有其各自的内存单元,因此可以赋予不同的值。但是共同体则不同,共同体所有成员共用同一块存储空间,对每个成员的修改,会直接影响到其他成员的内容。继续前面identity的例子:

struct person couple[2] = {{"Tom", 25, 0},{"Susan", 23, 1}
};
my_identity.employee = couple[0];
output_person_information(&my_identity.employee);//输出的是Tom的信息
output_person_information(&my_identity.child);//输出的也是Tom的信息
my_identity.parent = couple[1]; //由于共用同一块内存,因此这次赋值将会影响employee和child的数据
output_person_information(&my_identity.employee);//输出的是Susan的信息
output_person_information(&my_identity.child);//输出的也是Susan的信息

枚举

枚举这个词的意思大致是:列出某些有穷序列集的所有成员。举个网上随处可见的例子:一周有7天,周一到周日。一周就是一个有穷序列集,而它的所有成员就是周一到周日。

在C语言中,枚举定义的一般形式为

enum 枚举类型名 {成员名1,成员名2,成员名3 =3,...
};

或定义枚举同时定义枚举变量

enum 枚举类型名 {成员名1,成员名2,成员名3 =3,...
} 枚举变量/数组[数组长度];

其中:

  • 成员名符合变量的命名规范
  • 采用后一种形式定义枚举同时定义枚举变量或数组时,枚举名可以省略不写
  • 成员名后可以给出值,也可以不给值,值是一个整数,值也可以是简单的表达式,如:成员名1+10
  • 在不给值的情况下,第一个成员的值为0,第二个为1,逐个加1,以此类推;遇到给定值,则该成员的值为该值,并且其后成员的值默认情况下基于该值累加1

例如:

enum week {Monday,Tuesday = 0,Wednesday,Thursday,Friday = Wednesday+99,Saturday,Sunday
};

这个例子中,每个成员的值如下:

  • Monday——0
  • Tuesday——0
  • Wednesday——1
  • Thursday——2
  • Friday——100
  • Saturday——101
  • Sunday——102

枚举类型的用处主要是:用成员名取代具体数值,有些类似常量,让代码更加直观,便于维护。

综合示例

#include <stdio.h>enum data_type {integer,real
};
union data {int i;float r;
};
struct input_value {enum data_type type;union data data;
};int main(void)
{struct input_value arr[2];arr[0].type = integer;arr[0].data.i = 10;arr[1].type = real, arr[1].data.r = 0.1;int i;for (i = 0; i < sizeof(arr)/sizeof(struct input_value); ++i) {if (arr[i].type == integer) {printf("integer %d\n", arr[i].data.i);} else {printf("real %f\n", arr[i].data.r);}}return 0;
}

这个例子中,定义了一个input_value结构体,这个结构体有两个成员,一个类型为枚举data_type的变量,一个类型为共同体data的变量。main中定义了一个结构体数组,先对数组内成员进行初始化。然后利用for和if遍历并输出数组元素的data成员数据。



喜欢的小伙伴可以关注码哥,也可以给码哥留言评论,如有建议或者意见也欢迎私信码哥,我会第一时间回复。
感谢阅读!

这篇关于零基础学C语言——结构体、共同体、枚举的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/537989

相关文章

Oracle查询表结构建表语句索引等方式

《Oracle查询表结构建表语句索引等方式》使用USER_TAB_COLUMNS查询表结构可避免系统隐藏字段(如LISTUSER的CLOB与VARCHAR2同名字段),这些字段可能为dbms_lob.... 目录oracle查询表结构建表语句索引1.用“USER_TAB_COLUMNS”查询表结构2.用“a

python panda库从基础到高级操作分析

《pythonpanda库从基础到高级操作分析》本文介绍了Pandas库的核心功能,包括处理结构化数据的Series和DataFrame数据结构,数据读取、清洗、分组聚合、合并、时间序列分析及大数据... 目录1. Pandas 概述2. 基本操作:数据读取与查看3. 索引操作:精准定位数据4. Group

C语言中%zu的用法解读

《C语言中%zu的用法解读》size_t是无符号整数类型,用于表示对象大小或内存操作结果,%zu是C99标准中专为size_t设计的printf占位符,避免因类型不匹配导致错误,使用%u或%d可能引发... 目录size_t 类型与 %zu 占位符%zu 的用途替代占位符的风险兼容性说明其他相关占位符验证示

C语言进阶(预处理命令详解)

《C语言进阶(预处理命令详解)》文章讲解了宏定义规范、头文件包含方式及条件编译应用,强调带参宏需加括号避免计算错误,头文件应声明函数原型以便主函数调用,条件编译通过宏定义控制代码编译,适用于测试与模块... 目录1.宏定义1.1不带参宏1.2带参宏2.头文件的包含2.1头文件中的内容2.2工程结构3.条件编

Go语言并发之通知退出机制的实现

《Go语言并发之通知退出机制的实现》本文主要介绍了Go语言并发之通知退出机制的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、通知退出机制1.1 进程/main函数退出1.2 通过channel退出1.3 通过cont

Go语言编译环境设置教程

《Go语言编译环境设置教程》Go语言支持高并发(goroutine)、自动垃圾回收,编译为跨平台二进制文件,云原生兼容且社区活跃,开发便捷,内置测试与vet工具辅助检测错误,依赖模块化管理,提升开发效... 目录Go语言优势下载 Go  配置编译环境配置 GOPROXYIDE 设置(VS Code)一些基本

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

Go语言中make和new的区别及说明

《Go语言中make和new的区别及说明》:本文主要介绍Go语言中make和new的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 概述2 new 函数2.1 功能2.2 语法2.3 初始化案例3 make 函数3.1 功能3.2 语法3.3 初始化

Go语言中nil判断的注意事项(最新推荐)

《Go语言中nil判断的注意事项(最新推荐)》本文给大家介绍Go语言中nil判断的注意事项,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.接口变量的特殊行为2.nil的合法类型3.nil值的实用行为4.自定义类型与nil5.反射判断nil6.函数返回的

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁