欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

golang雙鏈表的實現(xiàn)代碼示例

 更新時間:2019年08月12日 11:46:45   作者:百里  
這篇文章主要介紹了golang雙鏈表的實現(xiàn)代碼示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

雙鏈表的實現(xiàn)

基本概念

每一個節(jié)點都存儲上一個和下一個節(jié)點的指針

實現(xiàn)思路

創(chuàng)建一個節(jié)點結(jié)構(gòu)體

  • 每個節(jié)點都有上節(jié)點指針與下節(jié)點指針
  • 每個節(jié)點都有一個key => value

創(chuàng)建一個鏈表結(jié)構(gòu)體

  • 鏈表容量大小屬性
  • 鏈表大小屬性
  • 鏈表鎖, 實現(xiàn)并發(fā)安全
  • 鏈表頭節(jié)點
  • 鏈表尾節(jié)點

實現(xiàn)鏈表操作方法

  • 添加頭部節(jié)點操作AppendHead
  • 添加尾部節(jié)點操作AppendTail
  • 追加尾部節(jié)點操作Append
  • 插入任意節(jié)點操作Insert
  • 刪除任意節(jié)點操作Remove
  • 刪除頭部節(jié)點操作RemoveHead
  • 刪除尾部節(jié)點操作RemoveTail
  • 獲取指定位置的節(jié)點Get
  • 搜索任意節(jié)點Search
  • 獲取鏈表大小GetSize
  • 打印所有節(jié)點操作Print
  • 反轉(zhuǎn)所有節(jié)點操作Reverse

總結(jié)

  1. 學(xué)好算法是一個不斷積累的過程
  2. 學(xué)習(xí)時還需做到知行合一
  3. 實現(xiàn)時需要多做用例測試.

代碼解析

定義節(jié)點的結(jié)構(gòu)體

type DoubleNode struct {
  Key  int     //鍵
  Value interface{} //值
  Prev *DoubleNode //上一個節(jié)點指針
  Next *DoubleNode //下一個節(jié)點指針
  Freq int     //頻率次數(shù).為了給LFU使用的
}

定義一個雙鏈表的結(jié)構(gòu)體

//定義一個雙鏈表的結(jié)構(gòu)
type DoubleList struct {
  lock   *sync.RWMutex //鎖
  Capacity uint     //最大容量
  Size   uint     //當(dāng)前容量
  Head   *DoubleNode  //頭節(jié)點
  Tail   *DoubleNode  //尾部節(jié)點
}

初使雙鏈表

//初使雙鏈表
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é)點

實現(xiàn)思路

  1. 先判斷容量大小
  2. 判斷頭部是否為空,
    1. 如果為空則添加新節(jié)點
    2. 如果不為空則更改現(xiàn)有的節(jié)點,并添加上
func (list *DoubleList) AddHead(node *DoubleNode) bool {
  //判斷容量是否為0
  if list.Capacity == 0 {
    return false
  }
  list.lock.Lock()
  defer list.lock.Unlock()
  //判斷頭部節(jié)點是否為nil
  if list.Head == nil {
    list.Head = node
    list.Tail = node
  } else { //存在頭部節(jié)點
    list.Head.Prev = node //將舊頭部節(jié)點上一個節(jié)點指向新節(jié)點
    node.Next = list.Head //新頭部節(jié)點下一個節(jié)點指向舊頭部節(jié)點
    list.Head = node   //設(shè)置新的頭部節(jié)點
    list.Head.Prev = nil //將新的頭部節(jié)點上一個節(jié)點設(shè)置NIL
  }
  list.Size++
  return true
}

添加尾部元素

實現(xiàn)思路

  • 先判斷容量大小
  • 判斷尾部是否為空,
    • 如果為空則添加新節(jié)點
    • 如果不為空則更改現(xiàn)有的節(jié)點,并添加上
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 {
    //舊的尾部下個節(jié)點指向新節(jié)點
    list.Tail.Next = node
    //追加新節(jié)點時,先將node的上節(jié)點指向舊的尾部節(jié)點
    node.Prev = list.Tail
    //設(shè)置新的尾部節(jié)點
    list.Tail = node
    //新的尾部下個節(jié)點設(shè)置為空
    list.Tail.Next = nil
  }
  //雙鏈表大小+1
  list.Size++
  return true
}

添加任意位置元素

