【C语言__结构体__复习篇3】

目录

前言

 一、结构体基础知识

        1.1 结构体的语法形式

        1.2 创建结构体变量

        1.3 结构体变量的初始化

        1.4 点(.)操作符和箭头(->)操作符

二、匿名结构体

三、结构体自引用 

四、结构体内存对齐

        4.1 内存对齐的规则

        4.2 出现结构体内存对齐的原因

        4.3 修改默认对齐数

五、结构体传参

六、结构体实现位段

        6.1 什么是位段

        6.2 位段成员的内存分配

        6.3 位段的跨平台问题

        6.4 位段使用的注意事项


前言

本篇主要讨论以下问题:

结构体基础知识:

1. 结构体是用来做什么的,结构体的语法形式是怎样的

2. 如何创建一个结构体类型,结构体的全局变量和局部变量如何创建

3. 如何初始化结构体变量,如何自定义顺序初始化结构体变量,结构体中结构体如何初始化

4. 如何找到结构体变量的成员变量

匿名结构体:

5. 匿名结构体如何创建,它有什么需要注意的点

结构体自引用:

6. 结构体如何自引用,结构体自引用时有什么需要注意的点

结构体内存对齐:

7. 结构体内存对齐的规则有哪几点,怎样计算结构体成员变量相对于默认起始地址的偏移

8. 为什么会出现结构体内存对齐,如何改变结构体的默认对齐数

结构体传参:

9. 结构体传参采用传值调用还是传址调用好,为什么

结构体实现位段:

10. 结构体如何实现位段,用结构体实现位段有什么优点和缺点

11. 用结构体实现位段,位段的成员在内存中如何开辟空间的

12. 我们在使用位段时有什么注意事项,采用位段的示例

 一、结构体基础知识

1. 结构体是用来描述复杂对象的,例如,当我们想要描述学生、商品、书籍等自身包含多信息的复杂对象时,就可以创建结构体类型,再去创建结构体变量。

1.1 结构体的语法形式

1. 语法形式:struct tag

                      {

                              成员列表...

                      }变量列表;

① struct是结构体关键字,tag是结构体标签名,struct tag整体表示这个结构体类型的名称。

② { }内的成员列表,用于表示这个结构体类型中有哪些类型的变量(即,复杂对象包含哪些信息),这些成员变量不需要初始化。

③ 变量列表处创建的变量是全局变量,在此处我们可以一次性创建多个全局变量。

2. 创建结构体类型的举例:

    struct stu

    {

            char name[20];

            int age;

            int id[10];

    };

1.2 创建结构体变量

struct Stu
{
	char name[20];
	int age;
	char id[10];
}s1, s2;//全局变量

int main()
{
	struct Stu s3;//局部变量
	struct Stu s4;//局部变量
	return 0;
}

1.3 结构体变量的初始化

1. 按结构体成员的顺序初始化结构体变量,用{ }像数组一样直接初始化即可。

2. 自定义顺序初始化结构体变量,采用(.)找到结构体成员名,再赋值即可。

3. 结构体中的结构体的初始话也是用{ }, 类似于二维数组中一维数组的初始化。

struct Stu
{
	char name[20];
	int age;
	char id[10];
}s1 = { "张三", 18, "10023211" }, s2 = {"翠花", 19, "10023245"};

int main()
{
	struct Stu s3 = {"lisi", 17, "10023233"};
	struct Stu s4 = {.age = 16, .id = "10012323", .name = "kiki"}; //自定义顺序
	return 0;
}
struct Point
{
	int x;
	int y;
};

struct test
{
	float score;
	struct Point k;
};

int main()
{
	struct test t1 = { 90.8f, {2, 4} };//结构体中的结构体成员初始化
	struct test t2 = {.k.y = 4, .k.x = 9, .score = 89.7f};//自定义顺序
	return 0;
}

1.4 点(.)操作符和箭头(->)操作符

1. 点操作符:结构体变量名.成员名

2. 箭头操作符:结构体指针->成员名(表示通过地址找到它所指向的结构体变量的某个成员)

#include <stdio.h>

struct Stu
{
	char name[20];
	int age;
	char id[10];
}s1 = { "张三", 18, "10023211" }, s2 = {"翠花", 19, "10023245"};

int main()
{
	struct Stu s3 = {"lisi", 17, "10023233"};
	struct Stu s4 = {.age = 16, .id = "10012323", .name = "kiki"}; //自定义顺序

	struct Stu* ps3 = &s3;
	struct Stu* ps4 = &s4;
	printf("%s %d %s\n", s1.name, s1.age, s1.id);
	printf("%s %d %s\n", s2.name, s2.age, s2.id);
	printf("%s %d %s\n", ps3->name, ps3->age, ps3->id);
	printf("%s %d %s\n", ps4->name, ps4->age, ps4->id);
	return 0;
}

 

