结构体类型详细讲解(附带枚举,联合)

news2025/1/16 5:33:08

前言:

如果你还对结构体不是很了解,那么本篇文章将会从 为什么存在结构体,结构体的优点,结构体的定义,结构体的使用与结构体的大小依次介绍,同样会附带枚举与联合体

 

目录

 为什么存在结构体:

 结构体的优点

结构体的定义 

结构体的使用

方法一:

方法二   函数传值打印

方法三  函数传址打印   两种打印方法

第二种是利用->  这里的 ->就相当于(*).

方法四  跟方法三差不多,不利用函数打印 

结构体的typedef 

 结构体的大小计算

 ⾸先得掌握结构体的对⻬规则:

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

 2.其他成员变量要对⻬到某个数字(对⻬数)的整数倍的地址处。对⻬数=编译器默认的⼀个对⻬数与该成员变量⼤⼩的较⼩值。VS 中默认的值为 8

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

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

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

先解决对其数问题

 枚举

 枚举的使用案例

联合体(也叫共用体)


 为什么存在结构体:

当一个整体由多个数据构成时,于是人们就创造了数组,我们用数组来表示这个整体,但是数组有个特点:内部的每一个元素都必须是相同类型的数据。 ☀ 在实际应用中,我们通常需要由不同类型的数据来构成一个整体,比如学生这个整体可以由姓名、年龄、身高等数据构成,这些数据都具有不同的类型,姓名可以是字符串类型,年龄可以是整型,身高可以是浮点型。 ☀ 为此,C语言专门提供了一种构造类型来解决上述问题,这就是结构体,它允许内部的元素是不同类型的

 结构体的优点

 

  • 可以在一个结构中声明不同的数据类型。
  • 相同结构的结构体变量是可以相互赋值的。
  • 结构体的存储方式可以提高CPU对内存的访问速度。
  • 结构体可以将同一对象的多个数据类型存储在一起,方便数据的存储和处理。
  • 可以通过结构体变量名和成员名来访问数据。
  • 可以定义结构体数组,方便批量处理数据。

那么有有优点就会有缺点:

  • 如果项目的复杂性增加,管理所有数据成员就变得很困难 。
  • 对程序中的一个数据结构进行更改需要在其他几个地方进行更改。 所以很难跟踪所有的变化。
  • 结构体需要更多的存储空间,因为它为所有数据成员分配内存,甚至速度更慢。
  • 结构体占用更多存储空间,因为它为所有不同的数据成员提供内存,而联合仅占用最大数据大小参数所需的内存大小,并且与其他数据成员共享相同的内存。

结构体的定义 

 一般结构体定义便是如下

struct stu
{
	//变量成员
};

这里的struct是必不可少的部分,而stu是我们自拟定的,里面便是填充的变量成员

这个结构体就是struct stu类型的

同理下面这个结构体就是struct st类型的

struct st
{
	//变量成员
};

 这时候我们便可以在里面填充成员

比如这样

struct st
{
	int a;
    char b;
    struct st* ps;//也可以填充这个类型的指针
};

那么结构体定义完后,我们开始讲解怎么去使用

结构体的使用

这里的s为变量名 

struct st
{
    int a;
    char b; 
};
int main()
{
    struct st s = { 10,'b' };//初始化
    return 0;
}

 接下来介绍打印

方法一:

struct st
{
    int a;
    char b; 
};
int main()
{
    struct st s = { 10,'b' };//初始化
    printf("%d %c", s.a, s.b);
    return 0;
}

 

方法二   函数传值打印

struct st
{
    int a;
    char b; 
};
void print(struct st ps)
{
    printf("%d %c", ps.a, ps.b);
}
int main()
{
    struct st s = { 10,'b' };//初始化
    print(s);
    return 0;
}

 

方法三  函数传址打印   两种打印方法

第一种的打印是利用解引用

第二种是利用->  这里的 ->就相当于(*).

struct st
{
    int a;
    char b; 
};
void print(struct st* ps)
{
    printf("%d %c\n", (*ps).a, (*ps).b);
    printf("%d %c\n", ps->a, ps->b);
}
int main()
{
    struct st s = { 10,'b' };//初始化
    print(&s);
    return 0;
}

方法四  跟方法三差不多,不利用函数打印 

struct st
{
    int a;
    char b; 
};
int main()
{
    struct st s = { 10,'b' };//初始化
    struct st* ps = &s;
    printf("%d %c\n", (*ps).a, (*ps).b);
    printf("%d %c\n", ps->a, ps->b);
    return 0;
}

结构体的typedef 

 当一个结构体类型的类型名过于长的时候,我们不免过于不想去写,这时候就可以用typedef

