Golang标准库——container

  • list
  • heap
  • ring

list

list包实现了双向链表。

type Element

type Element struct {
    // Next and previous pointers in the doubly-linked list of elements.
    // To simplify the implementation, internally a list l is implemented
    // as a ring, such that &l.root is both the next element of the last
    // list element (l.Back()) and the previous element of the first list
    // element (l.Front()).
    next, prev *Element

    // The list to which this element belongs.
    list *List

    // The value stored with this element.
    Value interface{}
}

Element类型代表是双向链表的一个元素。

func (*Element) Next

func (e *Element) Next() *Element

Next返回链表的后一个元素或者nil。

func (*Element) Prev

func (e *Element) Prev() *Element

Prev返回链表的前一个元素或者nil。

type List

type List struct {
    root Element // sentinel list element, only &root, root.prev, and root.next are used
    len  int     // current list length excluding (this) sentinel element
}

代表一个双向链表。List零值为一个空的、可用的链表。

func New

func New() *List

New创建一个链表。

func (*List) Init

func (l *List) Init() *List

Init清空链表。

func (*List) Len

func (l *List) Len() int

Len返回链表中元素的个数,复杂度O(1)。

func (*List) Front

func (l *List) Front() *Element

Front返回链表第一个元素或nil。

func (*List) Back

func (l *List) Back() *Element

Back返回链表最后一个元素或nil。

func (*List) PushFront

func (l *List) PushFront(v interface{}) *Element

PushBack将一个值为v的新元素插入链表的第一个位置,返回生成的新元素。

func (*List) PushFrontList

func (l *List) PushFrontList(other *List)

PushFrontList创建链表other的拷贝,并将拷贝的最后一个位置连接到链表l的第一个位置。

func (*List) PushBack

func (l *List) PushBack(v interface{}) *Element

PushBack将一个值为v的新元素插入链表的最后一个位置,返回生成的新元素。

func (*List) PushBackList

func (l *List) PushBackList(other *List)

PushBack创建链表other的拷贝,并将链表l的最后一个位置连接到拷贝的第一个位置。

func (*List) InsertBefore

func (l *List) InsertBefore(v interface{}, mark *Element) *Element

InsertBefore将一个值为v的新元素插入到mark前面,并返回生成的新元素。如果mark不是l的元素,l不会被修改。

func (*List) InsertAfter

func (l *List) InsertAfter(v interface{}, mark *Element) *Element

InsertAfter将一个值为v的新元素插入到mark后面,并返回新生成的元素。如果mark不是l的元素,l不会被修改。

func (*List) MoveToFront

func (l *List) MoveToFront(e *Element)

MoveToFront将元素e移动到链表的第一个位置,如果e不是l的元素,l不会被修改。

func (*List) MoveToBack

func (l *List) MoveToBack(e *Element)

MoveToBack将元素e移动到链表的最后一个位置,如果e不是l的元素,l不会被修改。

func (*List) MoveBefore

func (l *List) MoveBefore(e, mark *Element)

MoveBefore将元素e移动到mark的前面。如果e或mark不是l的元素,或者e==mark,l不会被修改。

func (*List) MoveAfter

func (l *List) MoveAfter(e, mark *Element)

MoveAfter将元素e移动到mark的后面。如果e或mark不是l的元素,或者e==mark,l不会被修改。

func (*List) Remove

func (l *List) Remove(e *Element) interface{}

Remove删除链表中的元素e,并返回e.Value。

func main() {
   l := list.New()

   for i := 0; i < 5; i++ {
      l.PushBack(i)
   }
   for e := l.Front(); e != nil; e = e.Next() {
      fmt.Print(e.Value) //01234
   }
   fmt.Println("")
   fmt.Println(l.Front().Value) //0
   fmt.Println(l.Back().Value)  //4

   l.InsertAfter(6, l.Front())  //首部元素之后插入一个值为6的元素
   for e := l.Front(); e != nil; e = e.Next() {
      fmt.Print(e.Value) //061234
   }
   fmt.Println("")
   l.MoveBefore(l.Front().Next(), l.Front()) //首部两个元素位置互换
   for e := l.Front(); e != nil; e = e.Next() {
      fmt.Print(e.Value) //601234
   }
   fmt.Println("")
   l.MoveToFront(l.Back()) //将尾部元素移动到首部
   for e := l.Front(); e != nil; e = e.Next() {
      fmt.Print(e.Value) //,460123
   }
   fmt.Println("")
   l2 := list.New()

   for i := 0; i < 5; i++ {
      l2.PushBack(i)
   }
   l2.PushBackList(l) //将l中元素放在l2的末尾
   for e := l2.Front(); e != nil; e = e.Next() {
      fmt.Print(e.Value) //01234460123
   }
   fmt.Println("")

   l.Init() //清空l
   fmt.Print(l.Len())                  //0
   for e := l.Front(); e != nil; e = e.Next() {
      fmt.Print(e.Value) //
   }
}

heap

heap包提供了对任意类型(实现了heap.Interface接口)的堆操作。(最小)堆是具有“每个节点都是以其为根的子树中最小值”属性的树。

树的最小元素为其根元素,索引0的位置。