二、匿名结构体

1. 匿名结构体,即结构体类型在定义时 tag 不写。

2. 匿名结构体的特点,可以定义多个全局变量,但不可以定义局部变量。(如果对这个匿名结构体用typedef重命名后,是可以定义局部变量的)

3. 定义两个完全相同的匿名结构体类型,会被编译器认为是不同的结构体类型。

struct
{
	int num1;
	float num2;
};//匿名结构体

三、结构体自引用 

1. 结构体自引用,即在结构体类型定义时,结构体成员变量中存在本结构体类型的指针变量。

2. 匿名结构体不能自引用,这样写的代码可读性很差。

3. 一般在结构体自引用时,会先定义结构体,再用typedef对结构体重命名,注意!重命名结构体后不需要去更改自引用指针变量的类型名称,否则会出现未定义先使用的错误。

typedef struct Stu
{
	char name[20];
	struct Stu* ps;
}Stu;

typedef struct Stu
{
	char name[20];
	Stu* ps;//err,先使用后定义的错误
}Stu;

四、结构体内存对齐

1. 结构体内存对齐==结构体大小如何计算。

4.1 内存对齐的规则

1. 结构体的第⼀个成员对⻬到和结构体变量起始位置偏移量为0的地址处。

2. 其他成员变量要对⻬到对⻬数的整数倍的偏移地址处。

    对⻬数 = 编译器默认的⼀个对⻬数 与 该成员变量⼤⼩的较⼩值。

    - VS 中默认的值为 8

    - Linux中 gcc 没有默认对⻬数,对⻬数就是成员⾃⾝的⼤⼩

3. 结构体总⼤⼩为最⼤对⻬数(结构体中每个成员变量都有⼀个对⻬数,所有对⻬数中最⼤的)的整数倍。

4. 如果嵌套了结构体的情况,嵌套的结构体成员对⻬到⾃⼰的成员中最⼤对⻬数的整数倍的偏移位置处,结构体的整体⼤⼩是所有最⼤对⻬数(含嵌套结构体中成员的对⻬数)的整数倍。

补充:计算结构体成员变量相对于默认起始地址的偏移量用到的宏:offsetof(type, member),type是结构体类型,member是结构体成员名,头文件<stddef.h>,计算结果为size_t类型。

 

 4.2 出现结构体内存对齐的原因

1. 平台原因 (移植原因): 不是所有的硬件平台都能访问任意地址上的任意数据,某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。

2. 性能原因: 数据结构(尤其是栈)应该尽可能地在⾃然边界上对⻬。原因在于,为了访问未对⻬的内存,处理器需要作两次内存访问;⽽对⻬的内存访问仅需要⼀次访问。假设⼀个处理器总是从内存中取8个字节,则地址必须是8的倍数。如果我们能保证将所有的double类型的数据的地址都对⻬成8的倍数,那么就可以⽤⼀个内存操作来读或者写值了。否则,我们可能需要执⾏两次内存访问,因为对象可能被分放在两个8字节内存块中。

总体来说:结构体的内存对⻬是拿空间来换取时间的做法。

补充:在设计结构体的时候,我们既要满⾜对⻬,⼜要节省空间,如何做到:让占⽤空间⼩的成员尽量集中在⼀起

4.3 修改默认对齐数

1. #pragma 这个预处理指令,可以改变编译器的默认对⻬数。

2. 修改的默认对齐数不要用除1之外的奇数,因为类型的大小通常为偶数。(通常会将默认对齐数设置成1)

#include <stdio.h>
#pragma pack(1)//设置默认对⻬数为1
struct S
{
     char c1;
     int i;
     char c2;
};
#pragma pack()//取消设置的对⻬数,还原为默认
int main()
{
     //输出的结果是什么?
     printf("%d\n", sizeof(struct S));
     return 0;
}

五、结构体传参

1. 直接传结构体变量名,函数形参用结构体变量接收。

2. 传结构体变量地址,函数形参用结构体变量指针接收。

3. 结构体传参采用传结构体变量地址好,因为函数传参的时候,参数是需要压栈的,会有时间和空间上的系统开销,如果传递⼀个结构体对象所需开辟的内存空间过⼤,会导致参数压栈的系统开销⽐较⼤,代码性能的下降。

六、结构体实现位段

6.1 什么是位段

位段的声明和结构是类似的,有两个不同:

1. 位段的成员必须是 int、unsigned int 或 signed int ,在C99中位段成员的类型也可以选择其他类型。

2. 位段的成员名后边有⼀个冒号和⼀个数字。

struct A
{
     int _a:2;
     int _b:5;
     int _c:10;
     int _d:30;
};

 6.2 位段成员的内存分配

1. 位段的成员可以是 int、unsigned int、signed int 或者是 char 等类型。

