当前位置: 首页 > article >正文

【重新认识C语言----结构体篇】

目录

-----------------------------------------begin-------------------------------------

引言

1. 结构体的基本概念

1.1 为什么需要结构体?

1.2 结构体的定义

2. 结构体变量的声明与初始化

2.1 声明结构体变量

2.2 初始化结构体变量

3. 结构体成员的访问

3.1 使用点运算符(.)

3.2 结构体指针与箭头运算符(->)

4. 结构体的内存布局

4.1 内存对齐规则

4.2 手动控制内存对齐

5. 结构体的高级用法

5.1 结构体数组

5.2 嵌套结构体

5.3 结构体与函数

5.4 使用typedef简化结构体类型名

6. 结构体与联合体(Union)的区别

7. 综合示例:学生管理系统

8. 常见问题与注意事项

结语

路过的佬们点点关注哦~

你们的鼓励是我前进的动力~

-------------------------------------------end-------------------------------------


-----------------------------------------begin-------------------------------------

引言

在C语言编程中,结构体(Structure) 是一种非常重要的复合数据类型。它允许开发者将多个不同类型的变量组合成一个逻辑单元,从而更高效地管理复杂数据。无论是实现链表、树等数据结构,还是描述现实世界中的实体(如学生、商品等),结构体都扮演着核心角色。本文将详细讲解结构体的定义、使用及高级特性,帮助读者彻底掌握这一关键概念。


1. 结构体的基本概念

1.1 为什么需要结构体?

假设我们需要描述一个学生的信息,包括姓名(字符串)、学号(整数)、年龄(整数)和成绩(浮点数)。如果单独使用多个变量来存储这些信息,代码会变得冗长且难以维护。例如:

char name[20];
int id;
int age;
float score;

结构体可以将这些变量封装为一个整体,使数据管理更加清晰。

1.2 结构体的定义

结构体通过 struct 关键字定义,语法如下:

struct 结构体名 {数据类型 成员1;数据类型 成员2;// ...
};

示例:定义一个学生结构体

struct Student {char name[20];int id;int age;float score;
};

2. 结构体变量的声明与初始化

2.1 声明结构体变量

结构体定义后,可以通过两种方式声明变量:

  1. 在定义结构体时直接声明变量

    struct Student {// 成员定义
    } stu1, stu2;  // 直接声明变量stu1和stu2

  2. 通过结构体类型名声明变量

    struct Student stu3;  // 声明一个Student类型的变量stu3

    2.2 初始化结构体变量

    结构体变量可以在声明时初始化,或通过赋值操作初始化。
    示例:

// 声明时初始化
struct Student stu1 = {"Alice", 1001, 18, 90.5};// 单独赋值初始化
struct Student stu2;
strcpy(stu2.name, "Bob");
stu2.id = 1002;
stu2.age = 19;
stu2.score = 85.0;

3. 结构体成员的访问

