【C语言】struct结构体

简介: struct结构体

一. 结构体简述

具有相同或不同类型元素的集合叫做结构体。定义一个结构体,本质是在制作一个类型:


// 声明一个学生信息结构体
struct Student
{
  char name[20];
  int age;
};
int main()
{
  // 定义出两个学生变量
  struct Student s1 = { "张三", 18};
  struct Student s2 = { "李四", 20};
  return 0;
}


二. 结构体的声明和定义

1、简单地声明一个结构体和定义结构体变量

在C中,结构体内只能存放各种类型的变量,不能存函数:

961e6336336f46a78487b499aa81d7f9.png


像上面这样就是声明了一个结构体struct Student,此时的 struct Student 相当于一个类型名。


然后我们可以用这个自己声明的结构体类型去定义变量:

4aa3678072e648e6a7774b3948a7dc8a.png


补充:C 和 C++ 中定义结构体变量的区别


在 C 中使用结构体去定义变量时,需要在结构体名称前加上 struct 关键字。

在 C++ 中使用结构体去定义变量时,可以不加 struct 关键字

82fba15529d8450fb798bee0b8dcc149.png


2、声明结构体的同时也定义结构体变量

e60e2182b6224a70817098323c765e19.png


也许初期看不习惯容易困惑,其实这就相当于两步合并一步:先定义结构体 struct Student,再定义变量 s1 和 s2:


3、匿名结构体

使用方式:声明结构体的时候缺失结构体名,同时定义出一个或n个结构体变量:

787724e6a31b44d3bdbb8a60de416948.png


这种形式只能使用在声明结构体的同时也定义出结构体变量,由于没有结构体名,因此后续不可以再定义新的结构体变量。


4、配合typedef,声明结构体的同时为结构体取别名

6a3953b15ea344adbf2f13d2f60658d5.png


前面说过,使用结构体去定义结构体变量时,C 需要加 struct,C++ 不需要。那么使用结构体的别名去定义变量呢?


答:使用结构体别名去定义结构体变量时,C 和 C++ 都不需要加 struct,加了反而都会报错,因为取别名时把struct连同结构体名称一起包含进去了。


5、在声明匿名结构体时,使用typedef给这个匿名结构体取别名

7f107b4c65634c7698901c28cc20068f.png


这种形式声明了一个缺失结构体名的结构体,但同时使用 typedef 为结构体设置了别名,所以之后我们可以使用这个别名,去定义结构体变量。


三. 结构体变量的初始化

先弄清楚变量初始化和赋值的区别:

struct Student
{
  char name[20];
  int age;
};
int main()
{
  // 变量刚开始创建时给值,这个叫初始化
  struct Student s1 = {"nick", 18};
  // 变量创建后,再对它的值进行操作这个叫赋值
  strcpy(s1.name, "tony");
  s1.age = 24;
  return 0;
}


结构体只能被整体初始化,不能被整体赋值,想要赋值的话只能把成员逐个地取出来再赋值。


4bfc86a7d44a4cd99119a3c7af2196f4.png

补充:数组也是一样的道理:只能整体初始化,不能整体赋值。如果是字符数组想要整体赋值的话,可以使用 strcpy 函数:

4fab402a2d40428aa63cb87e6ef4cb62.png


本人推测结构体和数组不能被整体赋值的原因是:它们内部空间在逻辑上是独立一块块的,所以我们只能对这些独立的空间逐个赋值,而不能整体赋值。


四. 结构体成员的访问方法

我们可以通过变量或变量的地址去访问结构体的成员。

struct Student
{
  char name[20];
  int age;
};
int main()
{
  // 1、通过变量访问结构体成员
  struct Student s;
  strcpy(s.name, "张三");
  s.age = 18;
  // 2、通过指针访问结构体成员
  struct Student* p = &s;
  printf("%s\n", p->name);
  printf("%d\n", p->age);
  return 0;
}
--------结果如下--------
张三
18



为什么结构体会有两种访问方式?


