C语言程序设计核心详解 第九章 结构体与链表概要详解

简介: 本文档详细介绍了C语言中的结构体与链表。首先,讲解了结构体的定义、初始化及使用方法,并演示了如何通过不同方式定义结构体变量。接着,介绍了指向结构体的指针及其应用,包括结构体变量和结构体数组的指针操作。随后,概述了链表的概念与定义,解释了链表的基本操作如动态分配、插入和删除。最后,简述了共用体类型及其变量定义与引用方法。通过本文档,读者可以全面了解结构体与链表的基础知识及实际应用技巧。

第九章 结构体与链表概要

@[toc]

1.结构体类型

struct 结构体类型名
{
   
   
    成员1的定义;
    成员2的定义;
    .........
    成员n的定义;
}结构体名(可以省略);

1.1 构造与定义结构体类型

构造结构体一共有三种方法
方法一:

struct student
{
   
   
    int sn;
    int age;
    char sex;
    int s[3];
};

int main()
{
   
   
    struct student a;
    struct student b;
    struct student c[10];

}

方法二:

struct student
{
   
   
    int sn;
    int age;
    char sex;
    int s[3];
}a,b,c[10];

int main()
{
   
   

}

方法三:(不建议使用,省略类型名的方法)

struct 
{
   
   
    int sn;
    int age;
    char sex;
    int s[3];
}a,b,c[10];

int main()
{
   
   

}

使用这种方法,结构体只能一次性使用,后续没法添加。

补充:结构体变量在内存中占用字节数为各成员占用字节数总和。

struct aa
{
   
   
    int num; // 2
    char name[10]; //10
};
struct bb
{
   
   
    int a;  //2
    float b; //4
    struct aa c; //12
};

sizeof(a); //18

1.2 使用结构体变量,结构体数组,结构体指针变量

1.在定义结构体变量的同时可以将各成员的初值按顺序放在一对花括号中,来进行对结构体变量的初始化。若初值个数多于成员个数则出错,若初值个数少于成员个数,则多余成员自动赋0.

struct aa
{
   
   
    int a;
    char b[10]; 
    float c;
} a1={
   
   30,"china",40.5},a2={
   
   60,"kunming"},a3;

2.结构体变量不能整体引用,只能引用它的成员。(同数组相似)
引用结构体成员的方式:
结构体变量名.成员名
其中,(.)为成员运算符
如:
printf(a1="a1=%d,%s,%f,a1"); //非法
printf(a1="a1=%d,%s,%f,a1.a,a1.b,a2.c");
a1.a=80;
a1.b="xinjiang";
a1.c="60.5";

1.3 指向结构体数据类型的指针

1.3.1 指向结构体变量的指针

可以用指针变量指向结构变量也可以用指针变量指向结构体变量中的成员。要注意指针变量的类型必须与它所指向变量的类型相同。当指针变量指向结构体变量时,对指针变量+1则跳过整个结构体而不是跳过一个成员。

一般来说都是定义指向结构体的指针,用指向整个的指针控制内部的的成员也好,数组也好

struct struct
{
   
   
     int num;
     char name[20];
     chat sex;
     float score;
};
struct student aa={
   
   1001,"zhang",'M',80.5};
struct student *p=&aa;
char *q=aa.name;
int *r=&aa.num;

当指针变量p指向结构体变量aa时,引用aa中的成员的方式有三种:
aa.num
(*p).num
p->num

1.3.2 指向结构体数组的指针

struct student
{
   
   
    int num;
    char name[20];
    char sex;
    float score;
};

,struct student stu[3]={
   
   {
   
   1001,"zhang",'M',60.5},{
   
   1002,"peng",'M',100},{
   
   1003,"wang",'W',90.9}};
struct student *p=stu;

image.png

注:

  1. 可以用结构体变量的成员作为实参,它与普通变量作为实参的用法是一样的。
  2. 用结构体变量作为实参时,要求形参必须是同一结构体的变量,传递后形参与实参各对应成员值是一样的。
  3. 也可以用结构体类型的地址(指针变量或数组)作为实参,要求形参必须是同一结构体类型的指针变量或数组。只是地址传递,则可以通过形参来改变实参的值。

2.链表概要

数据集有两个大方向进行存储
一个是定义数组,连续存储
一个是定义链表,结点定义-不必连续

2.1 链表定义

链表是一种数据结构,它采用动态分配存储单元方式。它能够有效地节省存储空间(同数组比较)

