C语言学习——结构体_联合体_枚举用法直通[通俗易懂]

C语言学习——结构体_联合体_枚举用法直通[通俗易懂]一、结构体嵌套#include <stdio.h>// 出生日期:年月日struct sBirthday{ int iYear; int iMonth; int iDay;};// 公民信息struct sSCitizenIDInfo{ char cName[10]; char cSex[4]; sBirthday BirthDate; // 结构体变量…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

一、结构体嵌套

#include <stdio.h>
// 出生日期:年月日
struct sBirthday
{ 
   
	int iYear;
	int iMonth;
	int iDay;
};
// 公民信息
struct sSCitizenIDInfo
{ 
   
	char cName[10];
	char cSex[4];
	sBirthday  BirthDate;  // 结构体变量作为另一个结构体的成员
	char cAddress[100];
};
int main()
{ 
   
	struct sSCitizenIDInfo sInfo = { 
    "王中华", "女", { 
    2010, 12, 12 }, "北京市朝阳区" };
	struct sSCitizenIDInfo *pCID = &sInfo;
	printf("姓名:%s\n性别:%s\n出生日期:%d 年 %d 月%d 日\n地址:%s\n",pCID->cName,pCID->cSex,pCID->BirthDate.iYear,pCID->BirthDate.iMonth,pCID->BirthDate.iDay,pCID->cAddress);
	return 0;
}

结果为;
在这里插入图片描述

二、结构体数组

所谓结构体数组,是指数组中的每个元素都是一个结构体。在实际应用中,C语言结构体数组常被用来表示一个拥有相同数据结构的群体,比如一个班的学生、一个车间的职工等。

// 公民信息
struct sSCitizenIDInfo
{ 
   
	char cName[10];
	char cSex[4];
	sBirthday  BirthDate;
	char cAddress[100];
}pCArr[3] = { 
   
		{ 
    "王中华", "男", { 
    2010, 12, 12 }, "北京市朝阳区" },
		{ 
    "刘小小", "女", { 
    1990, 2, 22 }, "北京市东城区" },
		{ 
    "吴小莉", "女", { 
    2000, 9, 30 }, "北京市海淀区" }
};

#include <stdio.h>
// 出生日期:年月日
struct sBirthday{ 
   
	int iYear;
	int iMonth;
	int iDay;
};
// 公民信息
struct sSCitizenIDInfo{ 
   
	char cName[10];
	char cSex[4];
	sBirthday  BirthDate;
	char cAddress[100];
}pCArr[3] = { 
   	{ 
    "王中华", "男", { 
    2010, 12, 12 }, "北京市朝阳区" },
		{ 
    "刘小小", "女", { 
    1990, 2, 22 }, "北京市东城区" },
		{ 
    "吴小莉", "女", { 
    2000, 9, 30 }, "北京市海淀区" }
};

int main()
{ 
   
	struct sSCitizenIDInfo *pCID = pCArr;

	// 数组下标输出
	for (int i = 0; i < 3; i++)
	{ 
   
		printf("\n姓名:%s\n性别:%s\n出生日期:%d年%d月%d日\n地址:%s\n", pCArr[i].cName, pCArr[i].cSex, pCArr[i].BirthDate.iYear, pCArr[i].BirthDate.iMonth, pCArr[i].BirthDate.iDay, pCArr[i].cAddress);
	}
		// 结构指针变量输出
	printf("\n-----------------------------------------------------------------------------------\n");
	for (int i = 0; i < 3; i++)
	{ 
   
		printf("\n姓名:%s\n性别:%s\n出生日期:%d年%d月%d日\n地址:%s\n", (pCID[i]).cName, (pCID[i]).cSex, (pCID[i]).BirthDate.iYear, (pCID[i]).BirthDate.iMonth, (pCID[i]).BirthDate.iDay, (pCID[i]).cAddress);
	}

	return 0;
}

结果如下:
在这里插入图片描述
在这里插入图片描述

三、联合体

