C语言之结构体

news2025/2/28 20:16:17

一.前言引入.

我们知道在C语言中有内置类型,如:整型,浮点型等。但是只有这些内置类 型还是不够的,假设我想描述学⽣,描述⼀本书,这时单⼀的内置类型是不⾏的。描述⼀个学⽣需要名字、年龄、学号、⾝⾼、体重等;描述⼀本书需要作者、出版社、定价等。C语⾔为了解决这个问 题,增加了结构体,共用体等⾃定义的数据类型,让程序员可以⾃⼰创造适合的类型。今天我们来讲讲结构体。

二.结构体介绍.

定义:结构体是⼀些值的集合,这些值称为成员变量。结构的每个成员可以是不同类型的变量,如: 标量、数组、指针,甚⾄是其他结构体

结构体的声明:

struct tag
{
 member-list;
}variable-list;

例如:描述一个学生

struct Student
{
	char name[20];//名字
	int age;//年龄
	char sex[5];//性别
	char id[15];//学号
}s1;//s1为全局变量
int main()
{
	struct Student s2;//局部变量
	return 0;
}

结构体特殊声明

在声明结构的时候,可以不完全的声明
//在声明结构的时候,可以不完全的声明,即匿名结构体类型
struct
{
	int a;
	char b;
	float c;
}s1;

请判断下面代码是否正确:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
//在声明结构的时候,可以不完全的声明,即匿名结构体类型
struct
{
	int a;
	char b;
	float c;
}s1;
struct
{
	int a;
	char b;
	float c;
}s2[20], * p;
int main()
{
	p = &s1;
	return 0;
}
警告:
编译器会把上⾯的两个声明当成完全不同的两个类型,所以是⾮法的。
匿名的结构体类型,如果没有对结构体类型重命名的话,基本上只能使⽤⼀次

结构体变量的定义:

struct Point
{
	int x; int y;
}s1;//结构体变量的定义
struct Point s2;//结构体变量的定义
int main()
{
	struct Point s3;//结构体变量的定义
	return 0;
}

结构体变量的初始化:

struct Stu
{
	char name[20];
	int age;
};
int main()
{
	struct Stu s1 = { "zhangsan",18 };///结构体变量初始化	
	return 0;
}

 镶嵌初始化:

struct Point
{
	int x;
	int y;
};
struct Stu
{
	char name[20];
	int age;
	struct Point s2;
};
int main()
{
	struct Stu s1 = { "zhangsan",18,{2,3} };///结构体变量镶嵌初始化	
	return 0;
}

三.结构成员访问操作符.

3.1.结构体成员的直接访问

使⽤⽅式:结构体变量.成员名

结构体成员的直接访问是通过点操作符(.)访问的。点操作符接受两个操作数.
struct Point
{
	int x;
	int y;
}s1 = {3,5};
int main()
{
	struct Point s2 = { 2,4 };
	printf("%d %d\n", s1.x, s1.y);
	printf("%d %d\n", s2.x, s2.y);
	return 0;
}

3.2.结构体成员的间接访问

使⽤⽅式:结构体指针->成员名

有时候我们得到的不是⼀个结构体变量,⽽是得到了⼀个指向结构体的指针.

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
struct Point
{
	int x;
	int y;
}s1 = { 3,5 };
int main()
{
	struct Point s2 = { 2,3 };
	struct Point* pi1 = &s1;
	struct Point* pi2 = &s2;
	printf("%d %d\n", pi1->x, pi1->y);
	printf("%d %d\n", pi2->x, pi2->y);
	pi1->x = 10;
	pi1->y = 20;
	printf("%d %d\n", pi1->x, pi1->y);
	return 0;
}

综合使用结构体用例:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
struct Stu
{
	char name[10];
	int age;
};
void Print(struct Stu s2)
{
	printf("%s %d\n", s2.name, s2.age);
}
void Set(struct Stu* s3)
{
	s3->age = 15;
	strcpy(s3->name, "李四");
}
int main()
{
	struct Stu s1 = { "zhangsan",18 };
	Print(s1);
	Set(&s1);
	Print(s1);
	return 0;
}

