don't stop believing

Doubly Linked List - 이중 연결 리스트 본문

Golang/Basic

Doubly Linked List - 이중 연결 리스트

Tongchun 2019. 5. 13. 17:13

이번에는 자료구조입니다.
이중 연결 리스트는 하나의 노드(데이터)가 앞 뒤 노드를 참조하는 구조입니다.

https://en.wikipedia.org/wiki/Doubly_linked_list

 

Doubly linked list - Wikipedia

In computer science, a doubly linked list is a linked data structure that consists of a set of sequentially linked records called nodes. Each node contains three fields: two link fields (references to the previous and to the next node in the sequence of no

en.wikipedia.org

Go로 구현한 코드입니다.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
package main
import "sync"
import "fmt"
// List .
type List struct {
head *Item // .
last *Item // .
len int // List .
locker sync.RWMutex // (Item) , lock .
}
// Item .
type Item struct {
Val interface{} // . go interface{} .
prev *Item // .
next *Item // .
list *List // (Item) List .
}
// List .
func New() *List {
list := &List{} // List .
list.len = 0 // List 0.
return list
}
// List Item .
func Insert(value interface{}, list *List) *List {
// Item value, List head, last, list .
// newItem Item .
newItem := &Item{value, list.last, list.head, list}
// list Lock . Unlock .
list.locker.Lock()
defer list.locker.Unlock()
// list head nil List Item .
// list head last Item .
if list.head == nil {
list.head = newItem
list.last = newItem
} else {
list.last.next = newItem
list.head.prev = newItem
list.head = newItem
}
list.len++
return list
}
// List Item .
func (list *List) First() *Item {
return list.head
}
// List Item .
func (list *List) Last() *Item {
return list.last
}
// Item Item .
func (item *Item) Prev() *Item {
return item.prev
}
// Item Item .
func (item *Item) Next() *Item {
return item.next
}
// List Item .
func Has(value interface{}, list *List) bool {
if list.head == nil {
return false
}
first := list.First()
for {
if first.Val == value {
return true
} else {
if first.next != nil {
first = first.next
} else {
return false
}
}
}
}
// List Item .
// RLock/RUnlock . .
func Remove(value interface{}, list *List) *List {
list.locker.RLock()
if list.head == nil {
return list
}
list.locker.RUnlock()
list.locker.RLock()
first := list.First()
last := list.Last()
list.locker.RUnlock()
list.locker.Lock()
defer list.locker.Unlock()
for {
if last.next == nil {
return list
}
if first.Val == value {
first.next.prev = first.prev
first.prev.next = first.next
first.prev = nil
first.next = nil
first.Val = nil
first.list = nil
list.len--
return list
} else {
first = first.next
}
}
}
// List .
func Length(list *List) int {
return list.len
}
func main() {
list := New()
list = Insert(1, list)
list = Insert(2, list)
list = Insert(3, list)
list = Insert(10, list)
list = Insert(103, list)
list = Insert(56, list)
has := Has(103, list)
fmt.Println("103? ", has)
fmt.Println(": ", Length(list))
list = Remove(103, list)
first := list.First()
fmt.Println("1 :", first.Val)
fmt.Println(" : ", first.next.Val)
len := Length(list)
fmt.Println(": ", len)
}
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

역시 그려보면 이해가 빠릅니다.

노드(데이터)가 앞 뒤 노드를 찾조하고 있어 검색 등이 효율적입니다.

 

Comments