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로 구현한 코드입니다.

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)
}

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

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