C++을 이용하여 LinkedList를 사용하여 Stack을 구현해보도록 하겠습니다.
스택(Stack)을 설명하자면 다음과 같습니다. 가장 처음 데이터를 1을 넣으면 [1]이 됩니다. 거기에 2를 넣으면 [2, 1] 이 됩니다. 2->1을 가르키고 있는 것이죠. 그리고 4를 넣으면 [4,2,1]이 됩니다. 4->2->1 입니다. 여기에 10을 넣으면 [10, 4, 2, 1] 데이터가 됩니다. 이제 데이터를 꺼내면 10부터 나오게 됩니다. 마지막 넣은 데이터가 가장 처음 나오게 됩니다.
Last In First out 으로 줄여서 LIFO 라고 합니다.
10을 꺼내게 되면 [4, 2, 1]이 됩니다. 다시 데이터를 꺼내면 4가 나오게 됩니다. [2,1]이 됩니다. 그리고 하나더 꺼내면 2가 나오고 마지막으로 1이 나오게 됩니다.
비유를 하자면 바닥에 이불을 하나씩 쌓는것을 생각하면 됩니다. 빨간이불을 먼저 쌓고 그다음에 검정이불을 쌓고 마지막으로 하얀이불을 쌓으면 다음에 꺼낼 때 하얀이불이 가장위에 있기때문에 하얀이불부터 꺼낼 수 있습니다.
Stack은이렇듯 LIFO 으로 마지막에 들어온 데이터가 가장 처음에 나오는 방식 입니다.
연결리스트를 구현할때 Node들을 만들어서 서로 연결해야 합니다. Node는 기본적으로 데이터를 보관할 변수하나와 다른 노드를 가르킬 포인터 변수를 가집니다. 아래 예제에서는 data변수가 데이터를 저장할 변수가 됩니다. 그리고 next가 다른 노드를 가르킬 포인터가 됩니다.
그리고 추가로 생성자를 추가시켰습니다. 구조체에도 c++에서는 생성자나 함수등을 포함시킬 수 있습니다. class와의 차이는 모든 변수와 함수가 기본적으로 public입니다. 첫번째 생성자는 기본생성자로 초기화만 시킵니다. 다음생성자는 데이터와 다음 노드를 받습니다. 파라미터로 들어온 ptr 다음 노드로 자신을 가리키게 합니다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | struct Node { int data; Node* next; Node() { next = NULL; data = 0; } Node(int i, Node* ptr) //ptr 뒤에 추가 { data = i; next = ptr->next; ptr->next = this; } }; | cs |
연결 리스트(LinkedList)를 가지고 구현한 스택(stack) 코드입니다. 기본적으로 .cpp로 C++ 컴파일을 해야한다.
Stack에 생성자를 보면 new Node로 더미노드를 만들어서 head에 넣는 것을 볼 수 있습니다. 이는 pop과 push등을 할때 더미노드가 있어야 코드가 더 간결해집니다.
소멸자도 반드기 구현해줘야 합니다. 그래야 사용하지 않는 메모리를 heap에서 계속 점유하고 있기 때문에 프로그램이 계속 돌다가 메모리가 부족한 현상이 발생 할 수도 있습니다.
push 함수는 head 다음에 새로운 노드를 추가시킵니다. pop은 head 다음 노드를 꺼내는 역할을 합니다.
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 | #include<stdio.h> struct Node { int data; Node* next; Node() { next = NULL; data = 0; } Node(int i, Node* ptr) //ptr 뒤에 추가 { data = i; next = ptr->next; ptr->next = this; } }; struct Stack { Node *head; int count; Stack() { //생성자 head = new Node(); //더미노드 count = 0; } ~Stack() { //소멸자 while (count!=0) { pop(); } delete head; } void pop() { //데이터 제거 Node *tmp = head; head = head->next; delete tmp; count--; } void push(int i) { //데이터 삽입 new Node(i, head); count++; } void printAll() { //모두 출력 Node *cur = head; while (cur->next) { cur = cur->next; printf("%d\n", cur->data); } } }; int main() { Stack *stack = new Stack(); stack->push(1); stack->push(3); stack->push(5); stack->push(7); stack->printAll(); delete stack; } | cs |
단일 연결리스트 vs 이중 연결 리스트
단일 연결 리스트는 Node가 다른 Node를 하나만 가리키고 있지만 이중 연결리스트는 앞 뒤로 하나씩 연결을 가지고 있습니다.
그런데 이중연결리스트는 탐색을 할때 해당 노드에서 앞에 노드나 뒤에 노드로 쉽게 왔다갔다 할 수 있는 장점이 있습니다. 특정 노드를 탐색하다가 뒤로 마음대로 갈 수 있다는 의미이다. 프로그램을 작성하기전에 이러한 상황이 꼭 필요하고 자주 발생할 것 같다면 이중 연결 리스트를 고려해 보아야 합니다.
이중 연결리스트 : http://hijuworld.tistory.com/55
'알고리즘' 카테고리의 다른 글
C++ 이중 연결 리스트(double linked list) 설명 및 예제 (0) | 2018.06.07 |
---|---|
구간 합, 구간 최대값 등 구하기 (세크먼트 트리 Segment Tree 이용) (0) | 2018.06.02 |
C/C++ 2진 탐색(Binary Search) 예제 코드 (0) | 2018.05.24 |
C/C++ 병합정렬(Merge Sort) 코드 (0) | 2018.05.24 |
분할 정복을 이용 1부터 n까지 합 구하는 함수, 알고리즘 (0) | 2018.04.02 |