22.结构体-struct、结构体指针与常规结构体的区别

news/2025/3/30 1:34:58/文章来源:https://www.cnblogs.com/littlecc/p/18795109

C 数组允许定义可存储相同类型数据项的变量,结构体是 C 编程中另一种用户自定义的可用的数据类型,它允许您存储不同类型的数据项。

结构体中的数据成员可以是基本数据类型(如 int、float、char 等),也可以是其他结构体类型、指针类型等。
结构用于表示一条记录,假设您想要跟踪图书馆中书本的动态,您可能需要跟踪每本书的下列属性:

  1. Title
  2. Author
  3. Subject
  4. Book ID
  • 1.定义结构

结构体定义由关键字 struct 和结构体名组成,结构体名可以根据需要自行定义。

truct Books
{char  title[50];char  author[50];char  subject[100];int   book_id;
} book;  

Books:是结构体标签。
book:是结构变量,定义在结构的末尾,最后一个分号之前,您可以指定一个或多个结构变量

在一般情况下,Books、标准的变量定义体、book 这 3 部分至少要出现 2 个。以下为实例:

//此声明声明了拥有3个成员的结构体,分别为整型的a,字符型的b和双精度的c
//同时又声明了结构体变量s1
//这个结构体并没有标明其标签
struct 
{int a;char b;double c;
} s1;//此声明声明了拥有3个成员的结构体,分别为整型的a,字符型的b和双精度的c
//结构体的标签被命名为SIMPLE,没有声明变量
struct SIMPLE
{int a;char b;double c;
};
//用SIMPLE标签的结构体,另外声明了变量t1、t2、t3
struct SIMPLE t1, t2[20], *t3;//也可以用typedef创建新类型
typedef struct
{int a;char b;double c; 
} Simple2;
//现在可以用Simple2作为类型声明新的结构体变量
Simple2 u1, u2[20], *u3;

在上面的声明中,第一个和第二声明被编译器当作两个完全不同的类型,即使他们的成员列表是一样的,如果令 t3=&s1,则是非法的。

结构体的成员可以包含其他结构体,也可以包含指向自己结构体类型的指针,而通常这种指针的应用是为了实现一些更高级的数据结构如链表和树等。

//此结构体的声明包含了其他的结构体
struct COMPLEX
{char string[100];struct SIMPLE a;
};//此结构体的声明包含了指向自己类型的指针
struct NODE
{char string[100];struct NODE *next_node;
};

如果两个结构体互相包含,则需要对其中一个结构体进行不完整声明,如下所示:

struct B;    //对结构体B进行不完整声明//结构体A中包含指向结构体B的指针
struct A
{struct B *partner;//other members;
};//结构体B中包含指向结构体A的指针,在A声明完后,B也随之进行声明
struct B
{struct A *partner;//other members;
};
  • 2.结构体变量的初始化

和其它类型变量一样,对结构体变量可以在定义时指定初始值。

#include <stdio.h>struct Books
{char title[50];char author[50];char subject[100];int book_id;
} book = {"C语言", "RUNOOB", "编程语言", "1234"};int main()
{printf("title : %s\nauthor: %s\nsubject: %s\nbook_id: %d\n", book.title, book.author, book.subject, book.book_id);
}//执行输出结果为:
title : C 语言
author: RUNOOB
subject: 编程语言
book_id: 1234
  • 3.访问结构体成员

为了访问结构的成员,我们使用成员访问运算符(.)

#include <stdio.h>
#include <string.h>//定义结构体类型
struct Books
{char  title[50];char  author[50];char  subject[100];int   book_id;
};int main()
{struct Books Book1; /* 声明 Book1,类型为 Books */struct Books Book2; /* 声明 Book2,类型为 Books *///给结构体变量赋值//这里使用strcpy()函数来赋值字符串strcpy(Book1.title, "C Programming");strcpy( Book1.author, "Nuha Ali");strcpy( Book1.subject, "C Programming Tutorial");Book1.book_id = 67583;/* Book2 详述 */strcpy( Book2.title, "Telecom Billing");strcpy( Book2.author, "Zara Ali");strcpy( Book2.subject, "Telecom Billing Tutorial");Book2.book_id = 6495700;/* 输出 Book1 信息 */printf( "Book 1 title : %s\n", Book1.title);printf( "Book 1 author : %s\n", Book1.author);printf( "Book 1 subject : %s\n", Book1.subject);printf( "Book 1 book_id : %d\n", Book1.book_id);/* 输出 Book2 信息 */printf( "Book 2 title : %s\n", Book2.title);printf( "Book 2 author : %s\n", Book2.author);printf( "Book 2 subject : %s\n", Book2.subject);printf( "Book 2 book_id : %d\n", Book2.book_id);return 0;
}
  • 4.结构体作为函数参数