链表都有一个"头指针"变量,它用于指向链表中的第一个元素(地址)。链表中的元素都是结点,链表中的所有结点都是结构体类型,且同一链表中的结点都是同一结构体类型。每个结点都应包括数据部分和下个结点地址两部分内容。链表的最后一个元素(结点)称为链尾。指向NULL

链表的访问都是通过指针变量从头结点开始。

由于链表中的结点是一个结构体类型,并且结点中有一个成员用于指向下一个结点。所以定义作为结点的格式:

struct 结构体名
{
   
   
    定义数据成员:
    struct 结构体名 *指针变量名;

};
例如:
struct student
{
   
   
    int num;
    float score;
    struct student *next;

};
struct student a,*p;

2.2 动态分配函数

定义在头文件中

1.malloc()函数
格式:malloc(size)
作用是在内存的动态存储区中分配一个长度为size个字节的连续空间,函数返回值为一个指向分配域起始地址的指针若分配失败则返回NULL.

例如:开辟一个用于存在struct student 数据的内存空间,并让p指向该空间

struct student *p=(struct student *) malloc(sizeofstruct student));

(struct student *) 强制类型转换,因为默认是void类型,要强制类型转换
sizeofstruct student) 分配结构体那么大的内存空间供以使用

2.free()函数
格式:free(p);
作用是释放用malloc()分配的内存。

2.3 链表操作

(1)建立动态链表(假定若输入的成员为0则表示结束)

#include<stdlib.h>
struct node
{
   
   
    int data;
    struct node *next;
};
struct node *head,*p,*q;
p=(struct node *)malloc(sizeof(struct node));
p->data=10;

(2)链表的使用
p->data;
p->next;

(3)链表结点的删除

核心:先连后断,意味着,先与后一个结点连接,再让原来的连接断开

(4)链表的插入

3. 共用体类型

共用体中所有成员共用同一段内存(所有成员的起始地址都是一样的)

格式:
union 共用体名
{
成员列表;
};

注:
(1)成员列表为定义该共用体的成员,成员定义的方式与普通变量的方式一样。
(2)成员列表必须用一对花括号括起
(3)共用体名可以省略

3.1 共用体变量的定义

(1)先定义类型,再定义变量
(2)定义类型的同时,定义变量
(3)直接定义变量

union data
{
   
   
    int i;
    char ch[10];
    float s;
}a1;

注意:
由于共用体类型变量的所有成员都共用同一段内存,所以共用体类型变量所占的字节数等于该共用体类型中占用字节数最多的成员所占的字节数。 sizeof(a1) // 10

3.2 共用体变量的引用

注:

  1. 不能整体引用共用体变量,只能引用其成员。
  2. 同类型成员共享值
  3. 在内存中整型数据中的二进制低8位占用前面一个字节,高8位占用后面一个字节。

3.3 typedef

用typedef定义新类型名
在编程中可以用typedef来定义新的类型名来代替已有的类型名

格式:
typedef 已有类型名 新的类型名

如:
typedef int INTEGER;
以后在定义变量时int和INTEGER是等价的
INTEGER a[10],b; int a[10],b;

(1) typedef可用于定义各种类型名,但 不能定义变量。即只要见到typedef则该语句最后的标识符必定是一个类型名而不是变量名。
(2)typedef只能对已经存在的类型新增一个别名,而不是创造新类型。即在typedef后必须是一个已有的类型。

