C语言.自定义类型:结构体

news2024/11/20 20:37:52

自定义类型:结构体

  • 1.结构体类型的声明
    • 1.1结构体回顾
      • 1.1.1结构体的声明
      • 1.1.2结构体变量的创建和初始化
    • 1.2结构体的特殊声明
    • 1.3结构体的自引用
  • 2.结构体内存对齐
    • 2.1对齐规则
    • 2.2为什么存在内存对齐
    • 2.3修改默认对齐数
  • 3.结构体传参
  • 4.结构体实现位段
    • 4.1什么是位段
    • 4.2位段的分配
    • 4.3位段的跨平台
    • 4.4位段的应用
    • 4.5位段使用的注意事项

1.结构体类型的声明

1.1结构体回顾

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

1.1.1结构体的声明

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

例如描述一个学生:

struct stu
{
	char name[20];//姓名
	int age;//年龄
	char sex;//性别
	char id[20];//学号
};//分号不能丢

1.1.2结构体变量的创建和初始化


#include <stdio.h>

//描述一个学生:
struct stu
{
	char name[20];//姓名
	int age;//年龄
	char sex[10];//性别
	char id[20];//学号
};

int main()
{
	//按照结构体的顺序初始化
	struct stu s1 = { "zhangsan", 20, "男", "2072855200" };

	printf("name: %s\n", s1.name);
	printf("age : %d\n", s1.age);
	printf("sex : %s\n", s1.sex);
	printf("id  : %s\n", s1.id);

	//按照指定的顺序初始化
	struct stu s2 = { .age = 20, .id = "2072855200", .name = "zhangsan", .sex = "男" };

	printf("name: %s\n", s2.name);
	printf("age : %d\n", s2.age);
	printf("sex : %s\n", s2.sex);
	printf("id  : %s\n", s2.id);

	return 0;
}

1.2结构体的特殊声明

在生命结构的时候,可以不完全声明。

比如:

struct 
{
	int a;
	char c;
	float f;
}x;

struct 
{
	int a;
	char c;
	float f;
}a[20], *p;

在两个结构体声明的时候,省略掉了结构体标签(tag)。

那么问题来了?

//上面的代码合法吗?
p = &x;

警告:

  • 编译器回吧上面的两个声明当成完全两个不同的类型,所以是非法的
  • 匿名的结构体类型,如果没有对结构体类型重命名的话,基本上只能用一次

1.3结构体的自引用

在结构体中的包含一个类型为该结构体本身的成员是否可以呢?

比如,定义一个链表的节点:

struct Node
{
	int data;
	struct Node next;
};

上述的代码正确吗?如果正确,那 sizeof(struct Node) 是多少?

仔细分析,其实是不行的,因为一个结构体中在包含一个类型的结构体变量,这样结构体变量的大小就会无穷的大,是不合理的。

正确的自引用方式:

struct Node
{
	int data;
	struct Node* next;
};

在结构体自引用使用的过程中,夹杂了 typedef 对匿名结构体类型重命名,也容易引入问题,看看下面的代码,可行吗?

typedef struct
{
	int data;
	Node* next;
}Node;

答案是不行的,因为Node是对前面的匿名结构体类型的重命名产生的,但是在匿名结构体内部提前使用Node类型来创建成员变量,这是不行的。

2.结构体内存对齐

讨论一个问题:计算结构体的大小。

2.1对齐规则

首先得掌握结构体的对齐原则:

  1. 结构体的第一个成员对齐到和结构体变量起始位置偏移量为0的地址处
  2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。

对齐数 = 编译器默认的一个对齐数与该成员变量大小的较小值。
VS 中的默认的值为 8
Linux中gcc没有默认对齐数,对齐数就是成员自身的大小

  1. 结构体总大小为最大对齐数(结构体中的每个成员变量都有一个对齐数,所有对齐数中最大的)的整数倍。
  2. 如果嵌套了结构体的情况,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(韩嵌套结构体成员的对齐数)的整数倍。
#include <stdio.h>

//练习1:
struct S1
{
	char c1;
	int i;
	char c2;
};

//练习2:
struct S2
{
	char c1;
	char c2;
	int i;
};

//练习3:
struct S3
{
	double d;
	char c;
	int i;
};

//练习4:
struct S4
{
	char c1;
	struct S3 s3;
	double d;
};

int main()
{
	printf("%zd\n", sizeof(struct S1));
	printf("%zd\n", sizeof(struct S2));
	printf("%zd\n", sizeof(struct S3));
	printf("%zd\n", sizeof(struct S4));

	return 0;
}

