站長資訊網
        最全最豐富的資訊網站

        go語言中list怎么刪除元素

        在go語言中,可以使用remove()函數來刪除list元素,語法“list對象.Remove(element)”,參數element表示要刪除列表元素。element元素不能為空,如果不為空則返回被刪除的元素的值,如果為空則會報異常。

        go語言中list怎么刪除元素

        本教程操作環境:windows7系統、GO 1.18版本、Dell G3電腦。

        go提供了一個list包,類似python的list,可以存儲任意類型的數據,并提供了相應的API,如下:

        type Element     func (e *Element) Next() *Element     func (e *Element) Prev() *Element type List     func New() *List     func (l *List) Back() *Element     func (l *List) Front() *Element     func (l *List) Init() *List     func (l *List) InsertAfter(v interface{}, mark *Element) *Element     func (l *List) InsertBefore(v interface{}, mark *Element) *Element     func (l *List) Len() int     func (l *List) MoveAfter(e, mark *Element)     func (l *List) MoveBefore(e, mark *Element)     func (l *List) MoveToBack(e *Element)     func (l *List) MoveToFront(e *Element)     func (l *List) PushBack(v interface{}) *Element     func (l *List) PushBackList(other *List)     func (l *List) PushFront(v interface{}) *Element     func (l *List) PushFrontList(other *List)     func (l *List) Remove(e *Element) interface{}
        登錄后復制

        其中,remove()函數用于列表list刪除元素,刪除的元素不能為空,如果為空,會報異常。

        Remove(e *Element) interface{}
        登錄后復制

        參數 描述
        e 要刪除列表元素。

        返回值

        • 返回被刪除的元素的值。

        列表刪除元素的示例

        示例1:

        package main import ( 	"container/list" 	"fmt" ) func main() { 	//使用 Remove 在列表中刪除元素 	listHaiCoder := list.New() 	listHaiCoder.PushFront("Hello") 	listHaiCoder.PushFront("HaiCoder") 	element := listHaiCoder.PushFront("Hello") 	removeEle := listHaiCoder.Remove(element) 	fmt.Println("RemoveElement =", removeEle) 	for i := listHaiCoder.Front(); i != nil; i = i.Next() { 		fmt.Println("Element =", i.Value) 	} }
        登錄后復制

        go語言中list怎么刪除元素

        分析:

        • 我們通過 list.New 創建了一個列表 listHaiCoder,接著使用 PushFront 函數在列表中插入三個元素,接著使用 Remove 函數刪除了最后插入的元素。

        • 最后,我們打印被刪除的元素和刪除后的列表,Remove 函數返回的是被刪除的元素的值,同時,我們發現最后插入的元素已經被成功從列表刪除了。

        示例2:刪除空元素

        package main import ( 	"container/list" 	"fmt" ) func main() { 	//使用 Remove 在列表中刪除空元素,報錯 	listHaiCoder := list.New() 	listHaiCoder.PushFront("Hello") 	listHaiCoder.PushFront("HaiCoder") 	listHaiCoder.Remove(nil) }
        登錄后復制

        程序運行后,控制臺輸出如下:

        go語言中list怎么刪除元素

        擴展知識:list刪除所有元素

        借助list包提供的API,list用起來確實挺方便,但是在使用過程中,如果不注意就會遇到一些難以發現的坑,導致程序結果不是預想的那樣。這里要說的坑是通過for循環遍歷list,并刪除所有元素時會遇到的問題。例如,下面這個示例程序創建了一個list,并依次將0-3存入,然后通過for循環遍歷list刪除所有元素:

        package main import (     "container/list"     "fmt" ) func main() {     l := list.New()     l.PushBack(0)     l.PushBack(1)     l.PushBack(2)     l.PushBack(3)     fmt.Println("original list:")     prtList(l)     fmt.Println("deleted list:")     for e := l.Front(); e != nil; e = e.Next() {         l.Remove(e)     }     prtList(l) } func prtList(l *list.List) {     for e := l.Front(); e != nil; e = e.Next() {         fmt.Printf("%v ", e.Value)     }     fmt.Printf("n") }
        登錄后復制

        運行程序輸出如下:

        original list: 0 1 2 3 deleted list: 1 2 3
        登錄后復制

        從輸出可以知道,list中的元素并沒有被完全刪除,僅刪除了第一個元素0,和最初設想不一樣,按照go的使用習慣,遍歷一個list并刪除所有元素寫法應該如下:

        for e := l.Front(); e != nil; e = e.Next() {     l.Remove(e) }
        登錄后復制

        但是根據上面示例代碼的輸出,這樣刪除list所有元素是無效的,那么問題出在哪呢?由for循環的機制可以知道,既然刪除了第一個元素,沒有刪除第二個元素,肯定是第二次循環的條件無效,才導致循環退出,即執行完下面語句后:

        l.Remove(e)
        登錄后復制

        e應該為nil,所以循環退出。在for循環中的l.Remove(e)語句前添加打印語句驗證,例如添加如下語句:

        fmt.Println("delete a element from list")
        登錄后復制

        運行程序輸出如下:

        original list: 0 1 2 3 deleted list: delete a element from list 1 2 3
        登錄后復制

        可以看到,確實只循環了一次,循環就結束了。即當執行完語句l.Remove(e)后,e等于e.Next(),因為e.Next()為nil,導致e為nil,循環退出。為什么e.Next()會是nil呢?通過查看go list源碼,如下所示:

        // remove removes e from its list, decrements l.len, and returns e. func (l *List) remove(e *Element) *Element {     e.prev.next = e.next     e.next.prev = e.prev     e.next = nil // avoid memory leaks     e.prev = nil // avoid memory leaks     e.list = nil     l.len--     return e } // Remove removes e from l if e is an element of list l. // It returns the element value e.Value. func (l *List) Remove(e *Element) interface{} {     if e.list == l {         // if e.list == l, l must have been initialized when e was inserted         // in l or l == nil (e is a zero Element) and l.remove will crash         l.remove(e)     }     return e.Value }
        登錄后復制

        由源碼中可以看到,當執行l.Remove(e)時,會在內部調用l.remove(e)方法刪除元素e,為了避免內存泄漏,會將e.next和e.prev賦值為nil,這就是問題根源。

        修正程序如下:

        package main import (     "container/list"     "fmt" ) func main() {     l := list.New()     l.PushBack(0)     l.PushBack(1)     l.PushBack(2)     l.PushBack(3)     fmt.Println("original list:")     prtList(l)     fmt.Println("deleted list:")     var next *list.Element     for e := l.Front(); e != nil; e = next {         next = e.Next()         l.Remove(e)     }     prtList(l) } func prtList(l *list.List) {     for e := l.Front(); e != nil; e = e.Next() {         fmt.Printf("%v ", e.Value)     }     fmt.Printf("n") }
        登錄后復制

        運行程序輸出如下:

        original list: 0 1 2 3 deleted list:
        登錄后復制

        可以看見,list中的所有元素已經被正確刪除。

        贊(0)
        分享到: 更多 (0)
        網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
        主站蜘蛛池模板: 国产区精品一区二区不卡中文| 久久91精品久久91综合| 亚洲国产av无码精品| 国产精品日本一区二区在线播放 | 麻豆精品久久久一区二区| 日韩精品无码永久免费网站| 国产精品一区二区久久| 乱色精品无码一区二区国产盗| 久久精品免费大片国产大片| 精品久久香蕉国产线看观看亚洲| 一本色道久久88精品综合| 精品一区二区三区四区在线| 97久人人做人人妻人人玩精品| 国产乱人伦偷精品视频免下载| 亚洲七七久久精品中文国产| 久久99精品国产麻豆不卡| 国产精品粉嫩美女在线观看| 精品乱人伦一区二区三区| 国产三级久久久精品麻豆三级| 亚洲精品国产精品乱码不99| 日产国产精品亚洲系列| 国产一区二区三区精品视频 | 国产99精品久久| 精品黑人一区二区三区| 青草国产精品久久久久久| 亚洲国产一成人久久精品| 亚洲国产精品无码久久久秋霞2| 亚洲精品WWW久久久久久| 久久久久久国产精品免费免费| 国产精品永久免费| 国产亚洲精品国看不卡| 国产乱人伦偷精品视频| 国产精品午夜久久| 精品人妻少妇一区二区三区在线| 国内精品久久久久久久久电影网| 国产精品午夜免费观看网站| 国产一区二区三区在线观看精品| 精品国产黑色丝袜高跟鞋| 久久久无码精品亚洲日韩软件| 91精品国产91久久久久久| 精品人妻少妇嫩草AV无码专区|