實現(xiàn)思路

  • 判斷容量大小
  • 判斷鏈表大小
  • 第一種: 如果插入的位置大于當(dāng)前長度則尾部添加
  • 第二種: 如果插入的位置等于0則,頭部添加
  • 第三種: 中間插入節(jié)點
    • 先取出要插入位置的節(jié)點,假調(diào)為C節(jié)點
    • 介于A, C之間, 插入一個B節(jié)點
    • A的下節(jié)點應(yīng)該是B, 即C的上節(jié)點的下節(jié)點是B
    • B的上節(jié)點是C的上節(jié)點
    • B的下節(jié)點是C
//添加任意位置元素
func (list *DoubleList) Insert(index uint, node *DoubleNode) bool {
  //容量滿了
  if list.Size == list.Capacity {
    return false
  }
  //如果沒有節(jié)點
  if list.Size == 0 {
    return list.Append(node)
  }
  //如果插入的位置大于當(dāng)前長度則尾部添加
  if index > list.Size {
    return list.AddTail(node)
  }
  //如果插入的位置等于0則,頭部添加
  if index == 0 {
    return list.AddHead(node)
  }
  //取出要插入位置的節(jié)點
  nextNode := list.Get(index)
  list.lock.Lock()
  defer list.lock.Unlock()
  //中間插入:
  //假設(shè)已有A, C節(jié)點, 現(xiàn)在要插入B節(jié)點
  // nextNode即是C節(jié)點,
  //A的下節(jié)點應(yīng)該是B, 即C的上節(jié)點的下節(jié)點是B
  nextNode.Prev.Next = node
  //B的上節(jié)點是C的上節(jié)點
  node.Prev = nextNode.Prev
  //B的下節(jié)點是C
  node.Next = nextNode
  //C的上節(jié)點是B
  nextNode.Prev = node
  list.Size++
  return true
}

刪除頭部節(jié)點

實現(xiàn)思路

  1. 判斷頭部是否為空
  2. 將頭部節(jié)點取出來
  3. 判斷頭部是否有下一個節(jié)點
    1. 沒有下一個節(jié)點,則說明只有一個節(jié)點, 刪除本身,無須移動指針位置
    2. 如果有下一個節(jié)點,則頭部下一個節(jié)點即是頭部節(jié)點.
//刪除頭部節(jié)點
func (list *DoubleList) RemoveHead() *DoubleNode {
  //判斷頭部節(jié)點是否為空
  if list.Head == nil {
    return nil
  }
  list.lock.Lock()
  defer list.lock.Unlock()
  //將頭部節(jié)點取出來
  node := list.Head
  //判斷頭部是否有下一個節(jié)點
  if node.Next != nil {
    list.Head = node.Next
    list.Head.Prev = nil
  } else { //如果沒有下一個節(jié)點, 說明只有一個節(jié)點
    list.Head, list.Tail = nil, nil
  }
  list.Size--
  return node
}

刪除尾部節(jié)點

實現(xiàn)思路

  1. 判斷尾部節(jié)點是否為空
  2. 取出尾部節(jié)點
  3. 判斷尾部節(jié)點的上一個節(jié)點是否存在
    1. 不存在:則說明只有一個節(jié)點, 刪除本身,無須移動指針位置
    2. 如果存在上一個節(jié)點,則尾部的上一個節(jié)點即是尾部節(jié)點.
//刪除尾部節(jié)點
func (list *DoubleList) RemoveTail() *DoubleNode {
  //判斷尾部節(jié)點是否為空
  if list.Tail == nil {
    return nil
  }
  list.lock.Lock()
  defer list.lock.Unlock()
  //取出尾部節(jié)點
  node := list.Tail
  //判斷尾部節(jié)點的上一個是否存在
  if node.Prev != nil {
    list.Tail = node.Prev
    list.Tail.Next = nil
  }
  list.Size--
  return node
}

刪除任意元素

實現(xiàn)思路

  1. 判斷是否是頭部節(jié)點
  2. 判斷是否是尾部節(jié)點
  3. 否則為中間節(jié)點,需要移動上下節(jié)點的指針位置.并實現(xiàn)元素刪除
    1. 將上一個節(jié)點的下一節(jié)點指針指向下節(jié)點
    2. 將下一個節(jié)點的上一節(jié)點指針指向上節(jié)點