2. 位段的内存空间上是按照需要以一次开辟4个字节( int )或者1个字节( char )的⽅式来申请内存空间的。

3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使⽤位段。(要想知道位段成员在内存中如何分配空间的,需要针对不同的平台去研究)

6.3 位段的跨平台问题

1. int 位段被当成有符号数还是⽆符号数是不确定的。

2. 位段中最⼤位的数⽬不能确定。(16位机器最⼤16,32位机器最⼤32,写成27,在16位机器会出问题)

3. 位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。

4. 当⼀个结构包含两个位段,第⼆个位段成员⽐较⼤,⽆法容纳于第⼀个位段剩余的位时,是舍弃剩余的位还是利⽤,这也是不确定的。

总结: 跟结构相⽐,位段可以达到同样的效果,并且可以很好的节省空间,但是有跨平台的问题存在。

6.4 位段使用的注意事项

1. 位段的⼏个成员可能共用同⼀个字节,这样有些成员的起始位置并不是某个字节的起始位置。内存中每个字节有唯一的⼀个地址,⼀个字节内部的bit位是没有地址的。 所以不能对位段的成员使⽤&操作符,这样也就意味着不能使⽤scanf直接给位段的成员输⼊值,只能是先输⼊放在⼀个变量中,然后赋值给位段的成员。

2. 采用位段的示例:⽹络协议中,IP数据报就是使用的结构体位段。

struct A
{
	int a : 2;
	int b : 5;
	int c : 10;
	int d : 30;
};

int main()
{
	struct A sa = { 0 };
	//scanf("%d", &(sa.b));//这是错误的

	//正确的⽰范
	int b = 0;
	scanf("%d", &b);
	sa.b = b;
	return 0;
}

本篇文章已完结,谢谢支持!!! 

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

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

相关文章

8:系统开发基础--8.1:软件工程概述、8.2:软件开发方法 、8.3:软件开发模型、8.4:系统分析

转上一节&#xff1a; http://t.csdnimg.cn/G7lfmhttp://t.csdnimg.cn/G7lfm 课程内容提要&#xff1a; 8&#xff1a;知识点考点详解 8.1&#xff1a;软件工程概述 1.软件的生存周期 2.软件过程改进—CMM Capability Maturity Model能力成熟度模型 3.软件过程改进—CMMI—…

Jmeter配置服务器监控插件

1.安装插件管理器 插件官网地址&#xff1a;JMeter Plugins :: JMeter-Plugins.org 点击 Plugins Manager,如上图所示&#xff0c; &#xff0c;点击jar file下载“plugins-manager.jar”&#xff0c;下载后放到“jmeter\lib\ext”目录下&#xff0c;重启jmeter。 2.安装资源…

LeetCode 94 二叉树的中序遍历

题目描述 二叉树的中序遍历 给定一个二叉树的根节点 root &#xff0c;返回 它的 中序 遍历 。 示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&#xff1a;[1,3,2]示例 2&#xff1a; 输入&#xff1a;root [] 输出&#xff1a;[]示例 3&#xff1a; 输入…

Composite 组合

意图 将对象组合成树形结构以表示“部分-整体”的层级结构。Composite使得用户对单个对象和组合对象的使用具有一致性。 结构 其中&#xff1a; Component为组合中的对象声明接口&#xff1b;在适当情况下实现所有类共有接口的默认行为&#xff1b;声明一个接口用于访问和管…

Spring Boot(二)— 自定义Spring Boot Starter

在Spring Boot中&#xff0c;自定义Spring Boot Starter是一个常见且强大的功能&#xff0c;它允许开发者为特定的功能或库创建自己的自动配置&#xff0c;从而简化集成过程。 1 前置知识 Spring Boot的事件为应用的启动和关闭提供了详细的上下文信息&#xff0c;使得开发者能…

OSI七层网络模型 —— 筑梦之路

在信息技术领域&#xff0c;OSI七层模型是一个经典的网络通信框架&#xff0c;它将网络通信分为七个层次&#xff0c;每一层都有其独特的功能和作用。为了帮助记忆这七个层次&#xff0c;有一个巧妙的方法&#xff1a;将每个层次的英文单词首字母组合起来&#xff0c;形成了一句…

腾讯云优惠券详细介绍及领券步骤详解

随着云计算技术的不断发展和普及&#xff0c;越来越多的企业和个人开始选择使用云服务来满足自身的需求。腾讯云作为国内领先的云服务提供商&#xff0c;以其稳定、高效、安全的服务赢得了广大用户的信赖。为了回馈广大用户&#xff0c;腾讯云经常推出各种优惠活动&#xff0c;…

【JS】数组交换位置

