一、链表
1.1、什么是链表
1、链表是物理存储单元上非连续的、非顺序的存储结构,数据元素的逻辑顺序是通过链表的指针地址实现,有一系列结点(地址)组成,结点可动态的生成。
2、结点包括两个部分:
(1)存储数据元素的数据域(内存空间)
(2)存储指向下一个结点地址的指针域。
3、相对于线性表顺序结构,操作复杂。
1.2、链表的分类
链表的结构非常多样,以下的情况组合起来就有8种链表结构
(1)单项和双向
(2)带头和不带头
(3)循环和不循环
1.3、链表和数组(顺序表)的比较
(1)数组:使用一块连续的内存空间地址去存放数据,但
例如:
int a[5]={1,2,3,4,5}。突然我想继续加两个数据进去,但是已经定义好的数组不能往后加,只能通过定义新的数组
int b[7]={1,2,3,4,5,6,7}; 这样就相当不方便比较浪费内存资源,对数据的增删不好操作。
(2)链表:使用多个不连续的内存空间去存储数据, 可以 节省内存资源(只有需要存储数据时,才去划分新的空间),对数据的增删比较方便
注意:
1.链式结构在逻辑上是连续的,但在物理上不一定连续
2.现实中的结点一般都是从堆上申请出来的
3.从堆上申请的空间,是按照一定的策略来分配的,两次申请的空间可能连续,也可能不连续
二、无头单向非循环链表
2.1、无头单向非循环链表的结构
链表的构成:
链表由一个个节点构成,每个节点一般采用结构体的形式组织,例如:
typedef struct student
{
int num;
int score;
char name[20];
struct student *next;//保存下一个节点的地址
}STU;
链表节点分为两个域
**数据域**:存放各种实际的数据,如:num、score等
**指针域**:存放下一节点的首地址,如:next等.
2.1、链表的创建
//创建新节点
STU* creatLinkNode(void)
{
STU* newnode = (STU*)malloc(sizeof(STU));
if (newnode == NULL)
{
perror("malloc");
exit(-1);
}
memset(newnode, 0, sizeof(STU));
newnode->next = NULL;
return newnode;
}
创建一个新节点,用malloc开辟一个链表节点空间,强制转换成链表结构体,将数据域初始化空,将next置为空,并返回新节点。
2.2、单向链表尾插法
//在尾部添加新节点
void addLinkTail(STU **p_head,STU *p_new)
{
STU *p_mov = *p_head;
if(*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
p_new->next=NULL;
}
else //第二次及以后加入链表
{
/*找到原有链表的最后一个节点,然后在next域指向新节点,
新节点的next域置为NULL,完成新节点加入链表的操作
*/
while(p_mov->next!=NULL)
{
p_mov=p_mov->next;
}
p_mov->next = p_new;
p_new->next = NULL;
}
}
单链表的尾插首先需要判断是否是空链表,如果为空就把该节点置为头节点,若不为空,需要找到原链表的最后一个节点,将最后一个节点的next指向新节点,新节点的next置NULL
2.3、单向链表头插法
//单链表的头插法 效率高,简单
void addLinkHead(STU **p_head,STU *p_new)
{
assert(p_head);
assert(p_new);
p_new->next = *p_head;
*p_head = p_new;
}
头插法相对简单,只需要将新节点插到头结点的前面,并且将头结点指针赋给新节点。
2.4、链表的遍历
第一步:输出第一个节点的数据域,输出完毕后,让指针保存后一个节点的地址
第二步:输出移动地址对应的节点的数据域,输出完毕后,指针继续后移
第三步:以此类推,直到节点的指针域为NULL
//打印链表
void printLink(STU *head)
{
STU *pb = head;
//判断是否到达链表尾部
while(pb !=NULL)
{
printf("num=%d score=%d name:%s\n",pb->num,pb->score,pb->name);
pb=pb->next;
}
}
2.5、单向链表删除尾部节点
//单链表的尾部删出一个节点
void delLinkTail(STU** p_head)
{
assert(p_head);
//空
assert(*p_head);
// 1个节点
if ((*p_head)->next == NULL)
{
free((*p_head));
*p_head = NULL;
}
else //两个或者多个节点
{ //方法一
/*STU* pt = *p_head;
while (pt->next->next)
{
pt = pt->next;
}
free(pt->next);
pt->next = NULL;*/
//方法二
STU* pt = *p_head; //尾结点
STU* pf = NULL; //尾结点的前一个结点
while (pt->next)
{
pf = pt;
pt = pt->next;
}
free(pt);
pt = NULL;
pf->next = NULL;
}
}
和尾插法一样,首先先判断链表是否只有一个节点或者没有节点(为空),将会最后一个链表置空,如果超过一个节点,先找到倒数第二个节点,然后置空最后一个节点,将倒数第二个节点的next置空
2.6、单向链表删除头部
//链表的头删法 效率高,简单
void delLinkHead(STU** p_head)
{
assert(p_head);
assert(*p_head);
/*
STU* pb = *p_head;
*p_head = pb->next;
free(pb);
pb = NULL;
*/
STU* pb = (*p_head)->next;
free(*p_head);
*p_head = pb;
}
2.7、删除指定节点
如果链表为空,不需要删除 如果删除的是第一个结点,则需要将保存链表首地址的指针保存第一个结点的下一个结点的 地址 如果删除的是中间结点,则找到中间结点的前一个结点,让前一个结点的指针域保存这个结 点的后一个结点的地址即可
//删除指定节点pos
void delLinkNode(STU **p_head,STU *pos)
{
assert(p_head);
assert(*p_head);
assert(pos);
//删除头结点
if(*p_head == pos)
{
STU* pb = (*p_head)->next;
free(*p_head);
*p_head = pb;
}
else
{
STU *pb = *p_head;
//循环查找要删除的节点
while(pb->next != pos)
{
pb = pb->next;
}
//找到要删除的节点
if(pb->next == pos)
{
pb->next = pos->next;
free(pos);
}
else
{
printf("节点不存在!\n");
}
}
}
//链表结点的删除(按节点内信息)
void delLinkforNum(STU **p_head,int num)
{
STU *pb,*pf;
pb=pf=*p_head;
if(*p_head == NULL)//链表为空,不用删
{
printf("链表为空,没有您要删的节点");
return ;
}
while(pb->num != num && pb->next !=NULL)//循环找,要删除的节点
{
pf=pb;
pb=pb->next;
}
if(pb->num == num)//找到了一个节点的num和num相同
{
if(pb == *p_head)//要删除的节点是头节点
{
//让保存头结点的指针保存后一个结点的地址
*p_head = pb->next;
}
else
{
//前一个结点的指针域保存要删除的后一个结点的地址
pf->next = pb->next;
}
//释放空间
free(pb);
pb = NULL;
}
else//没有找到
{
printf("没有您要删除的节点\n");
}
}
2.8、单向链表的查找
先对比第一个结点的数据域是否是想要的数据,如果是就直接返回,如果不是则继续查找下一个结点,如果到达最后一个结点的时候都没有匹配的数据,说明要查找数据不存在
//链表的查找
STU * searLinkForNum(STU *p_head,int num)
{
assert(p_head);
//定义的指针变量保存第一个结点的地址
STU *p_mov = p_head;
//当没有到达最后一个结点的指针域时循环继续
while(p_mov != NULL)
{
//如果找到是当前结点的数据,则返回当前结点的地址
if(p_mov->num == num)//找到了
{
return p_mov;
}
//如果没有找到,则继续对比下一个结点的指针域
p_mov = p_mov->next;
}
//当循环结束的时候还没有找到,说明要查找的数据不存在,返回NULL进行标识
return NULL;//没有找到
}
2.9、单向链表插入一个节点
链表中插入一个结点,按照原本链表的顺序插入,找到合适的位置
情况(按照从小到大):
如果链表没有结点,则新插入的就是第一个结点。
如果新插入的结点的数值最小,则作为头结点。
如果新插入的结点的数值在中间位置,则找到前一个,然后插入到他们中间。
如果新插入的结点的数值最大,则插入到最后。
//链表的插入:按照学号的顺序插入
void insertLinkForNum(STU **p_head,STU *p_new)
{
STU *pb,*pf;
pb=pf=*p_head;
if(*p_head == NULL)// 链表为空链表
{
*p_head = p_new;
p_new->next = NULL;
return ;
}
//循环比较要插入的位置
while((p_new->num >= pb->num) && (pb->next !=NULL) )
{
pf = pb;
pb = pb->next;
}
//找到一个节点的num比新来的节点num大,插在pb的前面
if(p_new->num < pb->num)
{
if(pb == *p_head)//找到的节点是头节点,插在最前面
{
p_new->next= *p_head;
*p_head =p_new;
}
else
{
pf->next=p_new;
p_new->next = pb;
}
}
else//没有找到pb的num比p_new->num大的节点,插在最后
{
pb->next =p_new;
p_new->next =NULL;
}
}
2.10、单向链表在指定节点前插入节点
//在指定位置pos前插入一个新节点p_new
void insertLinkFront(STU **p_head,STU *pos,STU *p_new)
{
assert(p_head);
assert(*p_head);
assert(pos);
assert(p_new);
//头节点前插入
if(*p_head == pos)
{
p_new->next = *p_head;
*p_head = p_new;
return ;
}
else
{
//在pos前面插入
STU *pb = *p_head;
while(pb->next != pos)
{
pb=pb->next;
}
if(pb->next == pos) //判断是否找到pos
{
p_new->next = pb->next;
pb->next = p_new;
}
else
{
printf("没有找到pos节点\n");
}
}
}
2.11、单向链表在指定节点后插入节点
void insertLinkAfter(STU **p_head,STU *pos,STU *p_new)
{
assert(p_head);
assert(*p_head);
assert(pos);
assert(p_new);
//要插入的节点是头节点
if((*p_head) == pos)
{
printf("-------------------------------\n");
p_new->next = (*p_head)->next;
(*p_head)->next = p_new;
return ;
}
else
{
//在pos后面插入,先找到要插入的节点
#if 0
STU *pb = (*p_head)->next;
while(pb != pos)
{
pb = pb->next;
}
if(pb == pos) //判断是否找到pos
{
p_new->next = pb->next;
pb->next = p_new;
}
else
{
printf("没有找到pos节点\n");
}
#endif
p_new->next = pos->next;
pos->next = p_new;
}
}
2.12、链表节点的释放
重新定义一个指针q,保存p指向节点的地址,然后p后移保存下一个节点的地址,然后释放q对应的节点,以此类推,直到p为NULL为止。
//链表的释放
void freeLink(STU **p_head)
{
//定义一个指针变量保存头结点的地址
STU *pb=*p_head;
while(*p_head!=NULL)
{
//先保存p_head指向的结点的地址
pb=*p_head;
//p_head保存下一个结点地址
*p_head=(*p_head)->next;
//释放结点并防止野指针
free(pb);
pb = NULL;
}
}
三、双向链表
3.1、双向链表的创建
第一步:创建一个节点作为头节点,将两个指针域都保存NULL
第二步:先找到链表中的最后一个节点,然后让最后一个节点的指针域保存新插入节点的地址,新插入节点的两个指针域,一个保存上一个节点的地址,一个保存NULL
①、节点信息
**
//定义结点结构体
typedef struct student
{
//数据域
int num; //学号
int score; //分数
char name[20]; //姓名
//指针域
struct student *front; //保存上一个结点的地址
struct student *next; //保存下一个结点的地址
}STU;
②、节点创建
//创建新节点
STU* creatDoubleLinkNode(void)
{
STU* p_new = (STU*)malloc(sizeof(STU));
if (p_new == NULL)
{
perror("malloc");
exit(-1);
}
memset(p_new, 0, sizeof(STU));
p_new->front = NULL;
p_new->next = NULL;
return p_new;
}
3.2、双向链表的遍历
//打印链表
void printDoubleLink(STU *head)
{
STU *pb = head;
while(pb->next != NULL) //向后遍历
{
printf("num=%d score=%d name:%s\n",pb->num,pb->score,pb->name);
pb=pb->next;
}
printf("num=%d score=%d name:%s\n",pb->num,pb->score,pb->name);
printf("***********************\n");
while(pb != NULL) //向前遍历
{
printf("num=%d score=%d name:%s\n",pb->num,pb->score,pb->name);
pb=pb->front;
}
}
3.3、双向链表在尾部添加新节点
//在尾部添加新节点
void addDoubleLinkTail(STU **p_head,STU *p_new)
{
STU *p_mov = *p_head;
if(*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
}
else //第二次及以后加入链表
{
/*找到原有链表的最后一个节点,然后在next域指向新节点,
新节点的next域置为NULL,完成新节点加入链表的操作
*/
STU *p_mov = *p_head;
while(p_mov->next != NULL)
{
p_mov = p_mov->next; //找到原有链表的最后一个节点
}
p_mov->next = p_new; //将新申请的节点加入链表
p_new->front = p_mov;
p_new->next = NULL;
}
}
3.4、双向链表在头部添加节点
//链表的头插法 效率高,简单
void addDoubleLinkHead(STU **p_head,STU *p_new)
{
assert(p_head);
assert(p_new);
/*
1、头结点为空:将新节点赋值给头节点
2、头节点不为空:则改变头节点的头指针和插入节点的尾指针,最后将新节点赋值给头节点
*/
if(*p_head != NULL)
{
p_new->next = *p_head;
(*p_head)->front = p_new;
}
*p_head = p_new;
}
3.5、双向链表删除尾节点
//单链表的尾部删出一个节点
void delDoubleLinkTail(STU** p_head)
{
assert(p_head);
assert(*p_head);
// 1个节点
if ((*p_head)->next == NULL)
{
free((*p_head));
*p_head = NULL;
}
else //两个或者多个节点
{
STU* pt = *p_head; //尾结点
STU* pf = NULL; //保存尾结点的前一个结点
while (pt->next)
{
pf = pt;
pt = pt->next;
}
free(pt);
pt = NULL;
pf->next = NULL; //将尾节点前一个节点的位置针置NULL!!!
}
}
3.6、双向链表删除头节点
//链表的头删法 效率高,简单
void delDoubleLinkHead(STU** p_head)
{
assert(p_head);
assert(*p_head);
STU* pb = (*p_head)->next;
free(*p_head);
*p_head = pb;
}
3.7、双向链表在指定节点后插入一个节点
//在指定位置pos后插入一个新节点p_new
void insertDoubleLinkAfter(STU **p_head,STU *pos,STU *p_new)
{
assert(p_head);
assert(*p_head);
assert(pos);
assert(p_new);
p_new->next = pos->next;
if(pos->next != NULL) //插入的不是尾节点,需要考虑插入节点后面节点的头指针
pos->next->front = p_new;
pos->next = p_new;
p_new->front = pos;
}
3.8、双向链表在指定节点前插入节点
//在指定位置pos前插入一个新节点p_new
void insertDoubleLinkFront(STU **p_head,STU *pos,STU *p_new)
{
assert(p_head);
assert(*p_head);
assert(pos);
assert(p_new);
p_new->next = pos;
if(*p_head != pos) //插入的节点位置不是头节点,需考虑前后节点的头、尾指针指向
{
pos->front->next = p_new;
p_new->front = pos->front;
}
pos->front = p_new;
if(*p_head == pos) //插入的节点的位置为头节点,需要重新给头指针赋值
*p_head = p_new;
}
3.9、双向链表删除指定节点
//删除指定节点
void delDoubleLinkNode(STU **p_head,STU *pos)
{
assert(p_head);
assert(*p_head);
assert(pos);
//删除头结点
if(*p_head == pos)
{
STU* pb = (*p_head)->next;
free(*p_head);
*p_head = pb;
//将头节点的头指针置NULL!!!
(*p_head)->front = NULL;
}
else
{
if(pos->next == NULL) //删除尾节点
{
pos->front->next = NULL;
free(pos);
pos = NULL;
}
else //删除的不是尾节点
{
pos->front->next = pos->next;
pos->next->front = pos->front;
free(pos);
pos = NULL;
}
}
}