//刪除任意元素
func (list *DoubleList) Remove(node *DoubleNode) *DoubleNode {
  //判斷是否是頭部節(jié)點
  if node == list.Head {
    return list.RemoveHead()
  }
  //判斷是否是尾部節(jié)點
  if node == list.Tail {
    return list.RemoveTail()
  }
  list.lock.Lock()
  defer list.lock.Unlock()
  //節(jié)點為中間節(jié)點
  //則需要:
  //將上一個節(jié)點的下一節(jié)點指針指向下節(jié)點
  //將下一個節(jié)點的上一節(jié)點指針指向上節(jié)點
  node.Prev.Next = node.Next
  node.Next.Prev = node.Prev
  list.Size--
  return node
}

查看完整源碼

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Golang http包構(gòu)建RESTful API的實現(xiàn)

    Golang http包構(gòu)建RESTful API的實現(xiàn)

    在Go語言中實現(xiàn)RESTful API可以利用標(biāo)準(zhǔn)庫net/http提供的功能,它允許你輕松地創(chuàng)建和處理HTTP請求,本文主要介紹了Golang http包構(gòu)建RESTful API的實現(xiàn),感興趣的可以了解一下
    2024-01-01
  • GoLang完整實現(xiàn)快速列表

    GoLang完整實現(xiàn)快速列表

    這篇文章主要介紹了GoLang完整實現(xiàn)快速列表,列表是一種非連續(xù)的存儲容器,由多個節(jié)點組成,節(jié)點通過一些 變量 記錄彼此之間的關(guān)系,列表有多種實現(xiàn)方法,如單鏈表、雙鏈表等
    2022-12-12
  • go mock server的簡易實現(xiàn)示例

    go mock server的簡易實現(xiàn)示例

    這篇文章主要為大家介紹了go mock server的簡易實現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-07-07
  • 深入講解Go語言中函數(shù)new與make的使用和區(qū)別

    深入講解Go語言中函數(shù)new與make的使用和區(qū)別

    大家都知道Go語言中的函數(shù)new與函數(shù)make一直是新手比較容易混淆的東西,看著相似,但其實不同,不過解釋兩者之間的不同也非常容易,下面這篇文章主要給大家介紹了關(guān)于Go語言中函數(shù)new與make區(qū)別的相關(guān)資料,需要的朋友可以參考下。
    2017-10-10
  • 快速掌握Go 語言 HTTP 標(biāo)準(zhǔn)庫的實現(xiàn)方法

    快速掌握Go 語言 HTTP 標(biāo)準(zhǔn)庫的實現(xiàn)方法

    基于HTTP構(gòu)建的服務(wù)標(biāo)準(zhǔn)模型包括兩個端,客戶端(Client)和服務(wù)端(Server),這篇文章主要介紹了Go 語言HTTP標(biāo)準(zhǔn)庫的實現(xiàn)方法,需要的朋友可以參考下
    2022-07-07
  • golang語法使用的注意事項

    golang語法使用的注意事項

    這篇文章主要給大家介紹了關(guān)于golang語法使用的一些注意事項,Golang是一種靜態(tài)類型的編程語言,它支持基本的數(shù)據(jù)類型,包括整型、浮點型、布爾型、字符串型,需要的朋友可以參考下
    2023-07-07
  • GoLang內(nèi)存模型詳細(xì)講解

    GoLang內(nèi)存模型詳細(xì)講解

    go官方介紹go內(nèi)存模型的時候說:探究在什么條件下,goroutine 在讀取一個變量的值的時,能夠看到其它 goroutine 對這個變量進(jìn)行的寫的結(jié)果,Go內(nèi)存模型規(guī)定了一些條件,在這些條件下,在一個goroutine中讀取變量返回的值能夠確保是另一個goroutine中對該變量寫入的值
    2022-12-12
  • Go get命令使用socket代理的方法

    Go get命令使用socket代理的方法

    由于某些不可描述的原因,國內(nèi)使用 go get 命令安裝某些包的時候會超時導(dǎo)致失敗,比如 net 包、 sys 包、 tools 包等。這篇文章給大家介紹go get 命令使用socket 代理的方法,感興趣的朋友一起看看吧
    2018-10-10
  • golang croncli 定時器命令詳解

    golang croncli 定時器命令詳解

    定時器是執(zhí)行任務(wù)時的常用功能,配置系統(tǒng)的定時任務(wù)太麻煩,所以就想用golang簡單實現(xiàn)一個定時器命令,包括定時器命令格式、定時執(zhí)行命令的相關(guān)知識,感興趣的朋友跟隨小編一起看看吧
    2022-03-03
  • Go語言并發(fā)爬蟲的具體實現(xiàn)

    Go語言并發(fā)爬蟲的具體實現(xiàn)

    本文主要介紹了Go語言并發(fā)爬蟲的具體實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-12-12

最新評論