结果:

四.结构体的⾃引⽤.

五.结构体的内存对齐.

计算结构体的⼤⼩

例如:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
struct s1
{
	char a;
	int b;
	char c;
};
struct s2
{
	char a;
	char b;
	int c;
};
int main()
{
	printf("%zd\n", sizeof(struct s1));
	printf("%zd\n", sizeof(struct s2));

	return 0;
}

如果我给你一份这样的代码,请问你认为结果如何呢?两者一样吗?

答案如下:

和你想的一样吗?

如果不一样,不妨来看看我的解读:

结构体的对⻬规则:
1. 结构体的第⼀个成员对⻬到和结构体变量起始位置偏移量为0的地址处
2. 其他成员变量要对⻬到某个数字(对⻬数)的整数倍的地址处。 对⻬数 = 编译器默认的⼀个对⻬数与该成员变量⼤⼩的较⼩值。  VS 中默认的值为 8 ,Linux中 gcc 没有默认对⻬数,对⻬数就是成员⾃⾝的⼤⼩
3. 结构体总⼤⼩为最⼤对⻬数(结构体中每个成员变量都有⼀个对⻬数,所有对⻬数中最⼤的)的
整数倍
4. 如果嵌套了结构体的情况,嵌套的结构体成员对⻬到⾃⼰的成员中最⼤对⻬数的整数倍处,结构
体的整体⼤⼩就是所有最⼤对⻬数(含嵌套结构体中成员的对⻬数)的整数倍

例题1:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
struct s1
{
	//1. 结构体的第⼀个成员对⻬到和结构体变量起始位置偏移量为0的地址处
	char a;//对齐偏移量为0位置处,char一个字节
	//其他成员变量要对⻬到某个数字(对⻬数)的整数倍的地址处。
	//对⻬数 = 编译器默认的⼀个对⻬数与该成员变量⼤⼩的较⼩值。  
	//对齐数VS 中默认的值为 8 
	char b;//对齐偏移量为1位置处,char一个字节,即 1 8 1(第一个为成员变量的对齐数,第二个为VS对齐数,第三个为该行对齐数)
	int c;//对齐偏移量为4位置处,int四个字节,即 4 8 4
};	
int main()
{
	//结构体总⼤⼩为最⼤对⻬数(结构体中每个成员变量都有⼀个对⻬数,所有对⻬数中最⼤的)的整数倍,即1 1 4 8 ->8
	printf("%zd\n", sizeof(struct s1));//结果为8
	return 0;
}

结果为:

例题二:

struct s2
{
	//1. 结构体的第⼀个成员对⻬到和结构体变量起始位置偏移量为0的地址处
	char a;//对齐偏移量为0位置处,char一个字节
	//其他成员变量要对⻬到某个数字(对⻬数)的整数倍的地址处。
	//对⻬数 = 编译器默认的⼀个对⻬数与该成员变量⼤⼩的较⼩值。  
	//对齐数VS 中默认的值为 8 
	int b;//对齐偏移量为4位置处,四个字节,int: 4 8 4(第一个为成员变量的对齐数,第二个为VS对齐数,第三个为该行对齐数)
	char c;//对齐偏移量为9位置处,char 1个字节;1 8 1
	
};
int main()
{
	//结构体总⼤⼩为最⼤对⻬数(结构体中每个成员变量都有⼀个对⻬数,所有对⻬数中最⼤的)的整数倍,即1 4 1  ->4的整数倍为12(原来的数据已经达到偏移量为9了)
	printf("%zd\n", sizeof(struct s2));//结果为12
	return 0;
}

结果:

