婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁(yè) > 知識(shí)庫(kù) > golang雙鏈表的實(shí)現(xiàn)代碼示例

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

熱門(mén)標(biāo)簽:學(xué)海導(dǎo)航地圖標(biāo)注 江西轉(zhuǎn)化率高的羿智云外呼系統(tǒng) 廣州呼叫中心外呼系統(tǒng) 中國(guó)地圖標(biāo)注省會(huì)高清 南通如皋申請(qǐng)開(kāi)通400電話 地圖標(biāo)注的汽車(chē)標(biāo) 西部云谷一期地圖標(biāo)注 浙江高速公路地圖標(biāo)注 高德地圖標(biāo)注口訣

雙鏈表的實(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é)

  1. 學(xué)好算法是一個(gè)不斷積累的過(guò)程
  2. 學(xué)習(xí)時(shí)還需做到知行合一
  3. 實(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)思路

  1. 先判斷容量大小
  2. 判斷頭部是否為空,
    1. 如果為空則添加新節(jié)點(diǎn)
    2. 如果不為空則更改現(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
  }
  //如果沒(méi)有節(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)思路

  1. 判斷頭部是否為空
  2. 將頭部節(jié)點(diǎn)取出來(lái)
  3. 判斷頭部是否有下一個(gè)節(jié)點(diǎn)
    1. 沒(méi)有下一個(gè)節(jié)點(diǎn),則說(shuō)明只有一個(gè)節(jié)點(diǎn), 刪除本身,無(wú)須移動(dòng)指針位置
    2. 如果有下一個(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)取出來(lái)
  node := list.Head
  //判斷頭部是否有下一個(gè)節(jié)點(diǎn)
  if node.Next != nil {
    list.Head = node.Next
    list.Head.Prev = nil
  } else { //如果沒(méi)有下一個(gè)節(jié)點(diǎn), 說(shuō)明只有一個(gè)節(jié)點(diǎn)
    list.Head, list.Tail = nil, nil
  }
  list.Size--
  return node
}

刪除尾部節(jié)點(diǎn)

實(shí)現(xiàn)思路

  1. 判斷尾部節(jié)點(diǎn)是否為空
  2. 取出尾部節(jié)點(diǎn)
  3. 判斷尾部節(jié)點(diǎn)的上一個(gè)節(jié)點(diǎn)是否存在
    1. 不存在:則說(shuō)明只有一個(gè)節(jié)點(diǎn), 刪除本身,無(wú)須移動(dòng)指針位置
    2. 如果存在上一個(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)思路

  1. 判斷是否是頭部節(jié)點(diǎn)
  2. 判斷是否是尾部節(jié)點(diǎn)
  3. 否則為中間節(jié)點(diǎn),需要移動(dòng)上下節(jié)點(diǎn)的指針位置.并實(shí)現(xiàn)元素刪除
    1. 將上一個(gè)節(jié)點(diǎn)的下一節(jié)點(diǎn)指針指向下節(jié)點(diǎn)
    2. 將下一個(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í)有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • 詳解go語(yǔ)言單鏈表及其常用方法的實(shí)現(xiàn)
  • python/golang 刪除鏈表中的元素
  • python/golang實(shí)現(xiàn)循環(huán)鏈表的示例代碼
  • Go實(shí)現(xiàn)雙向鏈表的示例代碼
  • Go語(yǔ)言單鏈表實(shí)現(xiàn)方法
  • go實(shí)現(xiàn)反轉(zhuǎn)鏈表

標(biāo)簽:東營(yíng) 許昌 貴州 常州 曲靖 德宏 吐魯番 保定

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《golang雙鏈表的實(shí)現(xiàn)代碼示例》,本文關(guān)鍵詞  golang,雙鏈,表,的,實(shí)現(xiàn),代碼,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《golang雙鏈表的實(shí)現(xiàn)代碼示例》相關(guān)的同類(lèi)信息!
  • 本頁(yè)收集關(guān)于golang雙鏈表的實(shí)現(xiàn)代碼示例的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    主站蜘蛛池模板: 沁阳市| 安多县| 台江县| 旌德县| 凤庆县| 灵台县| 石屏县| 金川县| 中卫市| 冷水江市| 安阳市| 开封县| 苏尼特右旗| 谢通门县| 凭祥市| 涟水县| 南康市| 伊宁县| 靖江市| 应城市| 黄大仙区| 贡嘎县| 石棉县| 会泽县| 咸阳市| 正镶白旗| 成安县| 平乡县| 柳州市| 罗田县| 阜阳市| 乌恰县| 七台河市| 隆化县| 海门市| 柘城县| 邯郸市| 涿州市| 大田县| 东台市| 阆中市|