2.2为什么存在内存对齐

  1. 平台原因(移植原因):不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。
  2. 性能原因:数据结构(尤其是栈)应该尽可能地在自然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问。假设一个处理器总是从内存中取8个字节,则地址必须是8的倍数。如果我们能保证将所有的double类型的数据的地址都对齐成8的倍数,那么就可以用一个内存操作来读或者写值了。否则,我们可能需要执行两次内存访问,因为对象可能被分放在两个8字节内存块中。

总的来讲:结构体的对其实拿空间来换取时间的做法。

在设计结构体的时候,既要满足对齐,又要考虑节省空间,如何做到:

让占用空间小的成员尽量集中在一起。

//例如:
#include <stdio.h>

struct S1
{
	char c1;
	int i;
	char c2;
};

struct S2
{
	char c1; 
	char c2;
	int i;
};

int main()
{
	printf("%zd\n", sizeof(struct S1));
	printf("%zd\n", sizeof(struct S2));

	return 0;
}

S1和S2类型的成员一摸一样,但是S1和S2所占用空间的大小有了一些区别。

2.3修改默认对齐数

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

#include <stdio.h>
//设置默认对齐数为1
#pragma pack(1)
struct S
{
	char c1;
	int i;
	char c2;
};
//取消设置的对齐数,还原为默认
#pragma pack()


int main()
{
	printf("%zd\n", sizeof(struct S));

	return 0;
}

结构体在对齐方式不合适的时候,我们可以自己更改默认对齐数。

3.结构体传参

#include <stdio.h>

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

struct S s = { {1,2,3,4,5}, 10 };

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

int main()
{
	print1(s);//传结构体
	print2(&s);//传结构体的地址

	return 0;
}

上面的print1print2函数那个更好?

答案是:首选print2函数。

原因:

  • 函数传参的时候,参数是需要压栈,会有时间和空间上的系统开销。
  • 如果传递一个结构体对象的时候,结构体过打,参数压栈的的系统开销比较大,所以会导致性能的下降。

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

4.结构体实现位段

4.1什么是位段

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

  1. 位段的成员必须是 intunsigned intsigned int,在C99中位段成员的类型也可以选择其他类型。
  2. 位段的成员后面有一个冒号和一个数字

比如:

struct A
{
	int _a : 2;//单位是字节
	int _b : 3;
	int _c : 4;
	int _d : 30;
};

A就是一个位段类型。

那位段A所占内存空间大小是多少?

int main()
{
	printf("%zd\n", sizeof(struct A));//8

	return 0;
}

4.2位段的分配

  1. 位段的成员可以是intunsigned intsigned int或者是char等类型。
  2. 位段的空间是按照以4个字节(int)或者1个字节(char)的方式来开辟的。
  3. 位段涉及到很多不确定的因素,位段是不跨平台的,注意可移植的程序应该避免使用位段。
//一个例子:
#include <stdio.h>

struct S
{
	char a : 3;
	char b : 4;
	char c : 5;
	char d : 4;
};

int main()
{
	struct S s = { 0 };
	s.a = 10;
	s.b = 12;
	s.c = 3;
	s.d = 4;

	return 0;
}
//空间是如可开辟的?

在这里插入图片描述

4.3位段的跨平台

  1. int 位段被当成了有符号还是无符号是不确定的。
  2. 位段中最大位的数目不能确定。(16位机器最大的是16,32位机器最大32,写成17机器会出问题)。
  3. 位段中的成员在内存中从左向右分配还是从右向左分配,标准未定义。
  4. 当一个结构包含两个位段,第二个位段成员比较大,无法容纳第一个位段剩余的位段时,是舍弃剩余的位段还是利用,这是不确定的。

总结:

更结构体相比,位段可以达到相同的效果,并且可以很好的节省空间,但是存在跨平台的问题存在。

4.4位段的应用

下图是网络协议中,IP数据报的格式,我们可以看到其中很多的属性只需要几个bit位就能描述,这里使用位段,能够实现想要的效果,也节省了空间,这样网络传输的数据报大小也会较小一些,对网络的畅通是有帮助的。
在这里插入图片描述

4.5位段使用的注意事项

  • 位段的几个成员共有同一个字节,这样有些成员的起始位置并不是某个字节的起始位置,那么这些位置处是没有地址的。内存中每个字节分配一个地址一个字节内部的bit位是没有地址的。
  • 所以不能对位段的成员使用&操作符,这样就不能使用scanf直接给位段的成员输入值,只能先输入放在一个变量中,然后赋值给位段的成员。