#include <stdio.h>
#include <string.h>struct Books
{char  title[50];char  author[50];char  subject[100];int   book_id;
};/* 函数声明 */
void printBook( struct Books book );
int main( )
{struct Books Book1;        /* 声明 Book1,类型为 Books */struct Books Book2;        /* 声明 Book2,类型为 Books *//* Book1 详述 */strcpy( Book1.title, "C Programming");strcpy( Book1.author, "Nuha Ali"); strcpy( Book1.subject, "C Programming Tutorial");Book1.book_id = 6495407;/* Book2 详述 */strcpy( Book2.title, "Telecom Billing");strcpy( Book2.author, "Zara Ali");strcpy( Book2.subject, "Telecom Billing Tutorial");Book2.book_id = 6495700;/* 调用printBook函数,传入结构体 输出 Book1 信息 */printBook( Book1 );/* 输出 Book2 信息 */printBook( Book2 );return 0;
}
void printBook( struct Books book )
{printf( "Book title : %s\n", book.title);printf( "Book author : %s\n", book.author);printf( "Book subject : %s\n", book.subject);printf( "Book book_id : %d\n", book.book_id);
}
  • 5.指向结构体的指针

定义指向结构的指针,方式与定义指向其他类型变量的指针相似,如下所示

struct Books *struct_pointer;

现在,您可以在上述定义的指针变量中存储结构变量的地址。为了查找结构变量的地址,请把 & 运算符放在结构名称的前面,如下所示:

struct_pointer = &Book1;

为了使用指向该结构的指针访问结构的成员,您必须使用 -> 运算符,如下所示:

struct_pointer->title;

案例

#include <stdio.h>
#include <string.h>struct Books
{char  title[50];char  author[50];char  subject[100];int   book_id;
};/* 函数声明 */
void printBook( struct Books *book );
int main( )
{struct Books Book1;        /* 声明 Book1,类型为 Books */struct Books Book2;        /* 声明 Book2,类型为 Books *//* Book1 详述 */strcpy( Book1.title, "C Programming");strcpy( Book1.author, "Nuha Ali"); strcpy( Book1.subject, "C Programming Tutorial");Book1.book_id = 6495407;/* Book2 详述 */strcpy( Book2.title, "Telecom Billing");strcpy( Book2.author, "Zara Ali");strcpy( Book2.subject, "Telecom Billing Tutorial");Book2.book_id = 6495700;/* 通过传 Book1 的地址 来输出 Book1 信息 */printBook( &Book1 );  //只需传递地址/* 通过传 Book2 的地址来输出 Book2 信息 */printBook( &Book2 );return 0;
}
void printBook( struct Books *book )
{printf( "Book title : %s\n", book->title);printf( "Book author : %s\n", book->author);printf( "Book subject : %s\n", book->subject);printf( "Book book_id : %d\n", book->book_id);
}


  • 理解
    • void func(struct HugeData *data)
    • void func(struct HugeData data)
