golang雙鏈表的實(shí)現(xiàn)代碼示例
雙鏈表的實(shí)現(xiàn)
基本概念
每一個(gè)節(jié)點(diǎn)都存儲(chǔ)上一個(gè)和下一個(gè)節(jié)點(diǎn)的指針
實(shí)現(xiàn)思路
創(chuàng)建一個(gè)節(jié)點(diǎn)結(jié)構(gòu)體
- 每個(gè)節(jié)點(diǎn)都有上節(jié)點(diǎn)指針與下節(jié)點(diǎn)指針
- 每個(gè)節(jié)點(diǎn)都有一個(gè)key => value
創(chuàng)建一個(gè)鏈表結(jié)構(gòu)體
- 鏈表容量大小屬性
- 鏈表大小屬性
- 鏈表鎖, 實(shí)現(xiàn)并發(fā)安全
- 鏈表頭節(jié)點(diǎn)
- 鏈表尾節(jié)點(diǎn)
實(shí)現(xiàn)鏈表操作方法
- 添加頭部節(jié)點(diǎn)操作AppendHead
- 添加尾部節(jié)點(diǎn)操作AppendTail
- 追加尾部節(jié)點(diǎn)操作Append
- 插入任意節(jié)點(diǎn)操作Insert
- 刪除任意節(jié)點(diǎn)操作Remove
- 刪除頭部節(jié)點(diǎn)操作RemoveHead
- 刪除尾部節(jié)點(diǎn)操作RemoveTail
- 獲取指定位置的節(jié)點(diǎn)Get
- 搜索任意節(jié)點(diǎn)Search
- 獲取鏈表大小GetSize
- 打印所有節(jié)點(diǎn)操作Print
- 反轉(zhuǎn)所有節(jié)點(diǎn)操作Reverse
總結(jié)
- 學(xué)好算法是一個(gè)不斷積累的過程
- 學(xué)習(xí)時(shí)還需做到知行合一
- 實(shí)現(xiàn)時(shí)需要多做用例測(cè)試.
代碼解析
定義節(jié)點(diǎn)的結(jié)構(gòu)體
type DoubleNode struct { Key int //鍵 Value interface{} //值 Prev *DoubleNode //上一個(gè)節(jié)點(diǎn)指針 Next *DoubleNode //下一個(gè)節(jié)點(diǎn)指針 Freq int //頻率次數(shù).為了給LFU使用的 }
定義一個(gè)雙鏈表的結(jié)構(gòu)體
//定義一個(gè)雙鏈表的結(jié)構(gòu) type DoubleList struct { lock *sync.RWMutex //鎖 Capacity uint //最大容量 Size uint //當(dāng)前容量 Head *DoubleNode //頭節(jié)點(diǎn) Tail *DoubleNode //尾部節(jié)點(diǎn) }
初使雙鏈表
//初使雙鏈表 func New(capacity uint) *DoubleList { list := new(DoubleList) list.Capacity = capacity list.lock = new(sync.RWMutex) list.Size = 0 list.Head = nil list.Tail = nil return list }
添加頭部節(jié)點(diǎn)
實(shí)現(xiàn)思路
- 先判斷容量大小
- 判斷頭部是否為空,
- 如果為空則添加新節(jié)點(diǎn)
- 如果不為空則更改現(xiàn)有的節(jié)點(diǎn),并添加上
func (list *DoubleList) AddHead(node *DoubleNode) bool { //判斷容量是否為0 if list.Capacity == 0 { return false } list.lock.Lock() defer list.lock.Unlock() //判斷頭部節(jié)點(diǎn)是否為nil if list.Head == nil { list.Head = node list.Tail = node } else { //存在頭部節(jié)點(diǎn) list.Head.Prev = node //將舊頭部節(jié)點(diǎn)上一個(gè)節(jié)點(diǎn)指向新節(jié)點(diǎn) node.Next = list.Head //新頭部節(jié)點(diǎn)下一個(gè)節(jié)點(diǎn)指向舊頭部節(jié)點(diǎn) list.Head = node //設(shè)置新的頭部節(jié)點(diǎn) list.Head.Prev = nil //將新的頭部節(jié)點(diǎn)上一個(gè)節(jié)點(diǎn)設(shè)置NIL } list.Size++ return true }
添加尾部元素
實(shí)現(xiàn)思路
- 先判斷容量大小
- 判斷尾部是否為空,
- 如果為空則添加新節(jié)點(diǎn)
- 如果不為空則更改現(xiàn)有的節(jié)點(diǎn),并添加上
func (list *DoubleList) AddTail(node *DoubleNode) bool { //判斷是否有容量, if list.Capacity == 0 { return false } list.lock.Lock() defer list.lock.Unlock() //判斷尾部是否為空 if list.Tail == nil { list.Tail = node list.Head = node } else { //舊的尾部下個(gè)節(jié)點(diǎn)指向新節(jié)點(diǎn) list.Tail.Next = node //追加新節(jié)點(diǎn)時(shí),先將node的上節(jié)點(diǎn)指向舊的尾部節(jié)點(diǎn) node.Prev = list.Tail //設(shè)置新的尾部節(jié)點(diǎn) list.Tail = node //新的尾部下個(gè)節(jié)點(diǎn)設(shè)置為空 list.Tail.Next = nil } //雙鏈表大小+1 list.Size++ return true }
添加任意位置元素
實(shí)現(xiàn)思路
- 判斷容量大小
- 判斷鏈表大小
- 第一種: 如果插入的位置大于當(dāng)前長(zhǎng)度則尾部添加
- 第二種: 如果插入的位置等于0則,頭部添加
- 第三種: 中間插入節(jié)點(diǎn)
- 先取出要插入位置的節(jié)點(diǎn),假調(diào)為C節(jié)點(diǎn)
- 介于A, C之間, 插入一個(gè)B節(jié)點(diǎn)
- A的下節(jié)點(diǎn)應(yīng)該是B, 即C的上節(jié)點(diǎn)的下節(jié)點(diǎn)是B
- B的上節(jié)點(diǎn)是C的上節(jié)點(diǎn)
- B的下節(jié)點(diǎn)是C
//添加任意位置元素 func (list *DoubleList) Insert(index uint, node *DoubleNode) bool { //容量滿了 if list.Size == list.Capacity { return false } //如果沒有節(jié)點(diǎn) if list.Size == 0 { return list.Append(node) } //如果插入的位置大于當(dāng)前長(zhǎng)度則尾部添加 if index > list.Size { return list.AddTail(node) } //如果插入的位置等于0則,頭部添加 if index == 0 { return list.AddHead(node) } //取出要插入位置的節(jié)點(diǎn) nextNode := list.Get(index) list.lock.Lock() defer list.lock.Unlock() //中間插入: //假設(shè)已有A, C節(jié)點(diǎn), 現(xiàn)在要插入B節(jié)點(diǎn) // nextNode即是C節(jié)點(diǎn), //A的下節(jié)點(diǎn)應(yīng)該是B, 即C的上節(jié)點(diǎn)的下節(jié)點(diǎn)是B nextNode.Prev.Next = node //B的上節(jié)點(diǎn)是C的上節(jié)點(diǎn) node.Prev = nextNode.Prev //B的下節(jié)點(diǎn)是C node.Next = nextNode //C的上節(jié)點(diǎn)是B nextNode.Prev = node list.Size++ return true }
刪除頭部節(jié)點(diǎn)
實(shí)現(xiàn)思路
- 判斷頭部是否為空
- 將頭部節(jié)點(diǎn)取出來
- 判斷頭部是否有下一個(gè)節(jié)點(diǎn)
- 沒有下一個(gè)節(jié)點(diǎn),則說明只有一個(gè)節(jié)點(diǎn), 刪除本身,無須移動(dòng)指針位置
- 如果有下一個(gè)節(jié)點(diǎn),則頭部下一個(gè)節(jié)點(diǎn)即是頭部節(jié)點(diǎn).
//刪除頭部節(jié)點(diǎn) func (list *DoubleList) RemoveHead() *DoubleNode { //判斷頭部節(jié)點(diǎn)是否為空 if list.Head == nil { return nil } list.lock.Lock() defer list.lock.Unlock() //將頭部節(jié)點(diǎn)取出來 node := list.Head //判斷頭部是否有下一個(gè)節(jié)點(diǎn) if node.Next != nil { list.Head = node.Next list.Head.Prev = nil } else { //如果沒有下一個(gè)節(jié)點(diǎn), 說明只有一個(gè)節(jié)點(diǎn) list.Head, list.Tail = nil, nil } list.Size-- return node }
刪除尾部節(jié)點(diǎn)
實(shí)現(xiàn)思路
- 判斷尾部節(jié)點(diǎn)是否為空
- 取出尾部節(jié)點(diǎn)
- 判斷尾部節(jié)點(diǎn)的上一個(gè)節(jié)點(diǎn)是否存在
- 不存在:則說明只有一個(gè)節(jié)點(diǎn), 刪除本身,無須移動(dòng)指針位置
- 如果存在上一個(gè)節(jié)點(diǎn),則尾部的上一個(gè)節(jié)點(diǎn)即是尾部節(jié)點(diǎn).
//刪除尾部節(jié)點(diǎn) func (list *DoubleList) RemoveTail() *DoubleNode { //判斷尾部節(jié)點(diǎn)是否為空 if list.Tail == nil { return nil } list.lock.Lock() defer list.lock.Unlock() //取出尾部節(jié)點(diǎn) node := list.Tail //判斷尾部節(jié)點(diǎn)的上一個(gè)是否存在 if node.Prev != nil { list.Tail = node.Prev list.Tail.Next = nil } list.Size-- return node }
刪除任意元素
實(shí)現(xiàn)思路
- 判斷是否是頭部節(jié)點(diǎn)
- 判斷是否是尾部節(jié)點(diǎn)
- 否則為中間節(jié)點(diǎn),需要移動(dòng)上下節(jié)點(diǎn)的指針位置.并實(shí)現(xiàn)元素刪除
- 將上一個(gè)節(jié)點(diǎn)的下一節(jié)點(diǎn)指針指向下節(jié)點(diǎn)
- 將下一個(gè)節(jié)點(diǎn)的上一節(jié)點(diǎn)指針指向上節(jié)點(diǎn)
//刪除任意元素 func (list *DoubleList) Remove(node *DoubleNode) *DoubleNode { //判斷是否是頭部節(jié)點(diǎn) if node == list.Head { return list.RemoveHead() } //判斷是否是尾部節(jié)點(diǎn) if node == list.Tail { return list.RemoveTail() } list.lock.Lock() defer list.lock.Unlock() //節(jié)點(diǎn)為中間節(jié)點(diǎn) //則需要: //將上一個(gè)節(jié)點(diǎn)的下一節(jié)點(diǎn)指針指向下節(jié)點(diǎn) //將下一個(gè)節(jié)點(diǎn)的上一節(jié)點(diǎn)指針指向上節(jié)點(diǎn) node.Prev.Next = node.Next node.Next.Prev = node.Prev list.Size-- return node }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Golang http包構(gòu)建RESTful API的實(shí)現(xiàn)
在Go語(yǔ)言中實(shí)現(xiàn)RESTful API可以利用標(biāo)準(zhǔn)庫(kù)net/http提供的功能,它允許你輕松地創(chuàng)建和處理HTTP請(qǐng)求,本文主要介紹了Golang http包構(gòu)建RESTful API的實(shí)現(xiàn),感興趣的可以了解一下2024-01-01go mock server的簡(jiǎn)易實(shí)現(xiàn)示例
這篇文章主要為大家介紹了go mock server的簡(jiǎn)易實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07深入講解Go語(yǔ)言中函數(shù)new與make的使用和區(qū)別
大家都知道Go語(yǔ)言中的函數(shù)new與函數(shù)make一直是新手比較容易混淆的東西,看著相似,但其實(shí)不同,不過解釋兩者之間的不同也非常容易,下面這篇文章主要給大家介紹了關(guān)于Go語(yǔ)言中函數(shù)new與make區(qū)別的相關(guān)資料,需要的朋友可以參考下。2017-10-10快速掌握Go 語(yǔ)言 HTTP 標(biāo)準(zhǔn)庫(kù)的實(shí)現(xiàn)方法
基于HTTP構(gòu)建的服務(wù)標(biāo)準(zhǔn)模型包括兩個(gè)端,客戶端(Client)和服務(wù)端(Server),這篇文章主要介紹了Go 語(yǔ)言HTTP標(biāo)準(zhǔn)庫(kù)的實(shí)現(xiàn)方法,需要的朋友可以參考下2022-07-07Go語(yǔ)言并發(fā)爬蟲的具體實(shí)現(xiàn)
本文主要介紹了Go語(yǔ)言并發(fā)爬蟲的具體實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-12-12