基于Go的双向链表实践

基于Go的双向链表实践

前言

关于单向链表、双向链表,我向下面这片文档讲的非常详细了
链表参考地址

说明

本次以双向链表来进行实践

组成元素

  • Element:元素值
    • Prev(*Entry):prev节点
    • Next(*Entry):next节点
  • length

list方法列表

  • Length():链表长度
  • Add(int,interface{}):添加元素到指定位置
  • AddFirst(interface{}):添加元素到头部
  • Clear():清空链表
  • Contains(interface{}):检查链表中是否包含指定元素
  • Get(int):获取指定位置元素
  • GetFirst():获取头部元素
  • GetLast():获取尾部元素
  • Pull():移除头部元素
  • IndexOf(interface{}):查找元素所在index
  • Pop():移除尾部元素
  • Push():追加元素
  • Set(int,interface{}):修改指定位置的元素内容
  • Iterator(func(int,*Entry)):链表遍历
  • MoveTo(int,*Entry):移动元素到指定位置
  • MoveFront(*Entry):移动元素到头部
  • MoveBack(*Entry):移动元素到尾部
  • AsList():获取链表返回slice

Entry方法列表

  • HasNext():判断是否还有next节点
  • Next():获取next节点
  • Value():获取节点元素内容

大致思路

  • 节点控制(prev、next)
  • 递归

不足

  • 本次实践中MoveTo实现是先将当前节点从链表中移除,再进行Add操作,这样以来,其实失去了链表本身的优势(修改效率高),本来想做成通过改变节点指向来控制,但由于时间关系没能完成。如有提出更好方法的小伙伴,请在下方留言,谢谢!

代码

package list

import (
	"errors"
)

type (
	Entry struct {
		prev    *Entry
		next    *Entry
		element interface{}
	}
	LinkedList struct {
		element *Entry
		length  int
	}
)

var (
	ErrIndexOutOfBound = errors.New("the location out of index range")
)

func NewLinkedList() *LinkedList {
	return &LinkedList{
		element: nil,
	}
}

func (ll *LinkedList) Length() int {
	return ll.length
}

func (ll *LinkedList) Add(location int, v ...interface{}) error {
	length := ll.length
	if location < 0 || location > length {
		return ErrIndexOutOfBound
	}
	elements := ll.newElement(v...)
	prev, next, err := ll.getNodeAtInsertLocation(location)
	if err != nil {
		return err
	}
	first := elements[0]
	if length == 0 {
		ll.element = first
	}
	last := elements[len(elements)-1]
	first.prev = prev
	last.next = next
	if next != nil {
		next.prev = last
	}
	if prev != nil {
		prev.next = first
	}
	if location == 0 {
		ll.element = last
	}
	ll.length += len(elements)
	return nil
}

func (ll *LinkedList) AddFirst(v interface{}) error {
	return ll.Add(0, v)
}

func (ll *LinkedList) Clear() {
	ll.element = nil
	ll.length = 0
}

func (ll *LinkedList) Contains(v interface{}) bool {
	var contain = false
	ll.Iterator(func(index int, entry *Entry) bool {
		if entry.element == v {
			contain = true
			return true
		}
		return false
	})
	return contain
}

func (ll *LinkedList) Get(location int) (*Entry) {
	if location < 0 || location >= ll.length {
		return nil
	}
	var result *Entry
	ll.Iterator(func(index int, entry *Entry) bool {
		if index == location {
			result = entry
			return true
		}
		return false
	})
	return result
}

func (ll *LinkedList) GetFirst() (*Entry) {
	return ll.Get(0)
}

func (ll *LinkedList) GetLast() (*Entry) {
	return ll.Get(ll.Length() - 1)
}

func (ll *LinkedList) Pull() error {
	return ll.Remove(0)
}

func (ll *LinkedList) IndexOf(v interface{}) int {
	var result int
	ll.Iterator(func(index int, entry *Entry) bool {
		if entry.element == v {
			result = index
			return true
		}
		return false
	})
	return result
}