结构体(Struct):
是一种构造类型或复杂类型,可以包含多个类型不同的成员。
在C语言中,还有另外一种和结构体非常类似的语法,叫做共用体(union),它的定义格式为:

union 共用体名
{ 
   
    成员列表......
};

共用体有时也被称为联合或者联合体;

结构体和共用体的区别在于:结构体的各个成员会占用不同的内存,互相之间没有影响;而共用体的所有成员占用同一段内存,修改一个成员会影响其余所有成员。

结构体占用的内存大于等于所有成员占用的内存的总和(成员之间可能会存在缝隙),共用体占用的内存等于最长的成员占用的内存。共用体使用了内存覆盖技术,同一时刻只能保存一个成员的值,如果对新的成员赋值,就会把原来成员的值覆盖掉。
1、共用体也是一种自定义类型,可以通过它来创建变量:

union data
{ 
   
	int i;
	char ch;
	double d;
};

union data ud;

2、定义共用体的同时创建变量

union data
{ 
   
	int i;
	char ch;
	double d;
}ud;

3、如果不再定义新的变量,也可以将共用体的名字省略:

union
{ 
   
	int i;
	char ch;
	double d;
}ud,cd;

四、枚举

在程序中,可能需要为某些整数定义一个别名,我们可以利用预处理指令#define来完成本项工作:

如这样:

#define MONDAY 1
#define TUESDAY 2
#define WEDNESDAY 3
#define THURSDAY 4
#define FRIDAY 5
#define SATURDAY 6
#define SUNDAY 7

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
方法一:枚举类型的定义和变量的声明分开

enum DAY
{ 
   
      MON=1, TUE, WED, THU, FRI, SAT, SUN
};
 

enum DAY yesterday;
enum DAY today;
enum DAY tomorrow; //变量tomorrow的类型为枚举型enum DAY
enum DAY good_day, bad_day; //变量good_day和bad_day的类型均为枚举型enum DAY

方法二:类型定义与变量声明同时进行:

enum //跟第一个定义不同的是,此处的标号DAY省略,这是允许的。
{ 
   
    saturday,
    sunday = 0,
    monday,
    tuesday,
    wednesday,
    thursday,
    friday
} workday; //变量workday的类型为枚举型enum DAY
enum week { 
    Mon=1, Tue, Wed, Thu, Fri Sat, Sun} days; //变量days的类型为枚举型enum week 

enum BOOLEAN { 
    false, true } end_flag, match_flag; //定义枚举类型并声明了两个枚举型变量

方法三:用typedef关键字将枚举类型定义成别名,并利用该别名进行变量声明:

typedef enum workday
{ 
   
    saturday,
    sunday = 0,
    monday,
    tuesday,
    wednesday,
    thursday,
    friday
} workday; //此处的workday为枚举型enum workday的别名

workday today, tomorrow; //变量today和tomorrow的类型为枚举型workday,也即enum workday

五、C语言字节对齐

一、为什么要字节对齐  
   需要字节对齐的根本原因在于CPU访问数据的效率问题。

二、正确处理字节对齐  
   对于标准数据类型,它的地址只要是它的长度的整数倍就行了,而非标准数据类型按下面的原则对齐:  
   数 组 :按照基本数据类型对齐,第一个对齐了后面的自然也就对齐了。
   联 合 :按其包含的长度最大的数据类型对齐。
   结构体: 结构体中每个数据类型都要对齐。

三、什么时候需要设置对齐
在设计不同CPU下的通信协议时,或者编写硬件驱动程序时寄存器的结构这两个地方都需要按一字节对齐。即使看起来本来就自然对齐的也要使其对齐,以免不同的编译器生成的代码不一样.

案例1

#pragma pack (4)
#include <stdio.h>
struct stu
{ 
   
	char sex;
	int length;
	char name[10];
};
struct stu my_stu;

int main()
{ 
   
	printf("占据:%d字节\n", sizeof(my_stu));
	
	return 0;
}