(重命名)使用起来很简单

typedef struct st
{
    int a;
    char b; 
}st;
int main()
{
    struct st s = { 10,'b' };//初始化
    //等价于
    st s = { 10,'b' };//
    return 0;
}

 结构体的大小计算

先举例一个代码

代码跑出来这个结构体大小是8 

那么怎么计算的呢?  其原理就是对其原则

 ⾸先得掌握结构体的对⻬规则:

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

 2.其他成员变量要对⻬到某个数字(对⻬数)的整数倍的地址处。对⻬数=编译器默认的⼀个对⻬数与该成员变量⼤⼩的较⼩值。VS 中默认的值为 8

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

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

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

 

我们计算一下这个结构体大小 

struct S1
{
 char c1;
 int i;
 char c2;
};
printf("%d\n", sizeof(struct S1))

先解决对其数问题

char    自身对齐数为1  vs上默认对齐数为8    实际取小的   那么最后为1

int 为4 

 偏移量

在C语言中,偏移量通常指的是结构体中成员变量相对于结构体起始地址的偏移量。假设有一个结构体定义如下:

struct Person {
    int id;
    char name[20];
    int age;
};

假设struct Person p;是一个Person类型的结构体变量,我们可以通过&p.id&p.name&p.age来获取各个成员变量的地址。偏移量就是从结构体起始地址到成员变量地址的字节偏移量。

例如,假设&p表示结构体p的起始地址,&p.id表示id成员变量的地址,那么&p.id - &p就是id相对于结构体起始地址的偏移量。在C语言中,可以使用指针和强制类型转换来计算偏移量,例如:

int offset = (char *)&p.name - (char *)&p;

 这样就可以得到name成员变量相对于结构体p的偏移量。偏移量在一些底层编程中经常用来访问结构体中的特定成员变量,尤其是在处理二进制数据或者硬件寄存器时非常有用。

画图表示如下

 那么我们进行利用图进行计算

typedef struct st
{
    char c1;
    int i;
    char c2; 
}st;
int main()
{
    printf("%zd", sizeof(st));
    return 0;
}

第一个char偏移量为0放于第一个,int放于偏移量为4的整数倍数处  ,最后一个char偏移量为1,放于1的整数倍数处,

用图表示如下

图上的大小一共占了9个字节,

然后我们进行最后一步 最大对齐数为4  而9不是最大对齐数的整数倍   所以要浪费3个字节补充该结构体大小到12;

那么最大怎么算了就可以解答上面那个问题为什么是8

//练习3 
struct S3
{
 double d;
 char c;
 int i;
};
printf("%d\n", sizeof(struct S3));
//练习4-结构体嵌套问题 
struct S4
{
 char c1;
 struct S3 s3;
 double d;
};
printf("%d\n", sizeof(struct S4));

答案分别是16    32         

 枚举

 在《C语言深度剖析》这本书中留有一个问题,枚举变量的大小是多少?

大家猜一下大小为多少

enum Color
{
    GREEN = 1,
    RED,
    BLUE,
    GREEN_RED = 10,
    GREEN_BLUE,
    sss,
}c;
int main()
{
    printf("%zd", sizeof(c));
}

答案是4,为啥呢?

因为,枚举变量的取值为花括号内的任意一个值(有且只能有其中一个值),而这个值是int型的,在X86系统中,int型的数据占内存4个字节。所以sizeof(c) = 4,也就是枚举变量的值为4。

 枚举的使用案例

当我们写菜单的时候,会用到switch语句

当我们写case是一般会用到case 1: case 2:

但如果功能一旦多我们就会分不清该case  要实现哪一项功能

这时候就可以用到枚举  就比如通讯录

enum option
{
	EXIT,//等价于0
	ADD,//1
	DEL,//2
	SEARCH,//3
	MODIFY,//4
	SHOW, //5
	SORT,//6
	SAVE,//7
};

void menu()
{
	printf("**************************************\n");
	printf("*****    1.add        2.del      *****\n");
	printf("*****    3.search     4.modify   *****\n");
	printf("*****    5.show       6sort      *****\n");
	printf("*****    7.save       0.exit     *****\n");
	printf("**************************************\n");
}
int main()
{
	do {
		menu();
		printf("请输入你的选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			Addcontact(&con);
			break;
		case DEL:
			Delcontact(&con);
			break;
		case SEARCH:
			Searchcontact(&con);
			break;
		case MODIFY:
			Modifycontact(&con);
			break;
		case SHOW:
			Showcontact(&con);
			break;
		case SORT:
			SortContact(&con);
			break;
		case SAVE:
			SaveContact(&con);
			break;
		case EXIT:
			SaveContact(&con);
			DestroyContact(&con);
			printf("退出通讯录\n");
			break;
		default:
			printf("输入错误,请重新输入:>\n");
			break;
		}

	} while (input);
	return 0;
}

