c语言带头双链表的基础操作

1. 创建一个双链表结点

typedef int ltdatatype;

typedef struct listnode

{

int data;

struct listnode* prev;//前驱节点

struct listnode* next;//后驱结点

}ltnode;

2.创建一个新结点的函数

ltnode* ltbuynode(int x)//x是数据域

{

ltnode* newnode = (ltnode*)malloc(sizeof(ltnode));

if (newnode = NULL)

{

perror("malloc fail");

exit(1);

}

newnode->data = x;

newnode->next = NULL;

newnode->prev = NULL;

return newnode;

}

3.双链表结点的初始化

ltnode* ltinit()

{

ltnode* phead = ltbuynode(-1);

return phead;

}

4.双链表的打印

void ltprint(ltnode* phead)

{

ltnode* pcur = phead;
//用遍历链表的方式打印

assert(phead);

ltnode* pcur = phead->next;//创建指针变量pcur指向头结点

while (pcur != phead)

{

printf("%d->", pcur->data);

pcur = pcur->next;

}

printf("\n");

}

5.双链表的尾插

void ltpushback(ltnode* phead, ltdatatype x)

{

ltnode* newnode = ltbuynode(x);

//phead phead->prev newnode

newnode->next = phead;//

newnode->prev = phead->prev;

phead->prev->next = newnode;

phead->prev = newnode;

}

6.双链表的头插

void ltpushfront(ltnode* phead, ltdatatype x)

{

assert(phead);

ltnode* newnode = lbuynode(x);

//结点的连结

newnode->next = phead->next;

newnode->prev = phead;

phead->next->prev = newnode;

phead->next = newnode;

}

7.双链表的尾删

void ltpopback(ltnode* phead)

{

assert(phead);
//链表不能为空(不能只有头结点)

ltnode* del = phead->prev;

ltnode* prev = del->prev;

prev->next = phead;

phead->prev = prev;

free(del);

del = NULL;

}

8. 双链表的头删

//头删,不是删除头结点

void ltpopfront(ltnode* phead)

{

assert(phead);

assert(phead->next!=phead);

ltnode* del = phead->next;

ltnode* next = del->next;

next->prev = phead;

phead->next = next;

free(del);

del = NULL;

}

9.双链表的结点查找

//结点查找

ltnode* ltfind(ltnode* phead, ltdatatype x)

{

assert(phead);

ltnode* pcur = phead->next;//遍历循环双链表的同时又不让它成为死循环,所以应该将pcur指向第一节点

while (pcur != phead)//所以只要遍历到头结点,那么就停止遍历

{

if (pcur->data == x)//如果找到了

{

return pcur;

}

pcur = pcur->next;

}

return NULL;

}

10.双链表的指定节点之后插入数据

//指定位置pos之后插入数据

ltnode* ltfind(ltnode* pos, ltdatatype x)

{

assert(pos);

ltnode* newnode = ltbuynode(x);

newnode->next = pos->next;

newnode->prev = pos;

pos->next->prev = newnode;

pos->next = newnode;

}

11.删除指定结点

//删除pos位置的数据

void lterase(ltnode* pos)

{

assert(pos);

pos->next->prev = pos->prev;

pos->prev->next = pos->next;

free(pos);

pos->next;

}

12.双链表的销毁

void ltdestroy(ltnode* phead)

{

assert(phead);

ltnode* pcur = phead->next;

while (pcur != phead)

{

free(pcur);//每遍历一次就删除一次

pcur = pcur->next;

}

free(phead);

phead = NULL;//因为是一级指针,所以这里phead为空以后实参的一级指针仍然没空,
//所以如果实参时plist,那么经过此函数后仍要手动置空plist=NULL;

}

相关推荐
USER_A0011 小时前
【C语言】第五期——函数
c语言
计算机小白一个6 小时前
蓝桥杯 Java B 组之设计 LRU 缓存
java·算法·蓝桥杯
万事可爱^6 小时前
HDBSCAN:密度自适应的层次聚类算法解析与实践
算法·机器学习·数据挖掘·聚类·hdbscan
李白同学7 小时前
【C语言】结构体内存对齐问题
c语言·开发语言
楼台的春风8 小时前
【MCU驱动开发概述】
c语言·驱动开发·单片机·嵌入式硬件·mcu·自动驾驶·嵌入式
大数据追光猿8 小时前
Python应用算法之贪心算法理解和实践
大数据·开发语言·人工智能·python·深度学习·算法·贪心算法
Dream it possible!9 小时前
LeetCode 热题 100_在排序数组中查找元素的第一个和最后一个位置(65_34_中等_C++)(二分查找)(一次二分查找+挨个搜索;两次二分查找)
c++·算法·leetcode
夏末秋也凉9 小时前
力扣-回溯-46 全排列
数据结构·算法·leetcode
南宫生9 小时前
力扣每日一题【算法学习day.132】
java·学习·算法·leetcode
柠石榴9 小时前
【练习】【回溯No.1】力扣 77. 组合
c++·算法·leetcode·回溯