heap是常用的实现优先队列的方法。要创建一个优先队列,实现一个具有使用(负的)优先级作为比较的依据的Less方法的Heap接口,如此一来可用Push添加项目而用Pop取出队列最高优先级的项目。

type Interface

type Interface interface {
    sort.Interface
    Push(x interface{}) // 向末尾添加元素
    Pop() interface{}   // 从末尾删除元素
}

任何实现了本接口的类型都可以用于构建最小堆。最小堆可以通过heap.Init建立,数据是递增顺序或者空的话也是最小堆。最小堆的约束条件是:

!h.Less(j, i) for 0 <= i < h.Len() and 2*i+1 <= j <= 2*i+2 and j < h.Len()

注意接口的Push和Pop方法是供heap包调用的,请使用heap.Push和heap.Pop来向一个堆添加或者删除元素。

func Init

func Init(h Interface)

一个堆在使用任何堆操作之前应先初始化。Init函数对于堆的约束性是幂等的(多次执行无意义),并可能在任何时候堆的约束性被破坏时被调用。本函数复杂度为O(n),其中n等于h.Len()。

func Push

func Push(h Interface, x interface{})

向堆h中插入元素x,并保持堆的约束性。复杂度O(log(n)),其中n等于h.Len()。

func Pop

func Pop(h Interface) interface{}

删除并返回堆h中的最小元素(不影响约束性)。复杂度O(log(n)),其中n等于h.Len()。等价于Remove(h, 0)。

func Remove

func Remove(h Interface, i int) interface{}

删除堆中的第i个元素,并保持堆的约束性。复杂度O(log(n)),其中n等于h.Len()。

func Fix

func Fix(h Interface, i int)

在修改第i个元素后,调用本函数修复堆,比删除第i个元素后插入新元素更有效率。

复杂度O(log(n)),其中n等于h.Len()。

ring

ring实现了环形链表的操作。

type Ring

type Ring struct {
    next, prev *Ring
    Value      interface{} // for use by client; untouched by this library
}

Ring类型代表环形链表的一个元素,同时也代表链表本身。环形链表没有头尾;指向环形链表任一元素的指针都可以作为整个环形链表看待。Ring零值是具有一个(Value字段为nil的)元素的链表。

func New

func New(n int) *Ring

New创建一个具有n个元素的环形链表。

func (*Ring) Len

func (r *Ring) Len() int

Len返回环形链表中的元素个数,复杂度O(n)。

func (*Ring) Next

func (r *Ring) Next() *Ring

返回后一个元素,r不能为空。

func (*Ring) Prev

func (r *Ring) Prev() *Ring

返回前一个元素,r不能为空。

func (*Ring) Move

func (r *Ring) Move(n int) *Ring

返回移动n个位置(n>=0向前移动,n<0向后移动)后的元素,r不能为空。

func (*Ring) Link

func (r *Ring) Link(s *Ring) *Ring

Link连接r和s,并返回r原本的后继元素r.Next()。r不能为空。

如果r和s指向同一个环形链表,则会删除掉r和s之间的元素,删掉的元素构成一个子链表,返回指向该子链表的指针(r的原后继元素);如果没有删除元素,则仍然返回r的原后继元素,而不是nil。如果r和s指向不同的链表,将创建一个单独的链表,将s指向的链表插入r后面,返回s原最后一个元素后面的元素(即r的原后继元素)。

func (*Ring) Unlink

func (r *Ring) Unlink(n int) *Ring

删除链表中n % r.Len()个元素,从r.Next()开始删除。如果n % r.Len() == 0,不修改r。返回删除的元素构成的链表,r不能为空。

func (*Ring) Do

func (r *Ring) Do(f func(interface{}))

对链表的每一个元素都执行f(正向顺序),注意如果f改变了*r,Do的行为是未定义的。

func main() {
   r := ring.New(10) //初始长度10
   for i := 0; i < r.Len(); i++ {
      r.Value = i
      r = r.Next()
   }
   for i := 0; i < r.Len()*2; i++ {
      fmt.Print(r.Value)
      fmt.Print("|")
      r = r.Next()
   } // 0|1|2|3|4|5|6|7|8|9|0|1|2|3|4|5|6|7|8|9|
   fmt.Println("")
   r = r.Move(16)
   fmt.Println(r.Value) // 6
   r1 := r.Unlink(29)   //移除19%10=9个元素
   for i := 0; i < r1.Len(); i++ {
      fmt.Print(r1.Value)
      fmt.Print("|")
      r1 = r1.Next()
   }  // 7|8|9|0|1|2|3|4|5|1
   fmt.Println("")
   fmt.Println(r.Len())  // 1
   fmt.Println(r1.Len()) // 9

   r = r.Link(r1)
   fmt.Println(r.Len())  // 1

   r.Do(func(i interface{}) {
      fmt.Print(i.(int) * 5)
      fmt.Print("|")
   }) // 30|35|40|45|0|5|10|15|20|25|
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,080评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,422评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,630评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,554评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,662评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,856评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,014评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,752评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,212评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,541评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,687评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,347评论 4 331
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,973评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,777评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,006评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,406评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,576评论 2 349

推荐阅读更多精彩内容