#include <stdio.h>

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

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

	//先输入一个值放到一个变量中,然后赋值给位段的成员。
	int b = 0;
	scanf("%d", &b);
	sa.b = b;

	return 0;
}

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

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

相关文章

【Linux系统编程】30.pthread_exit、pthread_join、pthread_cancel

目录 pthread_exit 参数retval 测试代码1 测试结果 pthread_join 参数thread 参数retvsl 返回值 测试代码2 测试结果 pthread_cancel 参数thread 返回值 测试代码3 测试结果 pthread_exit 退出当前线程。 man 3 pthread_exit 参数retval 退出值。 NULL&#xf…

我用文心4.0给你做了一个“五一旅行助手”!行程规划、实时查询、景区讲解!

大家好&#xff0c;我是木易&#xff0c;一个持续关注AI领域的互联网技术产品经理&#xff0c;国内Top2本科&#xff0c;美国Top10 CS研究生&#xff0c;MBA。我坚信AI是普通人变强的“外挂”&#xff0c;所以创建了“AI信息Gap”这个公众号&#xff0c;专注于分享AI全维度知识…

使用Gradio搭建聊天UI实现质谱AI智能问答

一、调用智谱 AI API 1、获取api_key 智谱AI开放平台网址&#xff1a; https://open.bigmodel.cn/overview 2、安装库pip install zhipuai 3、执行一下代码&#xff0c;调用质谱api进行问答 from zhipuai import ZhipuAIclient ZhipuAI(api_key"xxxxx") # 填写…

Visual studio 2019 编程控制CH341A芯片的USB设备

1、硬件 买了个USB可转IIC、或SPI、或UART的设备&#xff0c;主芯片是CH341A 主要说明USB转SPI的应用&#xff0c;绿色跳线帽选择IIC&SPI&#xff0c;用到CS0、SCK、MOSI、MISO这4个引脚 2、软件 2.1、下载CH341A的驱动 点CH341A官网https://www.wch.cn/downloads/CH34…

OpenCV如何实现背投(58)

返回:OpenCV系列文章目录&#xff08;持续更新中......&#xff09; 上一篇&#xff1a;OpenCV直方图比较(57) 下一篇&#xff1a;OpenCV如何模板匹配(59) 目标 在本教程中&#xff0c;您将学习&#xff1a; 什么是背投以及它为什么有用如何使用 OpenCV 函数 cv::calcBackP…

Mac好用又好看的终端iTerm2 + oh-my-zsh

Mac好用又好看的终端iTerm2 1. iTerm2的下载安装2. oh-my-zsh的安装2.1 官网安装方式2.2 国内镜像源安装方式 3. oh-my-zsh配置3.1 存放主题的路径3.2 存放插件的路径3.3 配置文件路径 1. iTerm2的下载安装 官网下载&#xff1a; iTerm2 2. oh-my-zsh的安装 oh-my-zsh是一…

设备能源数据采集新篇章

在当今这个信息化、智能化的时代&#xff0c;设备能源数据的采集已经成为企业高效运营、绿色发展的重要基石。而今天&#xff0c;我们要向大家介绍的就是一款颠覆传统、引领未来的设备能源数据采集神器——HiWoo Box网关&#xff01; 一、HiWoo Box网关&#xff1a;一站式解决…

C++11:shared_ptr循环引用问题

一、shared_ptr的弊端 struct Listnode {int _val;std::shared_ptr<Listnode> _prev;std::shared_ptr<Listnode> _next;Listnode(int val ):_val(val),_prev(nullptr),_next(nullptr){}~Listnode(){cout << "~Listnode()" << endl;} }; in…

探索未来,开启元宇宙之旅!

一、什么是元宇宙 元宇宙&#xff0c;这个词汇逐渐进入了公众的视野&#xff0c;引发了人们无尽的想象。 首先&#xff0c;元宇宙是什么&#xff1f;元宇宙&#xff0c;顾名思义&#xff0c;是一个虚拟现实的世界&#xff0c;一个融合了数字、物理和社交空间的全息图。它不仅…

【数据结构】位图与布隆过滤器

