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

Go?語言數(shù)據(jù)結(jié)構(gòu)如何實(shí)現(xiàn)抄一個(gè)list示例詳解

 更新時(shí)間:2023年04月16日 11:54:44   作者:陪我去看海  
這篇文章主要為大家介紹了Go?語言數(shù)據(jù)結(jié)構(gòu)如何實(shí)現(xiàn)抄一個(gè)list示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

前言

閑來無事,自己實(shí)現(xiàn)一個(gè) Go提供的list

  • list是Go提供的一個(gè)內(nèi)置的包
  • 內(nèi)部就是實(shí)現(xiàn)了一個(gè)雙向循環(huán)鏈表以及各種API

目標(biāo)明確:就是實(shí)現(xiàn)一個(gè)雙向循環(huán)鏈表

文章源碼

list是個(gè)啥

在開始做之前,還是要先了解一下鏈表這個(gè)數(shù)據(jù)結(jié)構(gòu) ,長話短說:

  • 線性表的鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)稱為鏈表,如:
a.next = b
a.prev = c
b.next = c
b.prev = a
c.next = a
c.prev = b

這就是一個(gè)雙向循環(huán)鏈表

  • 鏈表可以提升存儲(chǔ)空間的利用率,實(shí)現(xiàn)了存儲(chǔ)空間動(dòng)態(tài)管理的鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)

接下來,我們來看看Go官方都為這個(gè)list提供了哪些操作,我們逐一實(shí)現(xiàn)

  • New:創(chuàng)建一個(gè)鏈表
  • Init:初始化一個(gè)鏈表
  • Back:返回鏈表中的最后一個(gè)元素
  • Front:返回鏈表中的第一個(gè)元素
  • InsertAfter(e,at):將e加入at元素后
  • InsertBefore(e,at):將e加入at元素前
  • Len:返回list的長度
  • PushBack(e):將e成為鏈表的最后一個(gè)元素
  • PushFront(e):將e成為鏈表的第一個(gè)元素
  • Remove(e):將list上的e刪除

list結(jié)構(gòu)

定義list結(jié)構(gòu),以及l(fā)ist內(nèi)部node節(jié)點(diǎn)的結(jié)構(gòu),這里采用struct實(shí)現(xiàn)

type Element struct {
	prev, next *Element
	Value      any
}
type List struct {
	root Element
	len  int
}

Init & New

Init就是提供初始化一個(gè)環(huán)鏈表的方法,并返回這個(gè)環(huán)形鏈表

之所以把 Init 和 New 放在一起,是因?yàn)樵?New 函數(shù)中其實(shí)就是對(duì) Init 的一層包裝,這樣就可以實(shí)現(xiàn)Go中的包名.New方法,比如:errors.New()

// 初始化一個(gè) 環(huán)list
func (list *List) Init() *List {
	// 形成環(huán)
	list.root.next = &list.root
	list.root.prev = &list.root
	list.len = 0
	return list
}
func NewList() *List {
	return new(List).Init()
}

InsertAfter & InsertBefore & PushBack & PushFront

這兩個(gè)方法的作用類似,就是將 e 插入到 at 的后/前位置

這里我們先看一個(gè)圖:

這個(gè)圖片就是一個(gè)雙向環(huán)形鏈表,我們要在這個(gè)里面進(jìn)行插入元素操作,比如,我們要插入 e 到 e1 前面我們應(yīng)該怎么做?

  • 將e的下一個(gè)變?yōu)閑1:e.next = e1
  • 將e的上一個(gè)變?yōu)閑1的上一個(gè):e.prev = e1.prev
  • 將e的上一個(gè)的下一個(gè)變?yōu)樽约海篹.prev.next = e
  • 將e的下一個(gè)的上一個(gè)變?yōu)樽约海篹.next.prev = e

這樣就完成了插入,回到方法實(shí)現(xiàn)上,一個(gè)是插入之后,一個(gè)插入之前,那么我們是不是可以看作是相同操作,其實(shí)都已插入操作,只是位置的變化。

這時(shí)候想象一下,比如讓你 e 插入 at 之前,但是只提供了,參數(shù)1插入?yún)?shù)2后面的操作,如何辦到呢?

將 e 插入到 at 的前一個(gè)的后面,是不是就ok了,就相當(dāng)于自己讓別人插個(gè)隊(duì),你在我前面的后面站就行了

// Insert 插入:將 currentElement 插入至 originElement 后
func (list *List) Insert(currentElement, originElement *Element) *Element {
	currentElement.next = originElement.next
	currentElement.prev = originElement
	currentElement.prev.next = currentElement
	currentElement.next.prev = currentElement
	list.len++
	return currentElement
}
// InsertAfter 插入在之后
func (list *List) InsertAfter(currentElement, originElement *Element) *Element {
	return list.Insert(currentElement, originElement)
}
// InsertBefore 插入在之前
func (list *List) InsertBefore(currentElement, originElement *Element) *Element {
	return list.Insert(currentElement, originElement.prev)
}

這樣一來,好像把 PushBack 和 PushFront都實(shí)現(xiàn)了,這就是封裝的好處

// PushBack 插入一個(gè)元素在最后
func (list *List) PushBack(originElement *Element) *Element {
	list.InsertBefore(originElement, &list.root)
	return originElement
}
// PushFront 插入一個(gè)元素在最前
func (list *List) PushFront(originElement *Element) *Element {
	list.InsertAfter(originElement, &list.root)
	return originElement
}

Back & Front

這兩個(gè)方式抽象上說,也是一樣的功能,一個(gè)是返回鏈表最后一個(gè),另一個(gè)是返回鏈表第一個(gè),因?yàn)檫@里提供了頭結(jié)點(diǎn),所以特別簡單