3.1 使用点运算符(.

通过 . 运算符可以直接访问结构体变量的成员:

printf("学生姓名:%s\n", stu1.name);
printf("学号:%d\n", stu1.id);

3.2 结构体指针与箭头运算符(->

如果通过指针访问结构体成员,需使用 -> 运算符:

struct Student *pStu = &stu1;
printf("年龄:%d\n", pStu->age);  // 等价于 (*pStu).age

4. 结构体的内存布局

4.1 内存对齐规则

为了提高内存访问效率,结构体的成员在内存中遵循对齐规则:

  • 对齐数(Alignment):成员的大小与编译器默认对齐数中的较小值。

  • 起始地址:每个成员的起始地址必须是对齐数的整数倍。

  • 总大小:结构体的总大小是对齐数最大值的整数倍。

示例:

struct Example {char c;     // 1字节,对齐数为1int i;      // 4字节,对齐数为4double d;   // 8字节,对齐数为8
};

内存布局分析:

  • char c 占1字节,起始地址0。

  • int i 对齐数为4,起始地址需是4的倍数,因此填充3字节(地址1~3),起始地址4。

  • double d 对齐数为8,起始地址8。
    总大小 = 1 + 3(填充) + 4 + 8 = 16字节。

4.2 手动控制内存对齐

通过 #pragma pack(n) 可以修改默认对齐数(n为1、2、4、8等):

#pragma pack(1)  // 设置为1字节对齐
struct PackedExample {char c;int i;double d;
};
#pragma pack()   // 恢复默认对齐

此时结构体总大小为 1 + 4 + 8 = 13字节。

5. 结构体的高级用法

5.1 结构体数组

结构体数组用于存储多个相同类型的结构体变量。
示例:

struct Student students[3] = {{"Alice", 1001, 18, 90.5},{"Bob", 1002, 19, 85.0},{"Charlie", 1003, 20, 92.0}
};

5.2 嵌套结构体

结构体可以嵌套其他结构体作为成员。
示例:

struct Date {int year;int month;int day;
};struct Employee {char name[20];struct Date birthday;
};

5.3 结构体与函数

结构体可以作为函数参数或返回值传递。
示例:

// 函数参数:按值传递
void printStudent(struct Student stu) {printf("姓名:%s\n", stu.name);
}// 函数参数:按指针传递(推荐,避免内存拷贝)
void updateScore(struct Student *pStu, float newScore) {pStu->score = newScore;
}// 函数返回结构体
struct Student createStudent() {struct Student stu = {"David", 1004, 21, 88.5};return stu;
}

5.4 使用typedef简化结构体类型名

通过 typedef 可以为结构体定义别名,简化代码:

typedef struct Student {// 成员定义
} Student;  // 别名// 声明变量
Student stu4;

6. 结构体与联合体(Union)的区别

结构体与联合体(union)的区别在于内存分配方式:

  • 结构体:每个成员拥有独立的内存空间,总大小为所有成员大小之和(考虑对齐)。

  • 联合体:所有成员共享同一块内存空间,总大小等于最大成员的大小。

7. 综合示例:学生管理系统

#include <stdio.h>
#include <string.h>typedef struct Student {char name[20];int id;float score;
} Student;void printStudent(const Student *stu) {printf("姓名:%s\t学号:%d\t成绩:%.1f\n", stu->name, stu->id, stu->score);
}int main() {Student students[3];// 输入学生信息for (int i = 0; i < 3; i++) {printf("输入第%d个学生的姓名、学号和成绩:", i+1);scanf("%s %d %f", students[i].name, &students[i].id, &students[i].score);}// 输出学生信息printf("\n学生列表:\n");for (int i = 0; i < 3; i++) {printStudent(&students[i]);}return 0;
}

8. 常见问题与注意事项

  • 结构体赋值

    结构体变量可以直接赋值给同类型的变量(按值拷贝):

Student stu1 = {"Alice", 1001, 90.5};
Student stu2 = stu1;  // 合法,内存拷贝
  • 结构体大小计算

    使用 sizeof 运算符获取结构体大小,但需注意内存对齐的影响。

  • 结构体与动态内存分配

    结构体指针可以结合 malloc 实现动态内存分配:

Student *pStu = (Student*)malloc(sizeof(Student));
free(pStu);

结语

结构体是C语言中组织复杂数据的核心工具,其灵活性和高效性使其在系统编程、嵌入式开发等领域广泛应用。通过本文的学习,读者应能够熟练定义、操作结构体,并理解其底层内存布局。建议结合实际项目练习,进一步巩固这一重要概念。

路过的佬们点点关注哦~
你们的鼓励是我前进的动力~

-------------------------------------------end-------------------------------------

相关文章:

【重新认识C语言----结构体篇】

目录 -----------------------------------------begin------------------------------------- 引言 1. 结构体的基本概念 1.1 为什么需要结构体&#xff1f; 1.2 结构体的定义 2. 结构体变量的声明与初始化 2.1 声明结构体变量 2.2 初始化结构体变量 3. 结构体成员的访…...

解决错误:CondaHTTPError: HTTP 000 CONNECTION FAILED for url

解决错误&#xff1a;CondaHTTPError: HTTP 000 CONNECTION FAILED for url 查看channels:vim ~/.condarcshow_channel_urls: true channels:- http://mirrors.tuna.tsinghua.edu.cn/anaconda/cloud/conda-forge/- http://mirrors.tuna.tsinghua.edu.cn/anaconda/cloud/msys2/…...

使用令牌桶算法通过redis实现限流

令牌桶算法是一种常用的限流算法&#xff0c;它可以平滑地控制请求的处理速率。在 Java 中结合 Redis 实现令牌桶算法&#xff0c;可以利用 Redis 的原子操作来保证多节点环境下的限流效果。 一 实现思路 初始化令牌桶&#xff1a;在 Redis 中存储令牌桶的相关信息&#xff0…...

Docker的进程和Cgroup概念

Docker的进程和Cgroup概念 容器里的进程组织或关系0号进程&#xff1a;containerd-shim1号进程&#xff1a;容器内的第一个进程进程收到信号后的三种反应两个特权信号在容器内执行 kill 命令的行为 Cgroup 介绍CPU Cgroup 中与 CFS 相关的参数Kubernetes 中的资源管理memory cg…...

Day68:类的多态

在面向对象编程(OOP)中,多态(Polymorphism)是指不同类的对象对同一消息作出响应的能力。换句话说,多态允许不同类的对象使用相同的方法名,但实现不同的行为。多态是通过继承和方法重写来实现的,通常可以分为方法重写和接口重载。 在 Python 中,多态常常通过方法重写来…...

一种解决SoC总线功能验证完备性的技术

1. 前言 通过总线将各个IP通过总线连接起来的SoC芯片是未来的大趋势&#xff0c;也是缩短芯片开发周期&#xff0c;抢先进入市场的常用方法。如何确保各个IP是否正确连接到总线上&#xff0c;而且各IP的地址空间分配是否正确&#xff0c;是一件很棘手的事情。本文提出了一种新…...

【Linux系统】线程:线程库 / 线程栈 / 线程库源码阅读学习

一、线程库 1、线程库介绍&#xff1a;命名与设计 命名&#xff1a;线程库通常根据其实现目的和平台特性进行命名。例如&#xff0c;POSIX标准定义了Pthreads&#xff08;POSIX Threads&#xff09;&#xff0c;这是一个广泛使用的线程库规范&#xff0c;适用于多种操作系统。此…...

深度剖析 Redis:缓存穿透、击穿与雪崩问题及实战解决方案

一、缓存基本使用逻辑 在应用程序中&#xff0c;为了提高数据访问效率&#xff0c;常常会使用缓存。一般的缓存使用逻辑是&#xff1a;根据 key 去 Redis 查询是否有数据&#xff0c;如果命中就直接返回缓存中的数据&#xff1b;如果缓存不存在&#xff0c;则查询数据库&#…...

如何使用el-table的多选框

对el-table再次封装&#xff0c;使得功能更加强大&#xff01; 本人在使用el-table时&#xff0c;因为用到分页&#xff0c;导致上一页勾选的数据在再次返回时&#xff0c;没有选中&#xff0c;故在原有el-table组件的基础之上再次进行了封装。 1.首先让某些不需要勾选的列表进…...

【工具变量】上市公司企业渐进式创新程度及渐进式创新锁定数据(1991-2023年)

测算方式&#xff1a; 参考顶刊《经济研究》孙雅慧&#xff08;2024&#xff09;老师的做法&#xff0c;用当期创新和往期创新的内容重叠度作为衡量渐进式创新程度的合理指标。通过搜集海量专利摘要&#xff0c;测算当前专利申请和既有专利的内容相似度&#xff0c;反映企业在…...

LM Studio 部署本地大语言模型

一、下载安装 1.搜索&#xff1a;lm studio LM Studio - Discover, download, and run local LLMs 2.下载 3.安装 4.更改成中文 二、下载模型(软件内下载) 1.选择使用代理&#xff0c;否则无法下载 2.更改模型下载目录 默认下载位置 C:\Users\用户名\.lmstudio\models 3.搜…...

嵌入式工程师面试经验分享与案例解析

嵌入式工程师岗位受到众多求职者的关注。面试流程严格&#xff0c;技术要求全面&#xff0c;涵盖C/C编程、数据结构与算法、操作系统、嵌入式系统开发、硬件驱动等多个方向。本文将结合真实案例&#xff0c;深入剖析嵌入式工程师的面试流程、常见问题及应对策略&#xff0c;帮助…...

英特尔至强服务器CPU销量创14年新低,AMD取得进展

过去几年是英特尔56年历史上最艰难的时期之一。该公司在晶圆代工、消费级处理器和服务器芯片等各个领域都面临困境。随着英特尔重组其晶圆代工业务&#xff0c;新的分析显示其服务器业务的现状和未来前景不容乐观。 英特尔最近发布的10-K文件显示&#xff1a;“数据中心和人工…...

判断您的Mac当前使用的是Zsh还是Bash:echo $SHELL、echo $0

要判断您的Mac当前使用的是Zsh还是Bash&#xff0c;可以使用以下方法&#xff1a; 查看默认Shell: 打开“终端”应用程序&#xff0c;然后输入以下命令&#xff1a; echo $SHELL这将显示当前默认使用的Shell。例如&#xff0c;如果输出是/bin/zsh&#xff0c;则说明您使用的是Z…...

使用Springboot实现MQTT通信

目录 一、MQ协议 MQTT 特点 MQTT 工作原理 MQTT 主要应用场景 MQTT 配置与注意事项 二、MQTT服务器搭建 三、参考案例 MQTT&#xff08;Message Queuing Telemetry Transport&#xff09;是一种基于发布/订阅模型的轻量级消息传输协议&#xff0c;常用于物联网&#xff…...

植物大战僵尸融合版(电脑/安卓)

《植物大战僵尸融合版》是一款由B站UP主“蓝飘飘fly”制作的同人策略塔防游戏&#xff0c;基于经典《植物大战僵尸》玩法&#xff0c;加入了独特的植物融合系统。 出于方便&#xff0c;软件是便携版&#xff0c;解压后双击即可畅玩。 游戏主页依旧是植物大战僵尸经典界面。右下…...

02DevOps基础环境准备

准备两台Linux的操作系统&#xff0c;最简单的方式就是在本机上使用虚拟机搭建两个操作系统&#xff08;实际生产环境是两台服务器&#xff0c;虚拟机的方式用于学习使用&#xff09; 我搭建的两台服务器的ip分别是192.168.1.10、192.168.1.11 192.168.1.10服务器用于安装doc…...

苍穹外卖-day12(工作台、数据导出)

工作台Apache POI导出运营数据Excel报表 功能实现&#xff1a;工作台、数据导出 工作台效果图&#xff1a; 数据导出效果图&#xff1a; 在数据统计页面点击数据导出&#xff1a;生成Excel报表 1. 工作台 1.1 需求分析和设计 1.1.1 产品原型 工作台是系统运营的数据看板&…...

说一下 Tcp 粘包是怎么产生的?

TCP 粘包是什么&#xff1f; TCP 粘包&#xff08;TCP Packet Merging&#xff09; 是指多个小的数据包在 TCP 传输过程中被合并在一起&#xff0c;接收方读取时无法正确分辨数据边界&#xff0c;导致数据解析错误。 TCP 是流式协议&#xff0c;没有数据包的概念&#xff0c;…...

详解享元模式

引言 在计算机中&#xff0c;内存是非常宝贵的资源&#xff0c;而程序中可能会有大量相似或相同的对象&#xff0c;它们的存在浪费了许多空间。而享元模式通过共享这些对象&#xff0c;从而解决这种问题的。 1.概念 享元模式(Flyweight Pattern)&#xff1a;运用共享技术有效地…...

第18章 不可变对象设计模式(Java高并发编程详解:多线程与系统设计)

1.线程安全 所谓共享的资源&#xff0c;是指在多个线程同时对其进行访问的情况下&#xff0c;各线程都会使其发生变化&#xff0c;而线程安全性的主要目的就在于在受控的并发访问中防止数据发生变化。除了使用synchronized关键字同步对资源的写操作之外&#xff0c; 还可以在线…...

openEuler22.03LTS系统升级docker至26.1.4以支持启用ip6tables功能

本文记录了openEuler22.03LTS将docker升级由18.09.0升级至26.1.4的过程&#xff08;当前docker最新版本为27.5.1&#xff0c;生产环境为保障稳定性&#xff0c;选择升级到上一个大版本26的最新小版本&#xff09;。 一、现有环境 1、系统版本 [rootlocalhost opt]# cat /etc…...

< OS 有关 > Ubuntu 版本升级 实践 24.04 -> 24.10, 安装 .NET

原因&#xff1a; 想安装 .NET 9 去编译 GitHut 项目&#xff0c;这回用不熟悉的 Ubuntu来做&#xff0c;不知道怎么拐去给 Ubuntu 升级&#xff0c;看到现在版本是 24.10 但不是 LTS 版本&#xff0c;记录下升级过程。 一、实践过程&#xff1a; 1. 查看当前版本 命令1: l…...

某咨询大数据解决方案介绍(32页PPT)

本文档介绍了一个大数据平台解决方案&#xff0c;旨在解决企业当前面临的数据问题&#xff0c;包括数据定义缺失、重复采集和存储、数据不完整以及缺乏可靠决策依据等。通过引入大数据技术&#xff0c;该方案强调从被动的IT支撑向主动的数据核心服务转型&#xff0c;以实现科学…...

ZooKeeper作为注册中心有什么问题? ZooKeeper作为注册中心,海量服务同时重启有什么问题?

目录 ZooKeeper作为注册中心存在的问题 性能瓶颈 一致性保证 复杂性 扩展性 单点故障 数据模型限制 社区和生态 安全性 总结 ZooKeeper作为注册中心,海量服务同时重启有的问题 1. ZooKeeper集群压力剧增 2. ZooKeeper Leader节点压力 3. 会话和临时节点管理 4.…...

matlab simulink 汽车四分之一模型主动被动悬架-LQR

1、内容简介 略 matlab simulink 可以交流、咨询、答疑 124- 2、内容说明 略汽车悬架系统由弹性元件、导向元件和减振器组成,是车身与车轴之间连接的所有组合体零件的总称,也是车架(或承载式车身)与车桥(或车轮)之间一切力传递装置的总称,其主要功能是使车轮与地面有很好的…...

从零开始:OpenCV 图像处理快速入门教程

文章大纲 第1章 OpenCV 概述 1.1 OpenCV的模块与功能  1.2 OpenCV的发展 1.3 OpenCV的应用 第2章 基本数据类型 2.1 cv::Vec类 2.2 cv&#xff1a;&#xff1a;Point类 2.3 cv&#xff1a;&#xff1a;Rng类 2.4 cv&#xff1a;&#xff1a;Size类 2.5 cv&#xff1a;&…...

基础相对薄弱怎么考研

复习总体规划 明确目标 选择专业和院校&#xff1a;根据你的兴趣、职业规划和自身实力&#xff0c;选择适合自己的专业和院校。可以参考往年的分数线、报录比、复试难度等。了解考试科目&#xff1a;不同专业考试科目不同&#xff0c;一般包括&#xff1a; 公共课&#xff1a…...

强化学习笔记6——异同策略、AC、等其他模型总结

异步两种方法&#xff1a;1&#xff1a;经验回放 2&#xff1a;数据动作非同时产生 举例QLearning为什么是异策略&#xff1f; 生成动作时e的概率从Q表选&#xff0c;1-e概况随机。 更新策略时&#xff0c;贪心策略选择Q_max作为动作。 策略优化两种主要方法&#xff1a;基于梯…...

Linux提权--passwd提权

passwd​ 命令用于更改用户密码。在 Linux 系统中&#xff0c;普通用户可以通过 passwd​ 更改自己的密码&#xff0c;但如果攻击者能够以某种方式执行 passwd​ 命令更改 root 用户的密码&#xff0c;他们就能获取 root 权限。 1.常见的 passwd 提权方法 SUID 设置&#xff1…...