C语言——结构体——(第21篇)

坚持就是胜利

文章目录

  • 一、结构体类型的声明
    • 1、结构的基础知识
    • 2、结构的声明
    • 3、结构成员的类型
  • 二、结构体变量的定义和初始化
      • 结构体中包含结构体
  • 三、结构体成员访问
    • 问题一:t.name = "张三",对吗?
    • 问题二:set_Stu(s);应该“值传递”,还是“址传递”?
  • 四、结构体传参


一、结构体类型的声明

1、结构的基础知识

结构是一些值的集合,这些值称为 成员变量。结构的每个成员可以是不同类型的变量。

数组:一组相同类型元素的集合。
结构体:一组不一定相同类型元素的集合。

生活中描述:
人:名字+性别+年龄+身高+身份证号码+地址…
书:书名+作者+出版社+定价+书号…

复杂对象,不能通过内置类型直接描述和表示。
就有结构体来描述复杂类型。

2、结构的声明

struct tag   //tag 随意更改
{
	member - list;   //成员列表
}variable=list;   //结构体变量列表

描述一个学生:

struct Stu
{
	char name[20];
	int age;
	char sex[5];
	char id[20];
}x1, x2;                 //x1,x2 是 全局变量

struct Stu y1,y2;        //y1,y2 是 全局变量

int main()
{
	struct Stu s1, s2;   //s1,s2 是 局部变量
	return 0;
}
typedef struct Stu    //typedef 重新起个名字
{
	char name[20];
	int age;
	char sex[5];
	char id[20];
}Stu;                 //Stu 就是 struct Stu 的别名

int main()
{
	struct Stu s1;   //局部变量 s1       
	Stu s2;   //局部变量 s2        s1 和 s2 是一样的
	return 0;
}

//在C语言中,没有对结构体类型typedef,struct关键字不能省略

3、结构成员的类型

结构的成员可以是 标量,数组,指针,甚至是 其他结构体

struct S
{
	int a;
	char arr[5];
	int* p;
};

struct B
{
	char ch[10];

	struct S s1;  //结构体中包含结构体

	double d;
};

int main()
{
	return 0;
}

二、结构体变量的定义和初始化

struct S
{
	int a;
	char arr[5];
	int* p;
}s1;                //第一种 定义变量  全局变量

struct S s2;        //第二种 定义变量  全局变量

struct B
{
	char ch[10];
	struct S s;
	double d;
};

int main()
{
	struct S s3;    //第三种 定义变量  局部变量

	return 0;
}
struct S
{
	int a;
	char arr[5];
	int* p;
}s1 = { 100,"abcd",NULL };                                    //第一种 定义变量  全局变量

//注意:在给 arr[5] 赋值的时候,字符串中包含 '\0',所以最多就只可以 a b c d 这4个字符,再加上 '\0',一共5个字符

int n = 2013;
struct S s2 = { 98,"hehe",&n };                               //第二种 定义变量  全局变量

struct B
{
	char ch[10];
	struct S s;
	double d;
};

int main()
{
	               //如果不想按照顺序,那只要在前面加上 '.'

	struct S s3 = { .arr = "abc",.p = NULL,.a = 1 };         //第三种 定义变量  局部变量

	printf("%d %s %p\n", s3.a, s3.arr, s3.p);    // '.' 结构成员访问操作符
	return 0;
}

在这里插入图片描述

结构体中包含结构体

#include <stdio.h>

struct S
{
	int a;
	char arr[5];
	int* p;
}s1 = { 100,"abcd",NULL };                                   



int n = 2013;
struct S s2 = { 98,"hehe",&n };                               

struct B
{
	char ch[10];
	struct S s;
	double d;
};

int main()
{
	struct S s3 = { .arr = "abc",.p = NULL,.a = 1 };         
	//printf("%d %s %p\n", s3.a, s3.arr, s3.p);   

	struct B sa = { "hello",{20,"aaa",NULL},3.14 };   //结构体中包含结构体

	printf("%s %d %s %p %lf\n", sa.ch, sa.s.a, sa.s.arr, sa.s.p, sa.d);

	return 0;
}

在这里插入图片描述

三、结构体成员访问

1、结构体变量访问成员 结构变量的成员是通过点操作符(.)访问的。点操作符接受两个操作数。
在这里插入图片描述
我们可以看到 s 有成员 name 和 age ;那我们如何访问 s 的成员?

struct S s;
strcpy(s.name, "sunchao");   //使用 . 访问 name 成员
s.age = 20;  //使用 . 访问 age 成员