相关文章
|
3月前
|
存储 C语言
【C语言程序设计——函数】递归求斐波那契数列的前n项(头歌实践教学平台习题)【合集】
本关任务是编写递归函数求斐波那契数列的前n项。主要内容包括: 1. **递归的概念**:递归是一种函数直接或间接调用自身的编程技巧,通过“俄罗斯套娃”的方式解决问题。 2. **边界条件的确定**:边界条件是递归停止的条件,确保递归不会无限进行。例如,计算阶乘时,当n为0或1时返回1。 3. **循环控制与跳转语句**:介绍`for`、`while`循环及`break`、`continue`语句的使用方法。 编程要求是在右侧编辑器Begin--End之间补充代码,测试输入分别为3和5,预期输出为斐波那契数列的前几项。通关代码已给出,需确保正确实现递归逻辑并处理好边界条件,以避免栈溢出或结果
189 16
|
3月前
|
算法 C语言
【C语言程序设计——循环程序设计】求解最大公约数(头歌实践教学平台习题)【合集】
采用欧几里得算法(EuclideanAlgorithm)求解两个正整数的最大公约数。的最大公约数,然后检查最大公约数是否大于1。如果是,就返回1,表示。根据提示,在右侧编辑器Begin--End之间的区域内补充必要的代码。作为新的参数传递进去。这个递归过程会不断进行,直到。有除1以外的公约数;变为0,此时就找到了最大公约数。开始你的任务吧,祝你成功!是否为0,如果是,那么。就是最大公约数,直接返回。
149 18
|
3月前
|
Serverless C语言
【C语言程序设计——循环程序设计】利用循环求数值 x 的平方根(头歌实践教学平台习题)【合集】
根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码,求解出数值x的平方根;运用迭代公式,编写一个循环程序,求解出数值x的平方根。注意:不能直接用平方根公式/函数求解本题!开始你的任务吧,祝你成功!​ 相关知识 求平方根的迭代公式 绝对值函数fabs() 循环语句 一、求平方根的迭代公式 1.原理 在C语言中,求一个数的平方根可以使用牛顿迭代法。对于方程(为要求平方根的数),设是的第n次近似值,牛顿迭代公式为。 其基本思想是从一个初始近似值开始,通过不断迭代这个公式,使得越来越接近。
98 18
|
3月前
|
C语言
【C语言程序设计——循环程序设计】统计海军鸣放礼炮声数量(头歌实践教学平台习题)【合集】
有A、B、C三艘军舰同时开始鸣放礼炮各21响。已知A舰每隔5秒1次,B舰每隔6秒放1次,C舰每隔7秒放1次。编程计算观众总共听到几次礼炮声。根据提示,在右侧编辑器Begin--End之间的区域内补充必要的代码。开始你的任务吧,祝你成功!
96 13
|
3月前
|
存储 编译器 C语言
【C语言程序设计——函数】分数数列求和2(头歌实践教学平台习题)【合集】
函数首部:按照 C 语言语法,函数的定义首部表明这是一个自定义函数,函数名为fun,它接收一个整型参数n,用于指定要求阶乘的那个数,并且函数的返回值类型为float(在实际中如果阶乘结果数值较大,用float可能会有精度损失,也可以考虑使用double等更合适的数据类型,这里以float为例)。例如:// 函数体代码将放在这里函数体内部变量定义:在函数体中,首先需要定义一些变量来辅助完成阶乘的计算。比如需要定义一个变量(通常为float或double类型,这里假设用float。
103 3
|
3月前
|
存储 算法 安全
【C语言程序设计——函数】分数数列求和1(头歌实践教学平台习题)【合集】
if 语句是最基础的形式,当条件为真时执行其内部的语句块;switch 语句则适用于针对一个表达式的多个固定值进行判断,根据表达式的值与各个 case 后的常量值匹配情况,执行相应 case 分支下的语句,直到遇到 break 语句跳出 switch 结构,若没有匹配值则执行 default 分支(可选)。例如,在判断一个数是否大于 10 的场景中,条件表达式为 “num> 10”,这里的 “num” 是程序中的变量,通过比较其值与 10 的大小关系来确定条件的真假。常量的值必须是唯一的,且在同一个。
83 2
|
3月前
|
存储 编译器 C语言
【C语言程序设计——函数】回文数判定(头歌实践教学平台习题)【合集】
算术运算于 C 语言仿若精密 “齿轮组”,驱动着数值处理流程。编写函数求区间[100,500]中所有的回文数,要求每行打印10个数。根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码。如果操作数是浮点数,在 C 语言中是不允许直接进行。的结果是 -1,因为 -7 除以 3 商为 -2,余数为 -1;注意:每一个数据输出格式为 printf("%4d", i);的结果是 1,因为 7 除以 -3 商为 -2,余数为 1。取余运算要求两个操作数必须是整数类型,包括。开始你的任务吧,祝你成功!
94 1
|
10月前
|
存储 SQL 算法
LeetCode力扣第114题:多种算法实现 将二叉树展开为链表
LeetCode力扣第114题:多种算法实现 将二叉树展开为链表
|
10月前
|
存储 SQL 算法
LeetCode 题目 86:分隔链表
LeetCode 题目 86:分隔链表
|
10月前
|
存储 算法 Java
【经典算法】Leetcode 141. 环形链表(Java/C/Python3实现含注释说明,Easy)
【经典算法】Leetcode 141. 环形链表(Java/C/Python3实现含注释说明,Easy)
87 2