網站首頁 編程語言 正文
創建鏈表存儲結構
我們需要創建一個結構體來存儲一個鏈表結點的相關信息。
typedef int ListDataType;//將ListDataType先定義為int類型,根據需要可以改為不同的類型
//創建一個鏈表的結構體
typedef struct ListNode
{
ListDataType data;//存儲數據
struct ListNode* next;//存儲下一個結點的地址的指針
struct ListNode* prev;//存儲上一個結點的地址的指針
}ListNode;
創建結點
我們每次增加數據都要創建一個新的結點,但每次都創建就會非常的麻煩。所以我們考慮把創建一個結點這個功能封裝成一個函數,每次要使用只要調用一下就可以了。
ListNode* BuyListNode(ListDataType x)
{
ListNode* newnode = (ListNode*)malloc(sizeof(ListNode));//向內存申請一個新的結點
if (newnode == NULL)
{
printf("BuyListNode::%s\n", strerror(errno));//打印結點空間申請失敗的原因
exit(-1);//終止程序
}
newnode->data = x;//將x放入申請的結點數據區
newnode->next = NULL;//讓新結點的next指向空
newnode->prev = NULL;//讓新結點的prev指向空
return newnode;//返回新結點的地址
}
鏈表的初始化
帶頭雙向循環鏈表我們對其初始化就是申請一個帶哨兵位的頭結點。接下來我們看初始化的兩種方法。
void ListInit(ListNode** pphead)//這里我們需要對頭結點進行改變,所以傳二級指針
{
assert(pphead);//檢測傳過來的pphead的地址是否為空
*pphead = BuyListNode(0);//創建一個新的哨兵位頭結點
(*pphead)->next = *pphead;//讓這個結點指向自己
(*pphead)->prev = *pphead;
}
ListNode* ListInit()
{
ListNode* phead = BuyListNode(0);//創建一個新的哨兵位頭結點
phead->next = phead;//讓這個結點的next指向自己
phead->prev = phead;//讓prev指向自己
return phead;//返回哨兵位頭結點的地址
}
雙向鏈表的打印
我們才用遍歷鏈表的方式來打印鏈表中的數據。
void ListPrint(ListNode* phead)
{
assert(phead);
ListNode* cur = phead->next;//讓cur指向哨兵位的下一個結點
while (cur != phead)//鏈表打印是否結束的條件判斷
{
printf("%d ", cur->data);
cur = cur->next;
}
printf("\n");
}
雙向鏈表尾插
雙向循環鏈表結構比較優,所以很方便找到尾結點。尾結點就是哨兵位結點prev指向的結點。
void ListPushBack(ListNode* phead, ListDataType x)
{
assert(phead);//檢查傳過來的頭結點是否為空
ListNode* tail = phead->prev;//找到鏈表的尾結點
ListNode* newnode = BuyListNode(x);//創建一個新的結點
tail->next = newnode;//讓尾結點的next指向新的結點
newnode->prev = tail;//讓新結點的prev指向之前的尾結點
newnode->next = phead;//讓新的結點的next指向頭結點
phead->prev = newnode;//讓頭結點指向新的尾結點
}
雙向鏈表尾刪
void ListPopBack(ListNode* phead)
{
assert(phead);
if (phead->next == phead)//如果鏈表只有哨兵位結點的情況,就不能繼續刪除了
return;
ListNode* tail = phead->prev;//找到尾結點
ListNode* tailPrev = tail->prev;//定義尾結點的前一個結點
free(tail);//釋放要刪除的結點
tailPrev->next = phead;//讓前一個結點指向哨兵位結點
phead->prev = tailPrev;//讓哨兵位結點指向新的尾結點
}
雙向鏈表頭插
雙向鏈表的頭插就是在哨兵位結點的下一個位置插入一個新的數據。
注意:這一種方法種的指向關系不能隨意顛倒,否則就會出錯。
void ListPushFront(ListNode* phead, ListDataType x)
{
assert(phead);
ListNode* newnode = BuyListNode(x);//創建一個新結點
newnode->next = phead->next;//新結點的next指向頭結點的下一個結點
phead->next->prev = newnode;//頭結點的下一個結點指向新結點
phead->next = newnode;//頭結點指向新結點
newnode->prev = phead;//新結點prev指向頭結點
}
我們可以對上一種情況進行優化,定義一個next記錄頭結點的下一個結點的地址。這樣我們就可以不用在意指向順序的問題,可以減少出錯的概率。
void ListPushFront(ListNode* phead, ListDataType x)
{
assert(phead);
ListNode* newnode = BuyListNode(x);//創建一個新結點
ListNode* next = phead->next;//定義next記錄頭節點的下一個結點的位置
phead->next = newnode;//頭結點next指向新結點
newnode->prev = phead;//新結點的prev指向頭結點
next->prev = newnode;//頭結點的下一個結點的prev指向新階段
newnode->next = next;//新結點的next指向原頭結點的下一個結點
}
雙向鏈表頭刪
我們先找到頭結點的下一個結點,然后在把它釋放掉。這里需要注意的是如果鏈表為空,只有哨兵位頭結點的情況,我們需要對其進行特殊的處理。
void ListPopFront(ListNode* phead)
{
assert(phead);
if (phead->next == NULL)//只有哨兵位頭結點的情況
return;
ListNode* next = phead->next->next;//定義next指向要刪除結點的下一個結點
free(phead->next);//釋放要刪除的結點
phead->next = next;//讓哨兵位頭結點指向要刪除的下一個結點
next->prev = phead;//讓要刪除的下一個結點的prev指向toujied
}
雙向鏈表查找
若我們想要對指定的位置的結點進行相應的改變就得先找到對應的結點,所以我們將查找指定數據封裝成一個函數。方便后續調用。
ListNode* ListFind(ListNode* phead, ListDataType x)
{
assert(phead);
ListNode* cur = phead->next;//讓cur指向哨兵位頭結點的下一個結點
while (cur != phead)//鏈表循環是否結束的判斷條件
{
if (cur->data == x)
return cur;
cur = cur->next;
}
return NULL;//找不到對于的結點
}
雙向鏈表pos前插入結點
推薦使用第二種方法,不容易出錯。
void ListInsert(ListNode* pos, ListDataType x)
{
assert(pos);
ListNode* newnode = BuyListNode(x);//創建一個新的結點
pos->prev->next = newnode;//pos的前一個結點的next指向新結點
newnode->prev = pos->prev;//新結點的prev指向pos的前一個結點
newnode->next = pos;//新結點的next指向pos結點
pos->prev = newnode;//pos的prev指向新結點
}
void ListInsert(ListNode* pos, ListDataType x)
{
assert(pos);
ListNode* newnode = BuyListNode(x);//創建一個新的結點
ListNode* posPrev = pos->prev;//定義pos的前一個結點
posPrev->next = newnode;//讓pos的pos前一個結點的next指向新結點
newnode->prev = posPrev;//新結點的prev指向pos的前一個結點
newnode->next = pos;//新結點的next指向pos
pos->prev = newnode;//pos的prev指向新結點
}
雙向鏈表刪除pos位置的結點
void ListErase(ListNode* pos)
{
assert(pos);
ListNode* prev = pos->prev;//prev為pos的前一個結點
ListNode* next = pos->next;//next為pos的后一個結點
free(pos);//釋放posjied
prev->next = next;
next->prev = prev;
}
雙向鏈表的銷毀
void ListDestroy(ListNode* phead)
{
assert(phead);
ListNode* cur = phead->next;//指向哨兵位結點的下一個結點
while (cur != phead)//依次循環找鏈表的每一個結點
{
ListNode* next = cur->next;//記錄cur的下一個結點位置
free(cur);//釋放cur位置的結點
cur = next;//指向下一個結點
}
free(phead);//釋放哨兵位頭結點
}
注意:在寫雙向鏈表的頭插,頭刪,尾插,尾刪時我們可以復用雙向鏈表的任意位置插入和任意位置刪除那兩個函數。那兩個函數就可以完成頭插,頭刪,尾插,尾刪這幾個功能。
順序表和鏈表的區別
順序表優點:
1.物理空間是連續的,方便用下標隨機訪問。
2.CPU高速緩存命中率會更高。
順序表缺點:
1.由于需要物理空間連續,空間不夠需要擴容。擴容本身有一定消耗。其次擴容機制還存在一定的空間浪費。
2.頭部或者中部插入刪除,挪動數據,效率低。O(N)
鏈表優點:
1.任意位置插入刪除數據效率高。O(1)
2.按需申請和釋放空間
鏈表缺點:
不支持下標的隨機訪問。有些算法不適合在它上面運行。如:二分查找、排序等。
關于CPU相關的知識可以參考這一篇文章:CPU緩存知識
原文鏈接:https://blog.csdn.net/qq_61939403/article/details/123854225
相關推薦
- 2022-09-20 Xmind用例導入到TAPD的解決方案_其它綜合
- 2022-05-11 React中的Refs屬性你來了解嗎_React
- 2023-07-02 oracle實現根據字段分組排序,取其第一條數據_oracle
- 2022-08-14 Python全局變量關鍵字global的簡單使用_python
- 2023-05-29 Postgresql數據庫角色創建登錄詳解_PostgreSQL
- 2022-08-29 Python神器之Pampy模式匹配庫的用法詳解_python
- 2022-05-26 Nginx多個前端服務配置方式詳解_nginx
- 2022-02-27 Error in render: “TypeError: Cannot read propertie
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支