C语言结构体

简介: C语言结构体

1.结构的声明


结构体的声明格式:


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


例如描述一个学生:


struct Stu
{
  char name[20];//名字
  int age;//年龄
};


定义一个Stu类型的变量:


struct Stu
{
  char name[20];//名字
  int age;//年龄
}s1,s2; //声明类型的同时定义变量s1,s2,s1,s2是全局变量



int main()
{
  struct Stu s1,s2;//局部变量
}


在定义结构体变量时,都需要写struct这样显得有些麻烦,我们可以用typedef简化一下:

typedef struct Stu
{
  char name[20];
  int age;
}Stu;//Stu是一个类型


这里要注意一点:这里的Stu表示的是一个类型,而前面s1,s2是变量,不要弄混


这样,在后续的定义结构体变量时,就不用写struct了


int main()
{
  Stu s1,s2;
}


特殊的声明

在声明结构的时候,可以不完全的声明,在声明的时候省略掉了结构体标签(tag)

//匿名结构体类型
struct
{
  int a;
  char b;
  float c;
}x;


这样写,只有一个该结构体类型的变量x,在后续中无法再定义其他变量


再定义一个结构体:


struct
{
  int a;
  char b;
  float c;
}*p;


我们可以看到,这两个结构体中成员的类型与顺序都完全一致,那么是否可以把他们两个认为是同一个结构体呢?以及p = &x;正确吗?


答案是:编译器会把上面的两个声明当成完全不同的两个类型。所以p = &x;是非法的。


2.结构体的自引用


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


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


上述代码是否可行呢?


答案是不可行,但从这个结构体的大小方面考虑,一个Node结构体中包含一个int类型和一个Node结构体,而那个被包含的Node中又包含一个int类型和一个Node结构体……

一个包一个,就如套娃一般,这样的话sizeof(struct Node)是无穷大的


所以正确的自引用方式是:


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


在结构体中,包含的是自己结构体类型的指针变量,指针变量的大小只为4/8,所以这样写才正确


注意:

不可以这么写:


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


虽然typedef了,但是程序并不能识别出Node* next;是正确的,因为typedef后的新的标签在结构体的最后

所以只能这么写:


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


3.结构体的初始化

在声明结构体并定义变量的同时初始化


struct Point
{
  int x;
  int y;
}p1 = {1,2};


定义变量时初始化:


struct Point p1 = {1,2};
1

如果有结构体的嵌套:


struct Line
{
  int a;
  struct Point p ;
}l1 = {1,{1,2}};


上述的一些初始化都是按照结构体声明中成员顺序初始化的,也可以不用按照顺序初始化


struct Test
{
  int num;
  char c;
  double d;
  float f;
};
int main()
{
  struct Test t1 = {10,'a',1.0,2.0f};//按顺序定义
  struct Test t2 = {.f = 2.0f,.num = 10,.c = 'a',.d = 1.0};//不按顺序定义
}


4.结构体传参

有一个结构体:


struct S
{
  int data[1000];
  int num;
}s = {{0}, 1000};;


如果要传参,有两种方式,一种是传结构体,一种是传结构体的地址

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


显然是传结构体地址更好


原因是:

函数传参的时候,参数是需要压栈,会有时间和空间上的系统开销。

如果传递一个结构体对象的时候,结构体过大,参数压栈的的系统开销比较大,所以会导致性能的下降。


所以在结构体传参时,要传结构体的地址


目录
相关文章
|
22天前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
108 14
|
26天前
|
存储 编译器 C语言
【C语言】结构体详解 -《探索C语言的 “小宇宙” 》
结构体通过`struct`关键字定义。定义结构体时,需要指定结构体的名称以及结构体内部的成员变量。
121 10
|
1月前
|
存储 数据建模 程序员
C 语言结构体 —— 数据封装的利器
C语言结构体是一种用户自定义的数据类型,用于将不同类型的数据组合在一起,形成一个整体。它支持数据封装,便于管理和传递复杂数据,是程序设计中的重要工具。
|
1月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
130 13
|
1月前
|
存储 编译器 数据处理
C 语言结构体与位域:高效数据组织与内存优化
C语言中的结构体与位域是实现高效数据组织和内存优化的重要工具。结构体允许将不同类型的数据组合成一个整体,而位域则进一步允许对结构体成员的位进行精细控制,以节省内存空间。两者结合使用,可在嵌入式系统等资源受限环境中发挥巨大作用。
62 11
|
1月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
60 4
|
2月前
|
存储 C语言
如何在 C 语言中实现结构体的深拷贝
在C语言中实现结构体的深拷贝,需要手动分配内存并逐个复制成员变量,确保新结构体与原结构体完全独立,避免浅拷贝导致的数据共享问题。具体方法包括使用 `malloc` 分配内存和 `memcpy` 或手动赋值。
82 10
|
2月前
|
存储 大数据 编译器
C语言:结构体对齐规则
C语言中,结构体对齐规则是指编译器为了提高数据访问效率,会根据成员变量的类型对结构体中的成员进行内存对齐。通常遵循编译器默认的对齐方式或使用特定的对齐指令来优化结构体布局,以减少内存浪费并提升性能。
|
2月前
|
编译器 C语言
共用体和结构体在 C 语言中的优先级是怎样的
在C语言中,共用体(union)和结构体(struct)的优先级相同,它们都是用户自定义的数据类型,用于组合不同类型的数据。但是,共用体中的所有成员共享同一段内存,而结构体中的成员各自占用独立的内存空间。
|
2月前
|
存储 C语言
C语言:结构体与共用体的区别
C语言中,结构体(struct)和共用体(union)都用于组合不同类型的数据,但使用方式不同。结构体为每个成员分配独立的内存空间,而共用体的所有成员共享同一段内存,节省空间但需谨慎使用。