// 传指针
void func(struct HugeData *data); // 参数是指针
struct HugeData hd;
func(&hd); // 调用时  ,只需传递地址实际发生的过程:
1.系统仅需要复制hd的内存地址(如0x1000)
2.将这个地址值(通常4/8字节)存入新栈帧
3.函数通过这个地址直接访问原数据
------------------------------------
[调用前内存]
原结构体 hd (地址0x1000-0x2000) 
[4KB数据...][调用时内存]
参数 data (地址0x3000) 
[存储的值是 0x1000] ← 只复制了这个地址值↑└─── 指向原结构体中的值
--------------------------------------//传值
当直接传递结构体时:
void func(struct HugeData data); // 参数是结构体本身
struct HugeData hd;
func(hd); // 调用时,直接传递整个结构体的值,相当于重新开辟了一个新的内存地址,且完整的复制了整个结构体的值,消耗大实际发生的过程:
1.系统在新的栈帧中开辟一块4KB的空间
2.将原结构体hd的每一个字节逐位复制到新位置
3.函数内部操作的是一份完全独立的副本
---------------------------------------
[调用前内存]
原结构体 hd (地址0x1000-0x2000) 
[4KB数据...][调用时内存]
原结构体 hd (地址0x1000-0x2000) 
[4KB数据...]
↓ 完整复制
新结构体 data (地址0x3000-0x4000) 
[完全相同的4KB数据...]
----------------------------------------
  • 类比解释为什么传指针比传值更好

场景:你要给别人一本书的内容

方式 操作 对应编程概念
直接传值 把整本书复印一份给对方 复制整个结构体(4KB)
传指针 告诉对方"书在我家第三层书架" 只传递地址(4字节)

传值:需要消耗大量纸张和时间(内存和CPU周期)
传指针:只需要写一张小纸条(传递地址)

  • 计算机底层视角
  1. 函数调用时的参数传递规则
    所有参数都会被压入栈(stack)或存入寄存器
    传值:需要把整个对象放入栈/寄存器
    传指针:只需放入一个地址值

  2. 典型系统的限制
    系统类型 寄存器大小 典型地址长度
    32位系统 32位(4字节) 4字节
    64位系统 64位(8字节) 8字节

因此:
传4KB结构体 → 需要移动4096字节数据
传指针 → 只需移动4/8字节地址

  • 极端情况验证

假设结构体有100万个int(约4MB):

struct GiantData {int data[1000000]; // 4MB
};// 传值(灾难性)
void process1(struct GiantData d); // 调用时复制4MB// 传指针(高效)
void process2(struct GiantData *d); // 调用时复制8字节

process1():每次调用产生4MB内存拷贝,可能导致:
栈溢出(stack overflow)
严重性能下降

process2():无论结构体多大,永远只复制指针大小

  • 关键结论

传值是复制数据 → 数据多大就复制多少
传指针是复制地址 → 永远只复制4/8字节
指针的本质:是一种"轻量级引用",通过存储目标位置来避免数据搬运

这种设计使得C语言可以:
高效操作任意大小的数据
实现复杂数据结构(如链表、树)
直接与硬件交互(通过内存地址)

  • 结构体大小的计算

C 语言中,我们可以使用 sizeof 运算符来计算结构体的大小,sizeof 返回的是给定类型或变量的字节大小。
对于结构体,sizeof 将返回结构体的总字节数,包括所有成员变量的大小以及可能的填充字节。
以下实例演示了如何计算结构体的大小:

#include <stdio.h>struct Person {char name[20];int age;float height;
};int main() {struct Person person;printf("结构体 Person 大小为: %zu 字节\n", sizeof(person));return 0;
}//输出
结构体 Person 大小为: 28 字节

以上实例中,我们定义了一个名为 Person 的结构体,它包含了一个字符数组 name、一个整数 age 和一个浮点数 height。
在 main 函数中,我们声明了一个 Person 类型的变量 person,然后使用 sizeof 运算符来获取 person 结构体的大小

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

搜维尔科技:SenseGlove触觉反馈手套-自动化和培训的突破

触觉力反馈技术领导者SenseGlove和机器人操控创新者Aeon宣布推出HEART项目。此次合作将虚拟现实 、力反馈触觉手套(SenseGlove)和机器人系统(Aeon)集成在一起,以实现直观控制和实时力反馈,使机器人训练更加方便和有效。 自动化和培训的突破 由于产品需求不断变化以及机器…

搜维尔科技:Haption通用遥控控制器,可轻松集成到工业机器人控制中

TeleRobotics EXtender (TREX) 是一个专为力反馈远程操作而设计的框架。它为操作员必须在危险、具有挑战性或受限的环境中操作的情况提供了一种创新的解决方案,使他们能够在不暴露自己风险的情况下执行任务。借助 TREX,操作员可以手动控制远程机器人,具有很高的灵活性和精确…