2、结构体指针访问指向变量的成员 有时候我们得到的不是一个结构体变量,而是指向一个结构体的指针。
那该如何访问成员。如下:

#include <stdio.h>

struct Stu
{
	char name[20];
	int age;
};

void print(struct Stu* ps)
{
	printf("name = %s   age = %d\n", (*ps).name, (*ps).age);
	printf("name = %s   age = %d\n", ps->name, ps->age);
}

int main()
{
	struct Stu s = { "zhangsan",20 };
	print(&s);   //结构体地址传参
	return 0;
}

问题一:t.name = “张三”,对吗?

#include <stdio.h>

struct Stu
{
	char name[20];
	int age;
};

void set_Stu(struct Stu t)
{
	t.age = 20;        //问题一:
	t.name = "张三";   //错误的,因为 name 是 数组名,是 地址,不可以给 地址 赋值 字符串
}

int main()
{
	struct Stu s = { 0 };   //先不初始化
	set_Stu(s);   //值传递还是址传递? 这里先尝试 值传递
	return 0;
}

在这里插入图片描述
改正一:
用 strcpy 来解决

#include <stdio.h>
#include <stdio.h>

struct Stu
{
	char name[20];
	int age;
};

void set_Stu(struct Stu t)
{
	t.age = 20;
	//t.name = "张三";   //错误的,因为 name 是 数组名,是 地址,不可以给 地址 赋值 字符串

	//正确的方法
	strcpy(t.name, "张三");   //字符串拷贝
} 

int main()
{
	struct Stu s = { 0 };   //先不初始化
	set_Stu(s);   //值传递还是址传递? 这里是 值传递
	return 0;
}

问题二:set_Stu(s);应该“值传递”,还是“址传递”?

#include <stdio.h>
#include <stdio.h>

struct Stu
{
	char name[20];
	int age;
};

void set_Stu(struct Stu t)
{
	t.age = 20;
	//t.name = "张三";   //错误的,因为 name 是 数组名,是 地址,不可以给 地址 赋值 字符串

	//正确的方法
	strcpy(t.name, "张三");   //字符串拷贝
} 

void print_stu(struct Stu t)
{
	printf("%s %d", t.name, t.age);
}

int main()
{
	struct Stu s = { 0 };   //先不初始化
	               
	              //错误
	set_Stu(s);   //值传递还是址传递? 这里是 值传递。说明 值传递 是错误的。
	print_stu(s);
	return 0;
}

在这里插入图片描述
改正二:

#include <stdio.h>
#include <stdio.h>

struct Stu
{
	char name[20];
	int age;
};

void set_Stu(struct Stu* t)
{
	(*t).age = 20;       //改变成(*t)
	//t.name = "张三";   //错误的,因为 name 是 数组名,是 地址,不可以给 地址 赋值 字符串

	//正确的方法
	strcpy((*t).name, "张三");   //字符串拷贝  //改变成(*t)
} 

void print_stu(struct Stu t)
{
	printf("%s %d", t.name, t.age);
}

int main()
{
	struct Stu s = { 0 };   //先不初始化
	               
	              //错误
	//set_Stu(s);   //值传递还是址传递? 这里是 值传递。说明 值传递 是错误的。

	              //正确方法:址传递  并且对应的函数set_Stu(struct Stu* t),改变成对应类型的  struct Stu*
	set_Stu(&s);
	print_stu(s);
	return 0;
}

在这里插入图片描述
(*t).age
t -> age

#include <stdio.h>
#include <stdio.h>

struct Stu
{
	char name[20];
	int age;
};

void set_Stu(struct Stu* t)
{
	//(*t).age = 20;       //改变成(*t)
	//或者
	t->age = 20;
	//t.name = "张三";   //错误的,因为 name 是 数组名,是 地址,不可以给 地址 赋值 字符串

	//正确的方法
	//strcpy((*t).name, "张三");   //字符串拷贝  //改变成(*t)
	//或者
	strcpy(t->name, "张三");
} 

void print_stu(struct Stu t)
{
	printf("%s %d", t.name, t.age);
}

int main()
{
	struct Stu s = { 0 };   //先不初始化
	               
	              //错误
	//set_Stu(s);   //值传递还是址传递? 这里是 值传递。说明 值传递 是错误的。

	              //正确方法:址传递  并且对应的函数set_Stu(struct Stu* t),改变成对应类型的  struct Stu*
	set_Stu(&s);
	print_stu(s);
	return 0;
}

在这里插入图片描述

四、结构体传参

int Add(int x, int y)
{
	int z = 0;
	z = x + y;
	return z;
}