例题三:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
struct s3
{
	//下面简写
	//前三个为成员大小 VS大小 该行大小,偏移量的位置
	double d;//8 8 8,0-7
	char c;//1 8 1,8
	int i;//4 8 4.12-15
};
int main()
{
	printf("%zd\n", sizeof(struct s3));//结果8 1 4->8 15->16
	return 0;
}

结果:

例题四:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
练习4-结构体嵌套问题
struct s3
{
	//下面简写
	//前三个为成员大小 VS大小 该行大小,偏移量的位置
	double d;
	char c;
	int i;
};
struct s4
{
	char c1;//1 8 1,0
	//4. 如果嵌套了结构体的情况,嵌套的结构体成员对⻬到⾃⼰的成员中最⼤对⻬数的整数倍处,结构体的整体⼤⼩就是所有最⼤对⻬数(含嵌套结构体中成员的对⻬数)的整数倍
	struct s3 a;//16 8 8,8-23
	double d;//8 8 8,24-31
};
int main()
{
	printf("%zd\n", sizeof(struct s4));//1 16 8->16 31->32
	return 0;
}

你是否想过这样一个问题,为什么存在内存对⻬呢?
平台原因 (移植原因):
不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。
性能原因:
数据结构(尤其是栈)应该尽可能地在⾃然边界上对⻬。原因在于,为了访问未对⻬的内存,处理器需要作两次内存访问;⽽对⻬的内存访问仅需要⼀次访问。假设⼀个处理器总是从内存中取8个字节,则地址必须是8的倍数。如果我们能保证将所有的double类型的数据的地址都对⻬成8的倍数,那么就可以⽤⼀个内存操作来读或者写值了。否则,我们可能需要执⾏两次内存访问,因为对象可能被分放在两个8字节内存块中。
总体来说:结构体的内存对⻬是拿空间来换取时间的做法。
那在设计结构体的时候,我们既要满⾜对⻬,⼜要节省空间,
重点:如何做到: 让占⽤空间⼩的成员尽量集中在⼀起!!!
重点:如何做到: 让占⽤空间⼩的成员尽量集中在⼀起!!!
重点:如何做到: 让占⽤空间⼩的成员尽量集中在⼀起!!!

说三遍!

相信你现在一定会开头的那题了!

那么我们可以自己设置对齐数吗?答案是当然可以啦!

#pragma 这个预处理指令,可以改变编译器的默认对⻬数
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#pragma pack(1)
练习4-结构体嵌套问题
struct s3
{
	//下面简写
	//前三个为成员大小 VS大小 该行大小,偏移量的位置
	double d;
	char c;
	int i;
};
struct s4
{
	char c1;//1 8 1,0
	//4. 如果嵌套了结构体的情况,嵌套的结构体成员对⻬到⾃⼰的成员中最⼤对⻬数的整数倍处,结构体的整体⼤⼩就是所有最⼤对⻬数(含嵌套结构体中成员的对⻬数)的整数倍
	struct s3 a;//16 8 8,8-23
	double d;//8 8 8,24-31
};
int main()
{
	printf("%zd\n", sizeof(struct s4));//1 16 8->16 31->32
	return 0;
}

修改之后结果为:

#pragma pack()//取消设置的对⻬数,还原为默认

这样,当结构体在对⻬⽅式不合适的时候,我们可以⾃⼰更改默认对齐数。

六.结构体传参和对比

1.传数据

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
struct S
{
	int data[1000];
	int num;
};
void Print(struct S s1)
{
	printf("%d\n", s1.num);
}
int main()
{
	struct S s1 = { {1,2,3,4}, 1000 };
	Print(s1);
	return 0;
}

结果:

2.传地址

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
struct S
{
	int data[1000];
	int num;
};
void Print(struct S* s1)
{
	printf("%d\n", s1->num);
}
int main()
{
	struct S s1 = { {1,2,3,4}, 1000 };
	Print(&s1);
	return 0;
}

结果:

对比两者,你认为哪个好呢?

