2023-04-28 16:14:03 +03:00
|
|
|
package llx
|
|
|
|
|
|
|
|
// Linked list X .
|
|
|
|
// The package implements better variation of
|
|
|
|
// linked list than in standard library since it uses
|
|
|
|
// the new conception of generics.
|
|
|
|
|
2023-07-21 20:38:32 +03:00
|
|
|
import (
|
|
|
|
"github.com/mojosa-software/godat/src/iterx"
|
|
|
|
)
|
|
|
|
|
|
|
|
// The type represents linked list data structure.
|
2023-04-28 16:14:03 +03:00
|
|
|
type LinkedList[V any] struct {
|
|
|
|
// First empty element (not used to store values).
|
|
|
|
// For fast pushing.
|
|
|
|
before *Element[V]
|
|
|
|
// Points to the last for fast appending.
|
|
|
|
last *Element[V]
|
|
|
|
// Length.
|
|
|
|
ln int
|
|
|
|
}
|
|
|
|
|
2023-07-21 20:38:32 +03:00
|
|
|
// The type represents element of the linked list.
|
2023-04-28 16:14:03 +03:00
|
|
|
type Element[V any] struct {
|
|
|
|
next *Element[V]
|
|
|
|
value V
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns new empty linked list storing the V type.
|
|
|
|
func New[V any]() *LinkedList[V] {
|
|
|
|
return &LinkedList[V]{
|
|
|
|
&Element[V]{},
|
|
|
|
nil,
|
|
|
|
0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get length of the linked list.
|
2023-07-21 20:38:32 +03:00
|
|
|
func (ll *LinkedList[V]) Length() int {
|
2023-04-28 16:14:03 +03:00
|
|
|
return ll.ln
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the index-indexed element itself.
|
|
|
|
func (ll *LinkedList[V]) GetEl(index int) (*Element[V], bool) {
|
|
|
|
if ll.ln <= index {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
p := ll.before
|
|
|
|
for i := 0 ; i <= index ; i++ {
|
|
|
|
p = p.next
|
|
|
|
}
|
|
|
|
|
|
|
|
return p, true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the value of index-indexed element.
|
|
|
|
func (ll *LinkedList[V]) Get(index int) (V, bool) {
|
|
|
|
el, ok := ll.GetEl(index)
|
|
|
|
var v V
|
|
|
|
if ok {
|
|
|
|
v = el.value
|
|
|
|
}
|
|
|
|
|
|
|
|
return v, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the new value in i-indexed element.
|
|
|
|
func (ll *LinkedList[V]) Set(i int, v V) (bool) {
|
|
|
|
el, ok := ll.GetEl(i)
|
|
|
|
if !ok {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
el.value = v
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2023-07-21 20:51:24 +03:00
|
|
|
func (ll *LinkedList[V]) Swap(i1, i2 int) {
|
|
|
|
if i1 == i2 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
max := ll.ln - 1
|
|
|
|
if i1 < 0 || i2 < 0 || i1 > max || i2 > max {
|
|
|
|
panic("index out of range")
|
|
|
|
}
|
|
|
|
|
|
|
|
el1, _ := ll.GetEl(i1)
|
|
|
|
el2, _ := ll.GetEl(i2)
|
|
|
|
|
|
|
|
el1.value, el2.value =
|
|
|
|
el2.value, el1.value
|
|
|
|
}
|
|
|
|
|
2023-07-21 20:38:32 +03:00
|
|
|
// Deletes the element by its index.
|
|
|
|
func (ll *LinkedList[V]) Delete(i int) (bool) {
|
2023-04-28 17:01:57 +03:00
|
|
|
if ll.ln <= i {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if i == 0 {
|
|
|
|
ll.before.next =
|
|
|
|
ll.before.next.next
|
|
|
|
ll.ln--
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
el1, _ := ll.GetEl(i-1)
|
|
|
|
if i == ll.ln - 1 {
|
|
|
|
el1.next = nil
|
|
|
|
} else {
|
|
|
|
el2, _ := ll.GetEl(i+1)
|
|
|
|
el1.next = el2
|
|
|
|
}
|
|
|
|
|
|
|
|
ll.ln--
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2023-04-28 16:14:03 +03:00
|
|
|
// Push in the beginning of the list.
|
|
|
|
func (ll *LinkedList[V]) Push(v V) {
|
|
|
|
prevNext := ll.before.next
|
|
|
|
nextNext := &Element[V]{
|
|
|
|
next: prevNext,
|
|
|
|
value: v,
|
|
|
|
}
|
|
|
|
ll.before.next = nextNext
|
|
|
|
|
|
|
|
ll.ln++
|
|
|
|
if ll.ln == 1 {
|
|
|
|
ll.last = ll.before.next
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Append to the end of the list.
|
|
|
|
func (ll *LinkedList[V]) Append(v V) {
|
|
|
|
if ll.ln == 0 {
|
|
|
|
ll.Push(v)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
last := &Element[V]{
|
|
|
|
next: nil,
|
|
|
|
value: v,
|
|
|
|
}
|
|
|
|
|
|
|
|
lastBuf := ll.last
|
|
|
|
lastBuf.next = last
|
|
|
|
ll.last = last
|
|
|
|
|
|
|
|
ll.ln++
|
|
|
|
}
|
|
|
|
|
2023-07-21 20:38:32 +03:00
|
|
|
// Returns the first element of the linked list.
|
2023-04-28 17:51:39 +03:00
|
|
|
func (ll *LinkedList[V]) First() *Element[V] {
|
|
|
|
return ll.before.next
|
|
|
|
}
|
|
|
|
|
2023-07-21 20:38:32 +03:00
|
|
|
// Get elements value.
|
2023-04-28 17:51:39 +03:00
|
|
|
func (ll *Element[V]) Value() V {
|
|
|
|
return ll.value
|
|
|
|
}
|
|
|
|
|
2023-07-21 20:38:32 +03:00
|
|
|
// Returns the next element. If the returned value == nil,
|
|
|
|
// then it is the last element.
|
2023-04-28 17:51:39 +03:00
|
|
|
func (ll *Element[V]) Next() *Element[V] {
|
|
|
|
return ll.next
|
|
|
|
}
|
|
|
|
|
2023-07-21 20:38:32 +03:00
|
|
|
// Returns the last element.
|
2023-04-28 17:51:39 +03:00
|
|
|
func (ll *LinkedList[V]) Last() *Element[V] {
|
|
|
|
return ll.last
|
|
|
|
}
|
|
|
|
|
2023-04-28 16:14:03 +03:00
|
|
|
// Returns a channel of Pair that contains index and the value.
|
2023-07-21 20:38:32 +03:00
|
|
|
func (ll *LinkedList[V]) Chan() iterx.PairChan[int, V] {
|
|
|
|
chn := make(iterx.PairChan[int, V])
|
2023-04-28 16:14:03 +03:00
|
|
|
go func(){
|
|
|
|
i := -1
|
|
|
|
el := ll.before
|
|
|
|
for el.next != nil {
|
|
|
|
i++
|
|
|
|
el = el.next
|
2023-07-21 20:38:32 +03:00
|
|
|
chn <- iterx.Pair[int, V]{
|
|
|
|
K: i,
|
|
|
|
V: el.value,
|
|
|
|
}
|
2023-04-28 16:14:03 +03:00
|
|
|
}
|
|
|
|
close(chn)
|
|
|
|
}()
|
|
|
|
return chn
|
|
|
|
}
|
|
|
|
|