int main()
{
	int a = 3;
	int b = 5;
	int c = 0;
	c = Add(a, b);
	return 0;
}

在这里插入图片描述

struct S
{
	int data[1000];
	int num;
};

struct S s = { {1,2,3,4},1000 };   //全局变量

//结构体传参
void print1(struct S s)
{
	printf("%d\n", s.num);
}

//结构体地址传参
void print2(struct S* ps)
{
	printf("%d\n", ps->num);
}

int main()
{
	print1(s);   //传结构体  值传递

	print2(&s);  //传地址  址传递
	             
	return 0;
}

上面的 print1 和 print2 函数哪个好些?

答案是:首选 print2 函数。
原因:函数传参的时候,参数是需要压栈的。如果传递一个结构体对象的时候,结构体过大,参数压栈的系统开销比较大,所以会导致性能的下降。

实参传递给形参的时候,形参将是实参的一份临时拷贝。

如果在结构体中,采用值传递,会浪费非常多的空间。上面的例子中,就要开辟(1000*4+4)*2=8008个字节的内存空间。
然而使用址传递,则只要占用4字节或者8字节的地址空间。

结论:结构体传参的时候,要传结构体的地址。

微软雅黑字体
黑体
3号字
4号字
红色
绿色
蓝色

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

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

相关文章

【深度学习目标检测】十九、基于深度学习的芒果计数分割系统-含数据集、GUI和源码(python,yolov8)

使用深度学习算法检测芒果具有显著的优势和应用价值。以下是几个主要原因&#xff1a; 特征学习的能力&#xff1a;深度学习&#xff0c;特别是卷积神经网络&#xff08;CNN&#xff09;&#xff0c;能够从大量的芒果图像中自动学习和提取特征。这些特征可能是传统方法难以手动…

数学建模资料分享

1. 往年各赛题的优秀论文 可以用来参考一下论文是怎么写的。参考论文的结构&#xff0c;格式&#xff0c;思路等等。 链接&#xff1a;https://pan.baidu.com/s/1WG2t4-x9MjtaSgkq4ue5AQ?pwdnlzx 提取码&#xff1a;nlzx --来自百度网盘超级会员V4的分享 2.论文模板 链接&a…

操作系统访问控制机制

使用访问控制技术&#xff0c;可以设置用户对系统资源的访问权限&#xff0c;即限定用户只能访问允许访问的资源。访问控制还可以通过设置文件的属性&#xff0c;来保护文件只能被读而不能被修改&#xff0c;或只允许核准的用户对其进行修改等。 1.1 保护域 把一个进程能对某…

记录一次xpclr检测中报错(“No permission to write in the specified directory: {0}“)

这里写自定义目录标题 报错修正思路&#xff1a;将 -O参数修改成为相对目录加文件名称格式。 报错 看到这个问题&#xff0c;一头雾水&#xff0c;没有写入权限。结果是程序问题。 修正思路&#xff1a;将 -O’参数修改成为相对目录加文件名称格式。 代码&#xff1a;

软件实例,物流货运配货单打印模板软件单据打印查询管理系统软件教程,可以同时打印标签或补打

软件实例&#xff0c;物流货运配货单打印模板软件单据打印查询管理系统软件教程&#xff0c;可以同时打印标签或补打 一、前言 以下软件教程以 佳易王物流单打印查询系统V17.1为例说明 软件文件下载可以点击最下方官网卡片——软件下载——试用版软件下载 这个版本在原来基…

linux之JAVA环境配置Tomcat离线安装与启动

文章目录 一、jdk安装具体步骤二、tomcat安装具体步骤三、MySql具体步骤修改密码登录 四、部署单价项目具体步骤 一、jdk安装具体步骤 1、查询是否有jdk java -version 2、进入opt目录 cd /opt 3.连接服务器工具 进入opt目录&#xff0c;把压缩文件上传 4.等待传好之后&am…

二分算法(c++版)

二分的本质是什么&#xff1f; 很多人会认为单调性是二分的本质&#xff0c;但其实其本质并非单调性&#xff0c;只是说&#xff0c;有单调性的可以进行二分&#xff0c;但是有些题目没有单调性我们也可以进行二分。其本质其实是一个边界问题&#xff0c;给定一个条件&#xf…

【flutter】环境安装

安装flutter sdk 下载sdk flutter sdk就包含dart&#xff0c;所以我们只用安装flutter sdk就可以了。 我们去清华大学开源软件镜像站下载&#xff0c;flutter开发中&#xff0c;版本对不上基本项目就跑步起来&#xff0c;如果是团队协同开发的话&#xff0c;建议统一下载指定版…