联合体(也叫共用体)

 联合的成员是共⽤同⼀块内存空间的,这样⼀个联合变量的⼤⼩,⾄少是最⼤成员的⼤⼩(因为联合 ⾄少得有能⼒保存最⼤的那个成员)。

那么看以下代码

发现三者的空间是共用一块空间,所以这也对应了名字共用体,公用一块空间

 

#include <stdio.h>
union Un
{
 char c;
 int i;
};
int main()
{
 //联合变量的定义 
 union Un un = {0};
 un.i = 0x11223344;
 un.c = 0x55;
 printf("%x\n", un.i);
 return 0;
}

既然公用一块空间那么(小端)

这个代码运行结果如何呢?

我们发现将i的第4个字节的内容修改为55了(其实是改的低地址处的数据)

同样大端下会打印55223344; 

利用联合体写一个函数判断机器为大端还是小端

int check_sys()
{
 union
 {
 int i;
 char c;
 }un;
 un.i = 1;
 return un.c;//返回1是⼩端,返回0是⼤端 
}

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

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

相关文章

机器学习周报第34周

目录 摘要Abstract一、CNN复习二、目标检测2.1 背景2.2 目标检测发展脉络2.2.1传统目标检测算法2.2.2 Anchor-Based中的Two-stage目标检测算法 三、文献阅读&#xff1a;Faster R-CNN: Towards Real-Time Object Detection with Region Proposal Networks3.1 摘要3.2 背景3.3 解…

day06vue2学习

day06 路由的封装抽离 问题&#xff1a;所有的路由配置都堆在main.js中不太合适么&#xff1f;不好&#xff0c;会加大代码的复杂度 目标&#xff1a;将路由模块抽离出来。好处&#xff1a;差分模块&#xff0c;利于维护。 大致的做法就是&#xff0c;将路由相关的东西都提…

关于使用TCP-S7协议读写西门子PLC字符串的问题

我们可以使用TCP-S7协议读写西门子PLC&#xff0c; 比如PLC中定义一个String[50] 的地址DB300.20 地址DB300.20 DB块编号为300&#xff0c;偏移量【地址】是30 S7协议是西门子PLC自定义的协议&#xff0c;默认端口102&#xff0c;本质仍然是TCP协议的一种具体实现&#xff…

Linux的一些基本指令

​​​​​​​ 目录 前言&#xff1a; 1.以指令的形式登录 2.ls指令 语法&#xff1a; 功能&#xff1a; 常用选项&#xff1a; 3.pwd指令 4.cd指令 4.1 绝对路径与相对路径 4.2 cd .与cd ..&#xff08;注意cd后先空格&#xff0c;然后两个点是连一起的&#xff0…

数据库范式拆分实战

函数依赖 如果给定一个X&#xff0c;能唯一确定一个Y&#xff0c;就称X确定Y&#xff0c;或者说Y依赖于X&#xff0c;例如Y X*X函数。 X -> Y&#xff08;X确定Y&#xff0c;Y依赖于X&#xff09; 部分函数依赖 A可确定C&#xff0c;&#xff08;A&#xff0c;B&#xff09…

基于深度学习的生活垃圾智能分类系统(微信小程序+YOLOv5+训练数据集+开题报告+中期检查+论文)

摘要 本文基于Python技术&#xff0c;搭建了YOLOv5s深度学习模型&#xff0c;并基于该模型研发了微信小程序的垃圾分类应用系统。本项目的主要工作如下&#xff1a; &#xff08;1&#xff09;调研了移动端垃圾分类应用软件动态&#xff0c;并分析其优劣势&#xff1b;…

stm32使用定时器实现PWM与呼吸灯

PWM介绍 STM32F103C8T6 PWM 资源&#xff1a; 高级定时器&#xff08; TIM1 &#xff09;&#xff1a; 7 路 通用定时器&#xff08; TIM2~TIM4 &#xff09;&#xff1a;各 4 路 例如定时器2 PWM 输出模式&#xff1a; PWM 模式 1 &#xff1a;在 向上计数 时&#xff0…

Linux之git

一、什么叫做版本控制 版本控制&#xff08;Revision control&#xff09;是一种在开发的过程中用于管理我们对文件、目录或工程等内容的修改历史&#xff0c;方便查看更改历史记录&#xff0c;备份以便恢复以前的版本的软件工程技术。简单来说就是用于管理多人协同开发项目的技…

【链表】Leetcode 142. 环形链表 II【中等】