公式 arr.splice(oldIndex, delCount, ...arr.splice(newIndex, delCount, arr[oldIndex])) arr - 操作的数组delCount - 删除的数组个数oldIndex - 交换位置的数组下标1newIndex - 交换位置的数组下标2...arr - 提取数组里的元素 splice删除元素时&#xff0c;返回一个数组&a…

利用计算机视觉算法提取裂纹相关特征参数信息

ABCnutter/CrackFeature: &#x1f680;使用计算机视觉相关算法提取裂缝的骨架&#xff08;矢量化&#xff09;、轮廓【支持提前修复断裂裂缝】&#xff0c;以及几何特征参数&#xff08;长度、宽度、面积和主要方向&#xff09;【欢迎Star】。主要流程以及相关算法如下&#x…

异构超图嵌入的图分类 笔记

1 Title Heterogeneous Hypergraph Embedding for Graph Classification&#xff08;Xiangguo Sun , PictureHongzhi Yin , PictureBo Liu , PictureHongxu Chen , PictureJiuxin Cao , PictureYingxia Shao , PictureNguyen Quoc Viet Hung&#xff09;【WSDM 2021】 2 Co…

1038: 顺序表中重复数据的删除

解法&#xff1a; #include<iostream> #include<vector> #include<algorithm> using namespace std; int main() {int n, k;cin >> n;vector<int> arr(n);for (auto& x : arr) cin >> x;cin >> k;int sum 0;for (auto x : arr…

句柄ros::NodeHandle nh(“~“)与nh对launch文件参数配置(param)的影响

ros::NodeHandle nh("~"); 改为&#xff1a; ros::NodeHandle nh; 即可 /*************************分割线 ************************/ 如果原本是&#xff1a; ros::NodeHandle nh;可以改成&#xff1a; ros::NodeHandle nh("~"); 试试

反射与动态代理

一、反射 什么是反射? 反射允许对成员变量&#xff0c;成员方法和构造方法的信息进行编程访问 1.获取class对象的三种方式 Class这个类里面的静态方法forName&#xff08;“全类名”&#xff09;&#xff08;最常用&#xff09; 通过class属性获取 通过对象获取字节码文件对…

浏览器原理---事件循环

浏览器原理 学习浏览器原理对于我们开发是很有必要的 我们可以了解到浏览器内部工作原理对自己的代码进行优化 进程线程 首先了解进程和线程 进程就就是内存在正在进行的应用程序 在内存中独占一个内存空间 并且进程之间是隔离的 可以看到每个应用都有一个进程 占用空间内存…

【300套】基于Springboot+Vue的Java毕业设计项目(附源码+演示视频+LW)

大家好&#xff01;我是程序猿老A&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f9e1;今天给大家分享300的Java毕业设计&#xff0c;基于Springbootvue框架&#xff0c;这些项目都经过精心挑选&#xff0c;涵盖了不同的实战主题和用例&#xff0c;可做毕业…

【C++进阶】RAII思想&智能指针

智能指针 一&#xff0c;为什么要用智能指针&#xff08;内存泄漏问题&#xff09;内存泄漏 二&#xff0c;智能指针的原理2.1 RAII思想2.2 C智能指针发展历史 三&#xff0c;更靠谱的shared_ptr3.1 引用计数3.2 循环引用3.3 定制删除器 四&#xff0c;总结 上一节我们在讲抛异…

Vulnhub靶机 DC-1渗透详细过程

Vulnhub靶机:DC-1渗透详细过程 目录 Vulnhub靶机:DC-1渗透详细过程一、将靶机导入到虚拟机当中二、攻击方式主机发现端口扫描web渗透利用msf反弹shell数据库信息web管理员密码提权 一、将靶机导入到虚拟机当中 靶机地址&#xff1a; https://www.vulnhub.com/entry/dc-1-1,29…

Open CASCADE学习|BRepOffsetAPI_DraftAngle

BRepOffsetAPI_DraftAngle 是 Open CASCADE Technology (OCCT) 中用于创建带有草图斜面的几何体的类。草图斜面是一种在零件设计中常见的特征&#xff0c;它可以在零件的表面上创建一个倾斜的面&#xff0c;通常用于便于零件的脱模或是增加零件的强度。 本例演示了如何创建一个…

启动nginx时报错:signal process started

解决方案&#xff0c;直接使用该命令启动&#xff0c;指向nginx.conf配置文件&#xff1a; nginx -c /www/wdlinux/nginx/conf/nginx.conf 启动成功&#xff1a;

C语言高质量编程之assert()和const

目录 编程中常见的错误 assert() const 编程中常见的错误 在编程中我们通常会遇到三种错误形式&#xff0c;分别是&#xff1a;编译型错误&#xff0c;链接型错误&#xff0c;运行时错误。 编译型错误&#xff1a; 在编译阶段发生的错误&#xff0c;绝大多数情况是由语法错误…