【踩坑】PyTorch中指定GPU不生效和GPU编号不一致问题

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhang.cn] 指定GPU不生效问题 解释&#xff1a;就是使用os.environ["CUDA_VISIBLE_DEVICES"] "1"后&#xff0c;后面使用起来仍然是cuda0. 解决&#xff1a;在最开头就使用 import os os.environ[&…

python-mysql协程并发常用操作封装

目录 前言封装代码测试代码参考 前言 协程异步操作MYSQL是常用的&#xff0c;博主这里在GitHub上找了两个包&#xff0c;databases和aiomysql&#xff0c;第一个包除了mysql外还支持其他的数据库&#xff0c;且操作MYSQL时底层也是使用的aiomysql&#xff0c;但文档内容比较少…

【大数据】Flink 内存管理(三):TaskManager 内存分配(理论篇)

Flink 内存管理&#xff08;三&#xff09;&#xff1a;TaskManager 内存分配 1.配置 Total Memory2.配置 Heap and Managed Memory2.1 Task (Operator) Heap Memory2.2 Managed Memory 3.配置 Off-Heap Memory&#xff08;Direct or Native&#xff09;4.详细内存模型5.Framew…

YOLO系列论文阅读(v1--v3)

搞目标检测&#xff0c;绕不开的一个框架就是yolo&#xff0c;而且更糟糕的是&#xff0c;随着yolo的发展迭代&#xff0c;yolo网络可以做的事越来越多&#xff0c;语义分割&#xff0c;关键点检测&#xff0c;3D目标检测。。。这几天决定把YOLO系列彻底梳理一下&#xff0c;在…

C++的STL常用算法->常用遍历算法、常用查找算法、常用排序算法、常用拷贝和替换算法、常用算术生成算法、常用集合算法

#include<iostream> using namespace std; #include <algorithm> #include <vector> //常用遍历算法 for_each //普通函数 void print01(int val) { cout << val << " "; } //仿函数 //函数对象 class print02 { public: v…

Wireshark TS | Linux 系统对时问题

问题描述 节前业务运维同事提交了一个 case &#xff0c;说是部署在新业务区域的 Linux 服务器和老业务区域的 Linux 服务器无法对时&#xff0c;脚本里使用的是 clockdiff 命令&#xff0c;无法正常返回结果&#xff0c;而在老业务区域两台服务器之间执行命令就正常&#xff…

Java基于微信小程序的校园二手物品交易系统,附源码

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…

公厕智慧化_智慧化的公厕

公厕智慧化是现代城市建设中的重要一环。通过信息化、数字化和智慧化技术手段&#xff0c;实现对公共厕所的高效管理和服务&#xff0c;不仅提升了城市环境质量&#xff0c;还改善了居民生活品质。智慧公厕的智慧化包括监测、管理、服务和设备的智慧化&#xff0c;利用先进科技…

Unity中URP实现水体效果(水的深度)

文章目录 前言一、搭建预备场景1、新建一个面片&#xff0c;使其倾斜一个角度&#xff0c;来模拟水底和岸边的效果2、随便创建几个物体&#xff0c;作为与水面接触的物体3、再新建一个面片&#xff0c;作为水面 二、开始编写水体的Shader效果1、新建一个URP基础Shader2、把水体…

汇编语言movs指令学习

字符串传送指令(Move String Instruction) movs 该指令是把指针DS:SI所指向的字节、字或双字传送给指针ES:DI所指向内存单元&#xff0c;并根据标志位DF对寄存器DI和SI作相应增减。该指令的执行不影响任何标志位。 记不清这指令是8086就有的&#xff0c;还是386以后新加的&…

【Redis】常见的5种数据类型(上)

文章目录 1 :peach:前言:peach:2 :peach:Redis 基本的全局命令:peach:2.1 :apple:keys:apple:2.2 :apple:exists:apple:2.3 :apple:del:apple:2.4 :apple:expire:apple:2.5 :apple:ttl:apple:2.6 :apple:type:apple: 3 :peach:单线程架构:peach:4 :peach:Redis 的 5 种常见数据…

Qt_纯虚函数的信号和槽

简介 在C中&#xff0c;纯虚函数是一个在基类中声明但没有实现的虚函数。纯虚函数的声明以 “ 0” 结尾。纯虚函数的目的是为了提供一个接口&#xff0c;但是不提供实现。派生类必须实现纯虚函数&#xff0c;否则它也会成为一个抽象类。纯虚函数可以在基类中定义&#xff0c;也…