函数传参的时候,参数是需要压栈,会有时间和空间上的系统开销。
如果传递⼀个结构体对象的时候,结构体过⼤,参数压栈的的系统开销⽐较⼤,所以会导致性能的下降。
结论:
结构体传参的时候,要传结构体的地址。

最后,学习进步!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1279259.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

vue $nextTick 样式私有化

$nextTick 先updated中更新&#xff0c;再nextTick 状态更改做什么事情&#xff1a; updated $nextTick 同步执行完之后&#xff0c;把当前放到队列中 $forceUpdate->sub.update() // 把更新操作放在队列里面 队列机制 基于发布订阅模式&#xff0c;callbacks队列 更新完毕…

Java类的初始化顺序

类初始化顺序遵循以下三个原则&#xff08;优先级依次递减&#xff09; 1、静态对象&#xff08;变量&#xff09;优先于非静态对象&#xff08;变量&#xff09;初始化&#xff0c;其中静态对象&#xff08;变量&#xff09;只初始化一次&#xff0c;而非静态对象&…

揭秘原型链:探索 JavaScript 面向对象编程的核心(上)

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

[UnityWebGL]修改webgl启动模板

首先了解什么是WEBGGL启动模板&#xff0c;如下图&#xff1a; 其次&#xff0c;启动模板在哪里设置&#xff0c;如下图&#xff1a; 最后&#xff0c;怎么改启动模板里面的内容&#xff08;提供最简单的方式&#xff0c;直接改官方的&#xff09;&#xff1a; 1、首先在Asset…

Spring Cloud Gateway与spring-cloud-circuitbreaker集成与理解

官方文档 spring-cloud-gateway文档地址&#xff1a; https://spring.io/projects/spring-cloud-gateway#overview spring-cloud-circuitbreaker文档地址&#xff1a; https://spring.io/projects/spring-cloud-circuitbreaker 两者关系 首先spring-cloud-gateway集成了断路…

[c++]——string类____详细初步了解string类的运用

在成为大人的路上喘口气. 目录 &#x1f393;标准库类型string &#x1f393;定义和初始化string对象 &#x1f4bb;string类对象的常见构造 &#x1f4bb;string类对象的不常见构造 &#x1f4bb;读写string对象 &#x1f393; string类对象的修改操作 &#x1f4…

Opencv制作电子签名(涉及知识点:像素过滤,图片通用resize函数,像素大于某个阈值则赋值为其它的像素值)

import cv2def resize_by_ratio(image, widthNone, heightNone, intercv2.INTER_AREA):img_new_size None(h, w) image.shape[:2] # 获得高度和宽度if width is None and height is None: # 如果输入的宽度和高度都为空return image # 直接返回原图if width is None:h_ratio …

git-6

1.如何用project管理issue&#xff1f; 用project看板管理issue 有五种类型&#xff1a;None、Basic kanban、Automated kanban、Automated kanban with reviews、Bug triage 首先选用Bug triage 利用看板就会很直观&#xff0c;很便捷&#xff0c;Issues也支持&#xff0c;有…

OpenTelemetry系列 - 第4篇 OpenTelemetry K8S生态

目录 一、【Helm】添加OTel Helm repo二、【Helm Chart】OTel Collector2.1 daemonset2.2 deloyment 三、【K8S Operator】OTel Operator3.1 安装OTel Operator3.2 部署OpenTelemetryCollector3.2.1 Deloyment Mode3.2.2 DeamonSet Mode3.2.3 StatefulSetMode3.2.4 Sidecar Mod…

思维模型 莫扎特效应

本系列文章 主要是 分享 思维模型&#xff0c;涉及各个领域&#xff0c;重在提升认知。音乐激发无限潜能。 1 莫扎特效应的应用 1.1 莫扎特效应在教育领域的应用-Baby Einstein”公司 在美国&#xff0c;有一家名为“Baby Einstein”的公司&#xff0c;该公司生产和销售专门为…

SQL自学通之简介

