相信了解了结构体,指针,结构体指针的,对于链表也能比较容易的理解,我觉得该怎么去容易理解呢,不仅要看代码,而且还要自己亲自写一下代码,并输出结果,理解其中的每一步。
什么是链表
我觉得看懂这两段话就行:
链表是可以用如何排列你的数据的一种数据结构,相对于数组(array)链表有两个明显的优点:
- 链表可以随时改变大小(数据的多少),而数组的大小一旦在声明的时候定义了就不能被改变
- 在链表里面排序是非常简单的,而在数组里面如果想要插入新的元素或者移动一个元素,我们需要的statement会很多,也会浪费很多的时间
数据是不定长的,在存储第一个数据之前难以确定一个将来一共需要存储多少数据的上限,或者虽然可以确定上限,但这个上限又比通常大部分情况下数据可能达到的长度要大得多,因而一次性按照上限把空间分配好是不划算的。而链表则可以在每次需要增加新数据时才为之申请内存,不会造成浪费,也不会因一次申请不足而使数据的数量受到限制。
链表你不一定经常使用,但是代码一定要看得懂
静态链表
静态链表不怎么用,但是可以很好的理解什么是链表,看一段代码:
#include <stdio.h>
// 定义链表节点
typedef struct node{
int num;
struct node *next;
}Node;
int main()
{
//创建三个链表节点,实际就是三个结构体
Node a;
Node b;
Node c;
//初始化节点数据
a.num=1;
b.num=2;
c.num=3;
//将它们链接起来,链表有首有尾
Node *head=&a;
a.next=&b;
b.next=&c;
//c的后面没有了
c.next=NULL;
while(head != NULL){//判断是否有数据
int data = head->num;
printf("data = %i\n", data);
head = head->next;//进入下一个节点
}
return 0;
}
输出结果:
data = 1
data = 2
data = 3
看完静态链表就能初步理解什么是链表了,下面主要看一下什么是动态链表
动态链表
空链表
头指针带了一个空链表节点, 空链表节点中的next指向NUL
#include <stdio.h>
#include <stdlib.h>
// 定义链表节点
typedef struct node{
int num;
struct node *next;
}Node;
Node *CreateEmptyList(){//这是什么返回类型为Node的指针函数
Node *a=(Node *)malloc(sizeof(Node));//这什么意思:创建一个结构体指针,并分配内存,下面的内容就很熟悉了
if(a !=NULL){
a->next=NULL;//其实就是创建一个指向空的节点
return a;
}
}
int main()
{
Node *head=CreateEmptyList();//这个是不是就像静态链表中的Node *head=&a;
return 0;
}
非空链表
头指针带了一个非空节点, 最后一个节点中的next指向NULL
#include <stdio.h>
#include <stdlib.h>
typedef struct node{
int data;
struct node *next;
}Node;
Node *createList();
void printNodeList(Node *node);
int main()
{
Node *head = createList();
printNodeList(head);
return 0;
}
Node *createList(){
// 1.创建头节点
Node *head = (Node *)malloc(sizeof(Node));
if(head == NULL){
return NULL;
}
head->next = NULL;//这是什么意思:实际就是建了一个空链表,咱们要往里面见缝插针
// 2.接收用户输入数据
int num = -1;
printf("请输入节点数据\n");
scanf("%i", &num);//假设输入1
// 3.通过循环创建其它节点
while(num != -1){
// 3.1创建一个新的节点
Node *cur = (Node *)malloc(sizeof(Node));
cur->data = num;//这个就等于1
// 3.2让新节点的下一个节点指向头节点的下一个节点
cur->next = head->next;//有没有发现,这个cur->next=NULL,意味着它是链尾
// 3.3让头节点的下一个节点指向新节点
head->next = cur;//这又是什么意思,就是说倒数第二个节点,也就是链尾的前一个节点
// 3.4再次接收用户输入数据
scanf("%i", &num);//咱们再输入2,最后输出-1
}
// 3.返回创建好的节点
return head;//这个循环已经建立好了链表,你输出这个head就行,咱们根据这个头去找数据
}
/**
* @brief printNodeList 遍历链表
* @param node 链表指针头
*/
void printNodeList(Node *node){
Node *head = node->next;
while(head != NULL){
int currentData = head->data;
printf("currentData = %i\n", currentData);
head = head->next;
}
}
//假如我输入1,2,3,4
//那么它的输出结果将会是什么呢
//currentData =4
//currentData =3
//currentData =2
//currentData =1