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

相关文章

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

hdu 2489 (dfs枚举 + prim)

题意: 对于一棵顶点和边都有权值的树,使用下面的等式来计算Ratio 给定一个n 个顶点的完全图及它所有顶点和边的权值,找到一个该图含有m 个顶点的子图,并且让这个子图的Ratio 值在所有m 个顶点的树中最小。 解析: 因为数据量不大,先用dfs枚举搭配出m个子节点,算出点和,然后套个prim算出边和,每次比较大小即可。 dfs没有写好,A的老泪纵横。 错在把index在d

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

自定义类型:结构体(续)

目录 一. 结构体的内存对齐 1.1 为什么存在内存对齐? 1.2 修改默认对齐数 二. 结构体传参 三. 结构体实现位段 一. 结构体的内存对齐 在前面的文章里我们已经讲过一部分的内存对齐的知识,并举出了两个例子,我们再举出两个例子继续说明: struct S3{double a;int b;char c;};int mian(){printf("%zd\n",s

AI基础 L9 Local Search II 局部搜索

Local Beam search 对于当前的所有k个状态,生成它们的所有可能后继状态。 检查生成的后继状态中是否有任何状态是解决方案。 如果所有后继状态都不是解决方案,则从所有后继状态中选择k个最佳状态。 当达到预设的迭代次数或满足某个终止条件时,算法停止。 — Choose k successors randomly, biased towards good ones — Close

C语言 | Leetcode C语言题解之第393题UTF-8编码验证

题目: 题解: static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num & MASK1) == 0) {return

hdu 6198 dfs枚举找规律+矩阵乘法

number number number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Problem Description We define a sequence  F : ⋅   F0=0,F1=1 ; ⋅   Fn=Fn