目录 一、SQL 简史 二、数据库简史 1、Dr. Codds 对关系型数据库系统的十二条规则 2、设计数据库的结构 3、数据库的前景 4、对于什么是客户机/服务器型电脑系统 BernardH.Boar的定义如下&#xff1a; 5、交互式语言 6、易于实现 7、SQL 总览 三、流行的 SQL 开发工具…

前端小记--2.element-ui中级联选择器cascader如何默认展开下拉框

最近做项目时&#xff0c;遇到一个需求&#xff1a;在一个排班表中&#xff0c;展示人员的值班情况&#xff0c;点击单元格&#xff0c;弹出下拉框&#xff0c;修改人员排班信息。 由于下拉框选择内容是树状结构&#xff0c;这里使用了element-ui中级联组件cascader&#xff0c…

一起学docker系列之十七Docker Compose 与手动操作的比较与优势分析

目录 1 前言2 不使用 Docker Compose2.1 启动 MySQL 容器2.2 启动 Redis 容器2.3 启动微服务容器 3 使用 Docker Compose4 使用 Docker Compose 的优势5 结语参考地址 1 前言 在当今容器化应用的开发与部署中&#xff0c;容器编排工具的选择对于简化流程、提高效率至关重要。本…

6-69.鸭子也是鸟

按要求完成下面的程序&#xff1a; 1、定义一个Bird类&#xff0c;包含一个void类型的无参的speak方法&#xff0c;输出“Jiu-Jiu-Jiu”。 2、定义一个Duck类&#xff0c;公有继承自Bird类&#xff0c;其成员包括&#xff1a; &#xff08;1&#xff09;私有string类型的成员na…

WPF实现文字纵向排布的TabItem

文章目录 基本用法文字竖排显示 WPF布局 基本用法 WPF中的TabControl是一个容器控件&#xff0c;用于在单个窗体或页面中承载多个选项卡。每个选项卡可以包含不同的控件&#xff0c;用于显示不同的内容&#xff0c;其最简单的调用方法如下&#xff0c;只需在TabControl中无脑…

德迅云安全的日常网站安全性措施、以及更多网站安全工具的推荐与使用。

要确保网站的安全性&#xff0c;可以采取以下措施&#xff1a; 更新和维护&#xff1a;定期更新网站的操作系统、应用程序和插件&#xff0c;确保使用的是最新版本&#xff0c;以修复已知的安全漏洞。 强密码策略&#xff1a;使用强密码&#xff0c;包含字母、数字和特殊字符的…

合成相机模型【图形学】

相机在计算机图形学中有两个方面的考虑&#xff1a;相机的位置和相机的形状。 要了解后者&#xff0c;我们需要了解相机的工作原理。 NSDT工具推荐&#xff1a; Three.js AI纹理开发包 - YOLO合成数据生成器 - GLTF/GLB在线编辑 - 3D模型格式在线转换 - 可编程3D场景编辑器 - R…

BearPi Std 板从入门到放弃 - 1 引气入体篇

安装相关开发工具 Keil MDK 工具下载 略, 自行体会 Keil 芯片支持包下载 Keil 包 网址 https://www.keil.com/pack 此处下载的是STM32L4xx的支持包 https://www.keil.com/pack/Keil.STM32L4xx_DFP.2.6.2.pack STM32CubeMX 下载与包下载 i. 下载&#xff08;需要使用用户&…

爬虫学习(三)用beautiful 解析html

安装库 import requests from bs4 import BeautifulSoup headers {"User-Agent" : "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.0.0 Safari/537.36 Edg/119.0.0.0"} for start_num in range(0,250…

【Gstreamer】自定义Plugin及调用Plugin

Gstreamer自定义Plugin及调用自定义Plugin Gstreamer支持开发者自己创建Plugin&#xff0c;创建后的Plugin可以通过工具gst-inspect-1.0查看&#xff0c;并在代码中调用自定义的plugin。 Gstreamer 官网中给出了Plugin创建教程&#xff0c;但实际上如果按照教程一步步走&…