零基础学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

相关文章

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

C#实现获得某个枚举的所有名称

《C#实现获得某个枚举的所有名称》这篇文章主要为大家详细介绍了C#如何实现获得某个枚举的所有名称,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... C#中获得某个枚举的所有名称using System;using System.Collections.Generic;usi

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

Go语言使用Buffer实现高性能处理字节和字符

《Go语言使用Buffer实现高性能处理字节和字符》在Go中,bytes.Buffer是一个非常高效的类型,用于处理字节数据的读写操作,本文将详细介绍一下如何使用Buffer实现高性能处理字节和... 目录1. bytes.Buffer 的基本用法1.1. 创建和初始化 Buffer1.2. 使用 Writ

深入理解C语言的void*

《深入理解C语言的void*》本文主要介绍了C语言的void*,包括它的任意性、编译器对void*的类型检查以及需要显式类型转换的规则,具有一定的参考价值,感兴趣的可以了解一下... 目录一、void* 的类型任意性二、编译器对 void* 的类型检查三、需要显式类型转换占用的字节四、总结一、void* 的

MySQL中my.ini文件的基础配置和优化配置方式

《MySQL中my.ini文件的基础配置和优化配置方式》文章讨论了数据库异步同步的优化思路,包括三个主要方面:幂等性、时序和延迟,作者还分享了MySQL配置文件的优化经验,并鼓励读者提供支持... 目录mysql my.ini文件的配置和优化配置优化思路MySQL配置文件优化总结MySQL my.ini文件

Java中switch-case结构的使用方法举例详解

《Java中switch-case结构的使用方法举例详解》:本文主要介绍Java中switch-case结构使用的相关资料,switch-case结构是Java中处理多个分支条件的一种有效方式,它... 目录前言一、switch-case结构的基本语法二、使用示例三、注意事项四、总结前言对于Java初学者

Java 枚举的常用技巧汇总

《Java枚举的常用技巧汇总》在Java中,枚举类型是一种特殊的数据类型,允许定义一组固定的常量,默认情况下,toString方法返回枚举常量的名称,本文提供了一个完整的代码示例,展示了如何在Jav... 目录一、枚举的基本概念1. 什么是枚举?2. 基本枚举示例3. 枚举的优势二、枚举的高级用法1. 枚举

结构体和联合体的区别及说明

《结构体和联合体的区别及说明》文章主要介绍了C语言中的结构体和联合体,结构体是一种自定义的复合数据类型,可以包含多个成员,每个成员可以是不同的数据类型,联合体是一种特殊的数据结构,可以在内存中共享同一... 目录结构体和联合体的区别1. 结构体(Struct)2. 联合体(Union)3. 联合体与结构体的

Rust中的Option枚举快速入门教程

《Rust中的Option枚举快速入门教程》Rust中的Option枚举用于表示可能不存在的值,提供了多种方法来处理这些值,避免了空指针异常,文章介绍了Option的定义、常见方法、使用场景以及注意事... 目录引言Option介绍Option的常见方法Option使用场景场景一:函数返回可能不存在的值场景