环形链表 II 给定一个链表的头节点 head &#xff0c;返回链表开始入环的第一个节点。 如果链表无环&#xff0c;则返回 null。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系…

YoloV8改进策略:BackBone改进|PKINet

摘要 PKINet是面向遥感旋转框的主干,网络包含了CAA、PKI等模块,给我们改进卷积结构的模型带来了很多启发。本文,使用PKINet替代YoloV8的主干网络,实现涨点。PKINet是我在作者的模型基础上,重新修改了底层的模块,方便大家轻松移植到YoloV8上。 论文:《Poly Kernel Ince…

zabbix企业微信的告警媒介配置

简介&#xff1a; Zabbix企业微信告警媒介可用于向特定群组成员发送提醒通知。 前提条件&#xff1a; 完成Zabbix告警平台的搭建后&#xff0c;需将群机器人添加至告警提醒群中。 企业微信群聊——右上角三个点——添加群机器人 保存好产生的webhook地址&#xff08;注意&…

Kotlin的lateinit关键词

Kotlin的lateinit关键词 lateinit&#xff0c;延迟初始化。有时&#xff0c;并不能定义一个变量或对象值为空&#xff0c;而也没办法在对象或变量在定义声明时就为它赋值初始化&#xff0c;那么这时就需要用到Kotlin提供的延迟初始化lateinit。比如&#xff0c;有些依赖注入框架…

Docker可视化管理工具DockerUI

什么是 DockerUI &#xff1f; DockerUI 是一款开源的、强大的、轻量级的 Docker 管理工具。DockerUI 覆盖了 docker cli 命令行 95% 以上的命令功能&#xff0c;通过可视化的 Web 界面操作&#xff0c;可以非常方便、轻松进行 docker 环境和 docker swarm 集群环境的管理和维护…

二分算法(查找)

问题&#xff1a;在数组中查找某一个数字x4的下标 例&#xff1a;arr:1 3 4 6 10 20 21 22 显然&#xff0c;数字4的下标为3。 1、线性查找&#xff0c;一个个地去遍历&#xff0c;时间复杂度为O(n) 2、二分查找&#xff0…

Centos7虚拟机中oracle19c数据库安装

目录[-] 1. Centos7虚拟机中oracle19c数据库安装 1.1. 1.先诀条件1.2. 2.oracle19c安装准备(root用户下执行)1.3. 3.CentOS7上安装oracle19c 1.先诀条件 本文在安装oracle19c时,各项oracle配置操作都通过图形界面进行。因此CentOS7系统需要安装gnome图形程序,虚拟机安装时未…

【自编码器】梳理(上)

every blog every motto: You can do more than you think. https://blog.csdn.net/weixin_39190382?typeblog 0. 前言 梳理有关自编码器 1. 自编码器 1.1 原理 Auto-Encoder&#xff0c;中文称作自编码器&#xff0c;是一种无监督式学习模型。利用输入数据 X X X本身作…

苍穹外卖笔记

苍穹外卖 DAY01nginx反向代理MD5加密yapi进行接口导入Swagger介绍 DAY02新增员工需求分析和设计写相关代码测试(1. 后端文档测试 2. 前后端联调代码完善 员工分页查询DAY01 02涉及到的知识 DAY01 nginx反向代理 MD5加密 拓展&#xff1a;spring security jwt 提供了更强大灵…

实战高效RPC方案在嵌入式环境中的应用与揭秘

实战高效RPC方案在嵌入式环境中的应用与揭秘 开篇 在嵌入式系统开发中&#xff0c;大型项目往往采用微服务架构来构建&#xff0c;其核心思想是将一个庞大的单体应用分割成一系列小型、独立、松耦合的服务模块&#xff0c;这些模块可以是以线程或进程形式存在的多个服务单元。…

C语言动态内存的管理

前言 本篇博客就来探讨一下动态内存&#xff0c;说到内存&#xff0c;我们以前开辟空间大小都是固定的&#xff0c;不能调整这个空间大小&#xff0c;于是就有动态内存&#xff0c;可以让我们自己选择开辟多少空间&#xff0c;更加方便&#xff0c;让我们一起来看看动态内存的有…

yolov5训练并生成rknn模型部署在RK3588开发板上,实现NPU加速推理

简介 RK3588是瑞芯微&#xff08;Rockchip&#xff09;公司推出的一款高性能、低功耗的集成电路芯片。它采用了先进的28纳米工艺技术&#xff0c;并配备了八核心的ARM Cortex-A76和Cortex-A55处理器&#xff0c;以及ARM Mali-G76 GPU。该芯片支持多种接口和功能&#xff0c;适…