在函数传参(传值、传址)时,会生成临时变量,如果要传的结构体变量太大的话,传值拷贝出来的临时对象也会很大,如果用传地址的方式来传结构体变量地址的话,可以很好的节省空间。


9c8833fe1f6346939809e350fcbdacec.png


当然如果可以直接拿到结构体变量的话,使用变量来访问结构体成员会更直观点。


五. 结构体大小的计算

1. 计算方法

结构体的大小不是结构体元素单纯相加就行的,因为我们现在主流的计算机使用的都是64位字长的CPU,对这类型的CPU取8个字节的数要比取一个字节要高效,也更方便。所以在结构体中每个成员的首地址都是8的整数倍的话,取数据元素时就会相对更高效,这就是内存对齐的由来。每个特定平台上的编译器都有自己的默认“对齐系数”(也叫对齐模数)。程序员可以通过预编译命令#pragma pack(n)来改变这一系数,其中的 n 就是你要指定的“对齐系数”。


但实际每个成员的类型可能是不同的,每个类型对应不同大小,为了更高效地读取结构体变量的成员,结构体的大小要遵循一套对齐规则:


第一个成员在与结构体变量偏移量为0的地址处。(即结构体的首地址处,即对齐到0处)

其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。

结构体的总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍。

如果嵌套了结构体,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。

对齐数 = 该结构体成员变量自身的大小与编译器默认的一个对齐数的较小值。


PS:VS中的默认对齐数为8,不是所有编译器都有默认对齐数,当编译器没有默认对齐数的时候,成员变量的大小就是该成员的对齐数。


2. 普通结构体

第一步:找出每个成员变量的大小将其与编译器的默认对齐数相比较,取其较小值为该成员变量的对齐数

ee70f85ab84349528265477f6a92716b.png


PS:这里使用的是VS编译器,故默认对齐数为8。


第二步:根据每个成员对应的对齐数画出它们在内存中的相对位置

90c8b4154bbe4d70b305aa3d08b30c36.png


第三步:通过最大对齐数决定最终该结构体的大小


通过图我们可以知道,绿色部分(double d成员占用)+红色部分(char c成员占用)+紫色部分(int i成员占用)+红色与紫色之间的白色部分(浪费掉了)总共占用了16个字节的内存空间。


我们需要将它们总共占用的内存空间(16)与结构体成员的最大对齐数(8)相比较,结构体的总大小为最大对齐数的整数倍,此时16正好是8的整数倍,所以该结构体在VS编译器下的大小就16个字节。即创建一个该类型的结构体变量,内存需为其开辟16个字节的内存空间。


PS:大多数情况下,成员变量已经占用的总字节个数并不一定正好为其成员变量中的最大对齐数的整数倍,这时我们需要将其扩大为最大对齐数的整数倍。


3. 包含数组成员的结构体

数组应拆开来看,不能看做一个整体

struct S
{
  char a; //对齐数为1。占1个字节
    char c[5]; //对齐数为1。可看成5个char占5个字节
    int b; //对齐数为4。占4个字节,因为前面所有成员占6个字节,不是4
           //个字节的整数倍,所以在第二个成员和第三个成员
           //之间要补2个字节
} //所以该结构体的大小为1+5+2(补)+4=12个字节


4. 成员包含结构体的结构体

1)如果结构体成员只是说明而没有定义变量,则这个结构体成员不占内存空间。


struct S
{
     char a; //对齐数为1。占1个字节
     struct s
     {
             int c;
             char d;
     }; //此处结构体只声明,没有定义结构体变量,所以该声明
        //的结构体在地址空间中并不占位置
     int f; //对齐数为4。占4个字节
     double b; //对齐数为8,
}; //该结构体的大小为1+3(补)+4+8=16个字节


2)如果内部定义并申明了其他结构体变量,这时需要把这个结构体看成一个整体,大小要独立计算,至于对齐数取其内部最大成员的对齐数。

