C语言之结构体

一.前言引入.

我们知道在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;
}

结果:

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

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

最后,学习进步!!!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【数据结构(五)】递归

文章目录 1. 递归的概念2. 递归能解决什么问题3. 递归的规则4. 递归实际应用案例4.1. 迷宫问题4.2. 八皇后问题4.2.1. 思路分析4.2.1. 代码实现 1. 递归的概念 简单的说: 递归就是方法自己调用自己&#xff0c;每次调用时传入不同的变量。递归有助于编程者解决复杂的问题&…

第二节:服务拆分(案例)

一、服务拆分注意事项 1.1 拆分原则 每个微服务&#xff0c;不要重复开发相同业务&#xff08;例如在单体项目中用到了一个查询&#xff0c;这个查询功能能够查询出订单信息、商品信息、用户信息&#xff0c;那么在拆分微服务时就不要将其写在一起了&#xff0c;订单的微服务只…

推荐3个完美替代 Navicat 的工具

现在企业&#xff0c;mysql数据库用的比较多&#xff0c;mysql数据库客户端的需求也就比较大&#xff0c;navicat就被大家所熟知。 这个工具&#xff0c;确实好用&#xff0c;功能也非常强大&#xff0c;但是&#xff0c;它的强大&#xff0c;是需要付费&#xff0c;或者用一些…

windows ce Remote Process Explorer定位程序崩溃地址

windows ce Remote Process Explorer定位程序崩溃地址 一&#xff1a;下载地址二&#xff1a;使用1&#xff09;找到程序基准地址2) 定位程序异常位置 一&#xff1a;下载地址 链接&#xff1a;https://pan.baidu.com/s/1fQVBpputtRmynqa95DaPrg 提取码&#xff1a;cx65 二&a…

hexo博客部署到云服务器

欢迎大家到我的博客浏览。hexo博客部署到云服务器 | YinKais Blog 这篇文章带大家将hexo博客部署到云服务器上&#xff01; 一、服务器环境安装 1、安装 node js yum install gcc-c make yum -y install nodejs yum -y install npm 验证 node -v npm -v 2、安装git、ngin…

DOM 事件的注册和移除

前端面试大全DOM 事件的注册和移除 &#x1f31f;经典真题 &#x1f31f;DOM 注册事件 HTML 元素中注册事件 DOM0 级方式注册事件 DOM2 级方式注册事件 &#x1f31f;DOM 移除事件 &#x1f31f;真题解答 &#x1f31f;总结 &#x1f31f;经典真题 总结一下 DOM 中如何…

SQL Server 数据库,为products表添加数据

在插入数据的时候&#xff0c;需要注意以下事项。 > 每次插入一整行数据&#xff0c;不可能只插入半行或几列数据。 > 数据值的数目必须与列数相同&#xff0c;每个数据值的数据类型、精度和小数位数也必须与相应的 列匹配。 > INSERT语句不能为标识列指定值&#…

DCCK“启航计划“3+2第三课相机参数于选型

面积小&#xff0c;通电发热都用以引入噪声

Leetcode—392.判断子序列【简单】

2023每日刷题&#xff08;四十七&#xff09; Leetcode—392.判断子序列 双指针实现代码 bool isSubsequence(char* s, char* t) {int lens strlen(s);int lent strlen(t);int left 0, right 0;if(lens 0) {return true;}while(right < lent) {if(t[right] s[left])…

百度下拉词挖掘工具,百度下拉词挖掘获取软件

百度下拉词挖掘工具 百度下拉词挖掘工具&#xff0c;作为站长和SEO人员必备的工具之一&#xff0c;有着令人瞩目的功能。它能够追踪用户在百度搜索栏中输入关键词时&#xff0c;百度自动为用户推荐的下拉关键词。这一推荐不仅仅是用户搜索历史的体现&#xff0c;更是一种市场需…

计算机网络TCP篇①

目录 一、TCP 基本信息 1.1、TCP 的头格式 1.2、什么是 TCP 1.3、什么是 TCP 连接 1.4、TCP 与 UDP 的区别 1.2、TCP 连接建立 1.2.1、TCP 三次握手的过程 1.2.2、为什么是三次握手&#xff1f;不是两次&#xff1f;四次&#xff1f;&#xff08;这个问题真是典中典&am…

二、设置三台虚拟机的内存、MAC地址、IP地址

目录 1、配置内存 2、配置MAC地址 2.1 配置node2的MAC地址

春秋云镜 CVE-2022-30887

春秋云镜 CVE-2022-30887 多语言药房管理系统 (MPMS) 靶场介绍 多语言药房管理系统 (MPMS) 是用 PHP 和 MySQL 开发的, 该软件的主要目的是在药房和客户之间提供一套接口&#xff0c;客户是该软件的主要用户。该软件有助于为药房业务创建一个综合数据库&#xff0c;并根据到期…

通义千问 模型学习 和 SDK试用

通义千问-14B-Chat-Int4 模型库 (modelscope.cn) **通义千问-14B&#xff08;Qwen-14B&#xff09;**是阿里云研发的通义千问大模型系列的140亿参数规模的模型。Qwen-14B是基于Transformer的大语言模型, 在超大规模的预训练数据上进行训练得到。预训练数据类型多样&#xff0…

elementUI实现根据屏幕大小自适应换行,栅格化布局

需求&#xff1a; 默认一行展示4个卡片&#xff1b;当屏幕小于某个大小的时候&#xff0c;一行展示3个卡片&#xff1b;再小就展示2个&#xff1b;以此类推&#xff0c;最小就展示1个。 效果卡片样式如下图&#xff1a; 默认一行4个 屏幕缩小到某个阈值&#xff0c;一行展示…

软件测试工程师如何面试?

首先作为HR的角度&#xff1a; 一般我们面试的时候都会问应聘者一些问题&#xff0c;但是问什么&#xff1f;怎么问&#xff1f;每个HR都会有不同的做法。 有的HR问的比较广泛&#xff0c;有的HR比较注重专业度&#xff0c;还有的HR喜欢问一些开放性的问题&#xff0c;没有标…

工业机器视觉megauging(向光有光)使用说明书(三,轻量级的visionpro)

下来我们说说第二个相机的添加&#xff1a; 第一步&#xff0c;点击相机二&#xff0c;如下&#xff1a; 第二步&#xff0c;点击&#xff1a;加载工具组.xml&#xff0c;加载toolgroupxml2目录下的&#xff1a;工具组.xml 注意&#xff0c;一个相机只能用一个toolgroupxml,第…

SQL Server 2016(在Products表中查询数据)

1、实验环境。 以实验案例一的结果为环境。 2、需求描述。 【1】查询成本低于10元的水果信息。 【2】将所有蔬菜的成本上调1源。 【3】查询成本大于3元并小于40元的产品信息&#xff0c;并按照成本从高到低的顺序显示结果。 【4】查询成本最高的5个产品信息。 【5】查询有…

Java开发实战(二):IDEA安装

工欲善其事&#xff0c;必先利其器。这句话同样适用于学习Java编程。在开始Java的学习旅程之前&#xff0c;我们必须首先配置好适合的开发环境。 通过事先准备好这些工具和配置&#xff0c;我们可以避免在学习过程中遇到因环境问题导致的代码异常或错误。一个稳定、高效的开发环…

centos7 yum安装nginx

1.安装源 yum install epel-release 2.安装 (-y 的意思是自动yes) yum install nginx -y 3.查找安装到哪里了 whereis nginx 一般都是在 /etc/nginx下面 4.常用命令 检查配置文件是否正确 nginx -t 启动 systemctl start nginx 查看状态 systemctl status nginx 设置开…