Elasticsearch 的搜索功能

Elasticsearch 的搜索功能建议阅读顺序:Elasticsearch 入门 Elasticsearch 搜索(本文)1. 介绍 使用 Elasticsearch 最终目的是为了实现搜索功能,现在先将文档添加到索引中,接下来完成搜索的方法。 查询的分类:叶子查询:叶查询子句在特定字段中查找特定值,例如 match、t…

20242213 实验二《Python程序设计》实验报告

20242213 2024-2025-2 《Python程序设计》实验2报告 课程:《Python程序设计》 班级: 2422 姓名: 刘宗林 学号:20242213 实验教师:王志强 实验日期:2025年3月26日 必修/选修: 公选课 1.实验内容设计并编写一个计算器程序,实现基本运算功能;功能包括加、减、乘、除、取余…

《实战Java高并发程序设计(第3版)》 | PDF免费下载

《实战Java高并发程序设计(第3版)》主要介绍基于Java的并行程序设计基础、思路、方法和实战。第一,立足于并行程序基础,详细介绍Java并行程序设计的基本方法。第二,进一步详细介绍JDK对并行程序的强大支持,帮助读者快速、稳健地进行并行程序开发。第三,详细讨论“锁”的…

《DeepSeek原理与项目实战》 | PDF免费下载

DeepSeek 是一种基于 Transformer 架构的生成式 AI(Artificial Intelligence)大模型,融合了MoE 架构、混合精度训练、分布式优化等先进技术,具备强大的文本生成、多模态处理和任务定制化能力。本书系统性地介绍了开源大模型 DeepSeek-V3 的核心技术及其在实际开发中的深度应…

实现极限网关(INFINI Gateway)配置动态加载

还在停机更新 Gateway 配置,OUT 了。 今天和大家分享一个 Gateway 的功能:动态加载配置(也称热更新或热加载)。 这个功能可以在 Gateway 不停机的情况下更新配置并使之生效。 配置样例如下: path.data: data path.logs: logconfigs:auto_reload: true # set true to auto …

ubuntu20.04安装Synergy

问题 最近在Ubuntu20.04上安装新版本的Synergy遇到一些问题,Synergy最后一个支持ubuntu20.04的版本是v3.1.3-beta,下面是下载地址: https://symless.com/synergy/download/other 在安装的时候遇到下面的问题:提示依赖libssl1,然后尝试安装下面的软件包: sudo apt install…

Netty源码—7.ByteBuf原理二

大纲 9.Netty的内存规格 10.缓存数据结构 11.命中缓存的分配流程 12.Netty里有关内存分配的重要概念 13.Page级别的内存分配 14.SubPage级别的内存分配 15.ByteBuf的回收9.Netty的内存规格 (1)4种内存规格 (2)内存申请单位(1)4种内存规格 一.tiny:表示从0到512字节之间的内存大…

一文速通Python并行计算:03 Python多线程编程-多线程同步(上)—基于互斥锁、递归锁和信号量

在 Python 多线程编程中,线程同步是确保多个线程安全访问共享资源的关键技术。本篇文章介绍了互斥锁(Lock)、递归锁(RLock) 和 信号量(Semaphore) 的概念与应用。互斥锁用于防止多个线程同时修改数据,递归锁适用于嵌套锁定场景,而信号量则限制同时访问资源的线程数。一…

K3D|高效创建轻量级 k8s 集群 (run in dokcer)

k3d 是一个轻量级的包装器,用于在 docker 中运行 k3s(Rancher Lab 的最小 Kubernetes 发行版)什么是 K3S 官网:https://k3s.io/ 指南:https://rancher.com/docs/k3s/latest/en/ 中文版指南:https://docs.rancher.cn/docs/k3s/_index/ K3s 是一个轻量级的 Kubernetes 发行…

记录一下安装适用于intel 显卡的torch

支持的硬件和驱动如下按照表格安装好驱动。使用conda创建好虚拟环境(Python 3.9, 3.10, 3.11, 3.12, 3.13.)后,我的显卡是arc A580# For hardware listed in the table above, other than Intel Core™ Ultra Series 2 Mobile Processors (Arrow Lake-H), use the commands …