结果:
在这里插入图片描述
案例分析:
上面定义了对其字节数为4字节,char本身1字节,在这占四位,int四位,char[10]占10位,但最后两位直接算四位,4+4+12=20字节

案例2:


#include <stdio.h>
#pragma pack(1) //让编译器对这个结构作1字节对齐
struct test
{ 
   
	char x1;
	short x2;
	float x3;
	char x4;
	int a[3];
	double d;
}TT;
//#pragma pack() //取消1字节对齐,恢复为默认4字节对齐
int main()
{ 
   
	printf("char占据:%d字节\n", sizeof(char));
	printf("short占据:%d字节\n", sizeof(short));
	printf("float占据:%d字节\n", sizeof(float));
	printf("char占据:%d字节\n", sizeof(char));
	printf("a占据:%d字节\n", sizeof(TT.a));
	printf("d占据:%d字节\n", sizeof(TT.d));

	printf("\n共计占据:%d字节\n", sizeof(TT));
	
	return 0;
}

结果:
在这里插入图片描述

六、typedef

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/170986.html原文链接:https://javaforall.net

(0)
上一篇 2022年8月18日 上午8:46
下一篇 2022年8月18日 上午9:00


相关推荐

  • 结合Bootstrap实现头像上传前预览

    结合Bootstrap实现头像上传前预览Html lt div gt lt divstyle position relative border 5pxsolidligh width 150px height 150px margin 5pxauto border radius 50 gt lt imgsrc static images default jpg alt 选择

    2026年3月18日
    2
  • window.location.href和window.location.hash的区别

    window.location.href和window.location.hash的区别综:window.location.href表示重定向,后面跟着的是完整的url地址,与其相似的还有window.location.hash,下面来比较window.location.href和window.locaiton.hash的区别。    (1)window.location.href        得到和使用的是完整的url,比如window.

    2022年7月16日
    23
  • 设计模式23模式介绍迭代器_二十三种设计模式

    设计模式23模式介绍迭代器_二十三种设计模式一、什么是设计模式设计模式是一套被反复使用、多数人知晓的、经过分类编写的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性。毫无疑问,设计模式使代码编程真正工程化,设计模式是软件工程的基石,如同大厦的一块块砖头一样。项目中合理的运用设计模式可以完美的解决很多问题,每种模式都有响应的原理与之对应,每一个模式描述了一个在我们周围不断重复发生的问题,以及该问题…

    2025年7月30日
    4
  • 碰撞圆周率_我让你背一遍圆周率

    碰撞圆周率_我让你背一遍圆周率#include”stdafx.h”#include<iostream>//碰撞后速度voidtxpz(doublem1,doublem2,double&v1,double&v2){ doublev1_=((m1-m2)*v1+2*m2*v2)/(m1+m2); doublev2_=((m2…

    2025年8月29日
    8
  • vue:详解vue中的代理proxy

    vue:详解vue中的代理proxy问题我们本地调试一般都是 npmrunserve 然后打开本机 ip 8080 localhost 8080 对吧 这时候我们要调接口调试 后端的接口的地址可能在测试环境 也可能是自己电脑的 ip 总之不是你的 lcoalhost 8080 那么你调接口就会产生跨域 那么怎么办呢 就需要 proxy 出场了复习一下跨域的解决方案 jsonpcorsNod 中间件代理 两次跨域 nginx 反向代理 CORS 支持所有类型的 HTTP 请求 是跨域 HTTP 请求的根本解决方案 JSONP 只支持 GET 请求

    2025年7月29日
    6
  • Apache配置虚拟主机_apache中配置虚拟主机的作用

    Apache配置虚拟主机_apache中配置虚拟主机的作用修改Apache的虚拟主机配置文件httpd-vhosts.conf,在文件的最后加上如下的代码:NameVirtualHost *:80    ServerNamea.com    DocumentRoot “E:\xampp\htdocs\a”    ServerName b.com    DocumentRoot “E:\xampp\

    2026年1月15日
    5

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号