目录 前言 位图的概念 经典面试题目 位图的模拟实现 set() reset() test() 位图整体代码 位图的应用 位图的优缺点 布隆过滤器 布隆过滤器的概念 哈希函数的个数与布隆过滤器长度的关系 布隆过滤器的模拟实现 插入 查找 删除 布隆过滤器整体代码 前言 哈希本质…

nginx缓存清理

背景 昨天打开我的gpt镜像网站&#xff0c;意外发现静态图片资源全都无法获取了 CoCo-AI 一番排查下来&#xff0c;发现是引用的cdn链接失效了 且cdn源是属于七牛云的&#xff0c;且不再维护&#xff0c;于是果断切换到cloudflare export function getEmojiUrl(unified: str…

iBarcoder for Mac:一站式条形码生成软件

在数字化时代&#xff0c;条形码的应用越来越广泛。iBarcoder for Mac作为一款专业的条形码生成软件&#xff0c;为用户提供了一站式的解决方案。无论是零售、出版还是物流等行业&#xff0c;iBarcoder都能轻松应对&#xff0c;助力用户实现高效管理。 iBarcoder for Mac v3.14…

Android4.4真机移植过程笔记(一)

1、RK源码编译 获取内核源码&#xff1a; git clone git172.28.1.172:rk3188_kernel -b xtc_ok1000 内核编译环境&#xff1a; 从172.28.1.132编译服务器的/data1/ZouZhiPing目录下拷贝toolchain.tar.gz&#xff08;交叉编译工具链&#xff09;并解压到与rk3188_kernel同级目…

计算机英文论文常见错误写作习惯2

目录 第一部分 非常长的句子 在一个句子的主要概念的前面&#xff0c;首先说明目的、地点或原因 将表示时间的短语放在句首的倾向 将最重要的主语放在句首&#xff0c;以示强调 ‘In this paper’, ‘in this study’ 第一部分 非常长的句子 由于作者经常直接从中文翻译…

通过ESXi主机和专业工具导出或导入虚拟机

关于导出虚拟机的用户场景 导出ESXi虚拟机是VMware内置功能之一&#xff0c;可用于数据迁移或作为ESXi备份解决方案。通常情况下&#xff0c;您可以将ESXi中的虚拟机导出为OVF模板&#xff0c;该模板可捕获虚拟机或虚拟设备的状态并存储在一个自包含的包中&#xff0c;其中磁盘…

使 Elasticsearch 和 Lucene 成为最佳向量数据库:速度提高 8 倍,效率提高 32 倍

作者&#xff1a;来自 Elastic Mayya Sharipova, Benjamin Trent, Jim Ferenczi Elasticsearch 和 Lucene 成绩单&#xff1a;值得注意的速度和效率投资 我们 Elastic 的使命是将 Apache Lucene 打造成最佳的向量数据库&#xff0c;并继续提升 Elasticsearch 作为搜索和 RAG&a…

启发式搜索算法4 -遗传算法实战:吊死鬼游戏

相关文章: 启发式搜索算法1 – 最佳优先搜索算法 启发式搜索算法2 – A*算法 启发式搜索算法2 – 遗传算法 有一个小游戏叫吊死鬼游戏&#xff08;hangman&#xff09;&#xff0c;在学习英语的时候&#xff0c;大家有可能在课堂上玩过。老师给定一个英文单词&#xff0c;同学们…

Python人脸识别全面教程

目录 第一部分&#xff1a;人脸识别基础 1.1 人脸检测 1.2 人脸识别算法 1.3 深度学习在人脸识别中的应用 1.4 人脸识别库 第二部分&#xff1a;人脸识别高级技术 2.1 特征提取与人脸编码 人脸编码示例 2.2 人脸识别流程 人脸识别流程示例 2.3 多人脸识别与跟踪 多…

LabVIEW航空发动机主轴承试验器数据采集与监测

LabVIEW航空发动机主轴承试验器数据采集与监测 随着航空技术的迅速发展&#xff0c;对航空发动机性能的测试与监测提出了更高的要求。传统的数据采集与监测方法已难以满足当前高精度和高可靠性的需求&#xff0c;特别是在主轴承试验方面。基于LabVIEW的航空发动机主轴承试验器…

翻译《The Old New Thing》 - How do I cover the taskbar with a fullscreen window?

How do I cover the taskbar with a fullscreen window? - The Old New Thing (microsoft.com)https://devblogs.microsoft.com/oldnewthing/20050505-04/?p35703 Raymond Chen 2005年5月5日 如何用全屏窗口覆盖任务栏&#xff1f; 很多时候&#xff0c;人们总是想得太多。…