func (ll *LinkedList) Pop() error {
	return ll.Remove(ll.Length() - 1)
}

func (ll *LinkedList) Push(v ...interface{}) error {
	if ll.Length() == 0 {
		return ll.Add(0, v...)
	} else {
		return ll.Add(ll.length, v...)
	}
}

func (ll *LinkedList) Remove(location int) error {
	if location < 0 || location >= ll.length {
		return ErrIndexOutOfBound
	}
	if ll.length == 0 {
		return nil
	}
	if ll.length == 1 {
		ll.Clear()
	}
	var resErr error
	ll.Iterator(func(index int, entry *Entry) bool {
		if index == location {
			if location == 0 {
				el := ll.Get(1)
				el.prev = nil
				ll.element = el
			} else if location == ll.length-1 {
				el := ll.Get(location - 1)
				el.next = nil
			} else {
				prev := ll.Get(location - 1)
				next := ll.Get(location + 1)
				prev.next = next
				next.prev = prev
			}
			ll.length -= 1
			return true
		}
		return false
	})
	return resErr
}

func (ll *LinkedList) Set(locate int, v interface{}) {
	ll.Iterator(func(index int, entry *Entry) bool {
		if index == locate {
			entry.element = v
			return true
		}
		return false
	})
}

func (ll *LinkedList) Iterator(iterator func(index int, entry *Entry) bool) {
	if ll.Length() == 0 {
		return
	}
	var index = 0
	entry := ll.element
	for {
		con := iterator(index, entry)
		if con {
			break
		}
		if entry.HasNext() {
			index++
			entry = entry.Next()
		} else {
			break
		}
	}
}

func (ll *LinkedList) MoveTo(location int, entry *Entry) error {
	if entry == nil {
		return nil
	}
	if location < 0 || location >= ll.length {
		return ErrIndexOutOfBound
	}
	v := entry.element
	ll.Iterator(func(index int, e *Entry) bool {
		if entry.element == e.element {
			ll.Remove(index)
			return true
		}
		return false
	})
	ll.Add(location, v)
	return nil
}

func (ll *LinkedList) MoveFront(entry *Entry) {
	ll.MoveTo(0, entry)
}

func (ll *LinkedList) MoveBack(entry *Entry) {
	ll.MoveTo(ll.length-1, entry)
}

func (ll *LinkedList) newElement(v ...interface{}) []*Entry {
	var elements = make([]*Entry, 0)
	for index, e := range v {
		var prev *Entry
		var element Entry
		element.element = e
		if index != 0 {
			prev = elements[index-1]
			prev.next = &element
			element.prev = prev
		}
		elements = append(elements, &element)
	}
	return elements
}

// returns prev and current element
func (ll *LinkedList) getNodeAtInsertLocation(insertLocation int) (*Entry, *Entry, error) {
	if insertLocation < 0 || insertLocation > ll.Length() {
		return nil, nil, ErrIndexOutOfBound
	}
	if ll.Length() == 0 {
		return nil, nil, nil
	}
	prevIndex := insertLocation - 1
	nextIndex := insertLocation
	var prev *Entry
	var next *Entry

	ll.Iterator(func(index int, entry *Entry) bool {
		if index == prevIndex {
			prev = entry
		}
		if index == nextIndex {
			next = entry
		}
		return false
	})
	return prev, next, nil
}

func (ll *LinkedList) AsList() []interface{} {
	result := make([]interface{}, 0)
	ll.Iterator(func(index int, entry *Entry) bool {
		result = append(result, entry.element)
		return false
	})
	return result
}

func (e *Entry) HasNext() bool {
	return e.next != nil
}

func (e *Entry) Next() *Entry {
	return e.next
}

func (e *Entry) Value() interface{} {
	return e.element
}

总结

源码地址

Github

你可能感兴趣的:(Golang)