struct t
{
    char a;  //对齐数1
    struct s //对齐数4
    {
            int c; //对齐数4
            char d;//对齐数1
    }g;//此处定义并申明了结构体变量,在这里需要把结构体
       //看成一个整体,独立计算这个结构体的大小为8字节
       //结构体整体的对齐数是内部最大成员的对齐数
       //之后把这个结构体看出对齐数为4,大小为8的成员
    char f; //对齐数1
    int b;  //对齐数4
}; //所以该结构体的大小为1+3(补)+8+1+3(补)+4=20个字节


5. 成员包含联合体的结构体

联合体的大小等同于联合体里面最大成员的大小,所以可以把联合体等效成一个变量,这个变量就是联合体里面最大的那个成员。


和前文所说的结构体一样,如果只声明联合体,没定义联合体变量,则联合体就当成不存在。


struct t
{
    char a;
    union s
    {
            int c;
            char d;
            double h;
    }g; 
    int f;
    double b;
};//所以该结构体的大小为1+7(补)+8+4+4(补)+8=32个字节


6. 空结构体的大小

1)在 VS2017 下测试


4a31dd73333f4298ae3e5fb89caee644.png

2)在 Centos7 下测试

fec545853ad845a7b3fc853719358eff.png


六. 柔性数组

1. 介绍

在 c99 中有明确的规定允许结构体中最后一个数组大小是未知的。


数组作为结构体的最后一个成员

数组元素可以不写或写成0

结构体中至少包含一个以上处数组外的其他类型的成员

struct T
{
  int a;
  char b;
  int arr[];//或者int arr[0];
};
int main()
{
  struct T t;
  // sizeof 求结构体大小时所求出的大小没有包括柔性数组的大小
  printf("%lu\n", sizeof(struct T));
  return 0;
}
--------结果如下--------
8


2. 使用方法

包含柔性数组的结构体,可以把整个结构体看成是变长的。


#include<stdio.h>   
#include<stdlib.h>                                          
#include<stdlib.h>     
struct d
{                                                                                    
  int nb;  
  int nn;                                 
  int arr[];                 
};                         
int main()                      
{
  //分别给结构体中其他类型的成员和柔性数组申请空间
    struct d *p=(struct d*)malloc(sizeof(struct d)+5*sizeof(int));
    p->nb=100;
    p->nn=50;  
    for(int i=0;i<5;i++)
    {
      p->arr[i]=i;//赋值
      printf("%d ",p->arr[i]);
    }     
    //重新调整所申请的空间,将柔性数组调整为40。
    struct d *pp=(struct d*)realloc(p,48); 
    if(pp!=NULL)
    {
       p=pp;
       for(int i=5;i<10;i++)
       {
         p->arr[i]=i;//赋值
         printf("%d ",p->arr[i]);    
       } 
       free(p);
       p=NULL;
    }                                                                                                    
    return 0;                                                                           
}       
--------结果如下--------
0 1 2 3 4 5 6 7 8 9


3. 柔性数组的特点

柔性数组只需在 malloc 创建时要独立于结构体申请空间,此后的 realloc 再分配空间和 free 释放都只需对一个结构体指针操作即可。

柔性数组申请的内存更加集中,有利于查找使用和减少内存碎片。

sizeof 求结构体大小时所求出的大小没有包括柔性数组的大小。

七. C++ 中 struct 与 class 的区别

class 成员的默认权限为 private,struct 成员的默认权限为 public。

class 的继承默认是 private 继承,struct 的继承默认是 public 继承。

class 可以作为一个关键字定义模板参数(与 typename 作用一样),而struct 不可以。

八. C 和 C++ 结构体的区别

C++ 结构体内部可以有成员变量和成员函数,而 C 中结构体只能有成员变量。

C 结构体的成员变量不能在声明时给初值,而 C++ 中可以

0eef09ecc9d74178bb59d36c0e2a8879.png


C++ 中定义结构变量时,可以不在名称前面加上 struct 关键字,而 C 一定要

C 结构体内不能有静态成员,而 C++ 可以。

C 结构没有访问修饰限定符,而 C++ 有。


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