最后一個(gè)節(jié)點(diǎn) = 頭結(jié)點(diǎn).prev

第一個(gè)節(jié)點(diǎn) = 頭結(jié)點(diǎn).next

// Back 返回最后一個(gè)元素
func (list *List) Back() *Element {
	if list.len == 0 {
		return nil
	}
	// 頭結(jié)點(diǎn)的上一個(gè)就是最后一個(gè)
	return list.root.prev
}
// Front 返回第一個(gè)元素
func (list *List) Front() *Element {
	if list.len == 0 {
		return nil
	}
	// 頭結(jié)點(diǎn)的下一個(gè)就是第一個(gè)元素
	return list.root.next
}

Remove

Remove方法就是提供了,刪除鏈表上的某個(gè)元素,怎么樣才能刪除某個(gè)節(jié)點(diǎn)呢,本質(zhì)也就是讓前后的節(jié)點(diǎn)相互鏈表,我就被排擠出來了,這樣就可以實(shí)現(xiàn)刪除

  • 將要?jiǎng)h除的元素 e.next.prev = e.prev
  • 將要?jiǎng)h除的元素 e.prev.next = e.next
// Remove 刪除某個(gè)元素
func (list *List) Remove(originElement *Element) (any,error) {
	if originElement == &list.root {
		return nil, errors.New("the origin Element can not be list.root")
	}
	for e := list.root.next; e != &list.root; e = e.next {
		if e == originElement {
			e.prev.next = e.next
			e.next.prev = e.prev
			return e.Value, nil
		} else {
			continue
		}
	}
	return nil, errors.New("the origin Element dose not belong to the list")
}

總結(jié)

  • 鏈表可以實(shí)現(xiàn)存儲(chǔ)空間動(dòng)態(tài)管理,它是一個(gè)鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)
  • 封裝的代碼更具靈活性

以上就是Go 語言數(shù)據(jù)結(jié)構(gòu)如何實(shí)現(xiàn)抄一個(gè)list示例詳解的詳細(xì)內(nèi)容,更多關(guān)于Go 語言數(shù)據(jù)結(jié)構(gòu)list的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 學(xué)習(xí)GO編程必備知識(shí)匯總

    學(xué)習(xí)GO編程必備知識(shí)匯總

    這篇文章主要介紹了學(xué)習(xí)GO編程必備知識(shí)匯總的相關(guān)資料,需要的朋友可以參考下
    2016-07-07
  • golang redis中Pipeline通道的使用詳解

    golang redis中Pipeline通道的使用詳解

    本文主要介紹了golang redis中Pipeline通道的使用詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • Go語言中的range用法實(shí)例分析

    Go語言中的range用法實(shí)例分析

    這篇文章主要介紹了Go語言中的range用法,實(shí)例分析了range的功能與使用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-02-02
  • Golang實(shí)現(xiàn)斷點(diǎn)續(xù)傳功能

    Golang實(shí)現(xiàn)斷點(diǎn)續(xù)傳功能

    這篇文章主要為大家詳細(xì)介紹了Golang實(shí)現(xiàn)斷點(diǎn)續(xù)傳、復(fù)制文件功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-07-07
  • GoFrame 框架緩存查詢結(jié)果的示例詳解

    GoFrame 框架緩存查詢結(jié)果的示例詳解

    GoFrame的gdb對(duì)查詢結(jié)果的緩存處理是不是非常的優(yōu)雅。尤其是*gcache.Cache對(duì)象采用了適配器設(shè)計(jì)模式,可以輕松實(shí)現(xiàn)從單進(jìn)程內(nèi)存緩存切換為分布式的Redis緩存,本文重點(diǎn)給大家介紹GoFrame 如何優(yōu)雅的緩存查詢結(jié)果,感興趣的朋友一起看看吧
    2022-06-06
  • Golang實(shí)現(xiàn)簡易的rpc調(diào)用

    Golang實(shí)現(xiàn)簡易的rpc調(diào)用

    RPC指(Remote Procedure Call Protocol)遠(yuǎn)程過程調(diào)用協(xié)議。本文將實(shí)現(xiàn)利用Golang進(jìn)行rpc調(diào)用(只實(shí)現(xiàn)一個(gè)rpc框架基本的功能,不對(duì)性能做保證),需要的可以參考一下
    2023-03-03
  • Go語言如何處理HTTP身份驗(yàn)證教程示例

    Go語言如何處理HTTP身份驗(yàn)證教程示例

    這篇文章主要為大家介紹了Go語言如何處理HTTP身份驗(yàn)證教程示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01
  • golang模擬實(shí)現(xiàn)帶超時(shí)的信號(hào)量示例代碼

    golang模擬實(shí)現(xiàn)帶超時(shí)的信號(hào)量示例代碼

    這篇文章主要給大家介紹了關(guān)于golang模擬實(shí)現(xiàn)帶超時(shí)的信號(hào)量的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面跟著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-09-09
  • go-micro集成RabbitMQ實(shí)戰(zhàn)和原理詳解

    go-micro集成RabbitMQ實(shí)戰(zhàn)和原理詳解

    本文主要介紹go-micro使用RabbitMQ收發(fā)數(shù)據(jù)的方法和原理,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-05-05
  • Go實(shí)現(xiàn)線程池(工作池)的兩種方式實(shí)例詳解

    Go實(shí)現(xiàn)線程池(工作池)的兩種方式實(shí)例詳解

    這篇文章主要介紹了Go實(shí)現(xiàn)線程池(工作池)的兩種方式實(shí)例詳解,需要的朋友可以參考下
    2022-04-04

最新評(píng)論