[Java] LinkedList 구현하고 JUnit5로 테스트하기

5 분 소요

Java로 LinkedList를 구현해보자! :raising_hand:

:bulb: Requirements

  • LinkedList에 대해 공부하세요.
  • 정수를 저장하는 ListNode 클래스를 구현하세요.
  • ListNode add(ListNode head, ListNode nodeToAdd, int position)를 구현하세요.
  • ListNode remove(ListNode head, int positionToRemove)를 구현하세요.
  • boolean contains(ListNode head, ListNode nodeTocheck)를 구현하세요.

목차

1. 리스트 자료구조

  • 1.1. 배열로 만든 리스트
  • 1.2. 연결리스트

2. 연결리스트

  • 2.1. 노드
  • 2.2. 연결리스트의 종류
  • 2.3. 자바의 연결리스트 클래스

3. 코드

  • 3.1. 연결리스트 구현코드
  • 3.2. 테스트코드

1. 리스트 자료구조

List는 다수의 데이터를 저장하는 자료구조로, 집합과 달리 순서가 있게 데이터가 나열되어 있는 형태이다.

기본적으로 삽입(insert), 삭제(remove), 검색(search) 등의 연산이 있으며, 대표적으로 배열연결리스트 형태로 리스트를 구현한다.

배열연결리스트로 만든 리스트 자료구조


1.1. 배열로 만든 리스트


리스트를 표현하는 가장 대표적인 방법이다. 데이터를 연속된 공간에 저장하므로서 데이터들의 순서 관계를 유지할 수 있다.

  • 랜덤 엑세스 가능 : 내가 원하는 칸이 어디에 위치해 있던지, 간단한 연산을 통해 메모리 주소를 계산하여 접근할 수 있다.
    • 내가 원하는 메모리 주소 == 배열의 시작주소 + (내가 찾고자하는 칸의 인덱스 * 1칸의 크기)
  • 크기가 고정 : 기존의 배열칸 이상의 데이터를 저장할 경우 rellocation이 필요하다.
  • 삽입과 삭제연산 비용 큼: 리스트 중간에 원소를 삽입하거나 삭제할 경우, 다수의 데이터를 옮겨야 하기 때문에 비용이 크다.

1.2. 연결리스트


노드들이 링크로 연결된 형태의 자료구조이다. 데이터 영역에 데이터를 저장하고, 포인터(링크)영역에 다음/이전 노드로 가는 주소를 저장하여 순서를 나타낸다.

  • 삽입과 삭제연산 비용 적음: 다른 데이터의 이동없이 리스트 중간에 삽입 삭제할 수 있다.
  • 길이제한 없음: 노드를 연결한 형태이기 때문에 길이제한이 없다.
  • 메모리 효율 낮음 : 노드를 연결하기 위한 포인터 저장 영역이 필요하기 때문에, 배열에 비해 메모리 효율이 낮다.
  • 랜덤 엑세스 불가능 : 연속된 메모리 주소에 데이터가 저장된 것이 아니기 때문에, 배열처럼 랜덤 액세스가 불가능하다.


종합적으로 보면 요소의 삽입/삭제가 빈번한 경우 연결리스트를 사용하고, 그렇지 않은 경우라면 배열을 사용하는 것이 효과적이다.

2. 연결리스트

2.1. 노드

노드데이터와 다음 데이터로의 주소가 하나로 묶여있는 단위를 의미한다.

image

  • 각 노드는 하나의 데이터 필드와 하나의 링크(next) 필드로 구성된다.
  • 첫 번째 노드의 주소는 따로 저장(head)해야 한다.
  • 마지막 노드의 next 필드에는 NULL을 저장하여 연결리스트의 끝임을 표시한다.

2.2. 연결리스트의 종류

연결 리스트의 종류로는 단일 연결 리스트이중 연결 리스트 등이 있다.

단일연결리스트

image

단일 연결 리스트는 각 노드에 자료 공간과 한 개의 포인터 공간을 가진 형태를 말한다. 노드의 포인터는 다음 노드를 가리킨다.

이중연결리스트

image

이중 연결 리스트는 단일 연결 리스트와 비슷하지만, 포인터 공간이 2개이고 각 포인터는 앞의 노드뒤의 노드를 가리킨다.

이중연결리스트는 앞 노드와 뒷 노드 모두 순회할 수 있기 때문에, 추가삭제 연산에 있어서 더 효율적인 성능을 낼 수 있다. 하지만, 2개의 포인터를 사용하기 때문에 추가적인 공간이 필요하다.

2.3. 자바의 연결리스트 클래스

image

java.util.LinkedList는 Serializable, Cloneable, Iterable, Collections, Deque, List, Queue와 같은 다양한 인터페이스를 구현하고 있으며, 특히 ListDeque 인터페이스의 이중연결리스트 구현체이다.

특징은 다음과 같다.

  • 모든 메서드는 이중연결리스트의 성능을 가지고 있다.
  • 비동기적으로 구현되기 때문에, 여러 스레드가 동시에 LinkedList의 노드를 추가/삭제할 수 없다.
    • 즉, 외부적으로만 동기화 될 수 있다.

3. 코드

단일연결리스트로 구현하였다.

ListNode

package datastructure.linkedlist;

public class ListNode {
    private int data;
    public ListNode next;

    public ListNode(int data) {
        this.data = data;
        this.next = null;
    }

    public int getData() {
        return data;
    }
}


구현코드

package datastructure.linkedlist;

import java.util.ArrayList;
import java.util.List;

public class LinkedListImpl implements LinkedList{
    public LinkedListImpl() {}

    public LinkedListImpl(int data) {
        add(null, new ListNode(data), 0);
    }

    public LinkedListImpl(ListNode nodeToAdd) {
        add(null, nodeToAdd, 0);
    }

    @Override
    public ListNode add(ListNode head, ListNode nodeToAdd, int position) {// position == index
        // position이 허용범위를 벗어난 경우
        if (!isAvailablePosition(head, position)) throw new IndexOutOfBoundsException();
        
        // 아직 연결된 노드가 없는 경우
        if (head == null) return nodeToAdd;

        // nodeToAdd를 head에 위치시키려는 경우
        if (position == 0) {
            nodeToAdd.next = head;
            return nodeToAdd;
        }

        ListNode prevNode = getNodeAtThePosition(head, position - 1);// 삽입하려는 위치의 바로 이전 노드(position - 1)를 가져온다.
        nodeToAdd.next = prevNode.next;                                     // 삽입하려는 노드(nodeToAdd)에 기존의 linkedlist에서 position에 위치했던 node를 연결시킨다.
        prevNode.next = nodeToAdd;                                          // 삽입하려는 위치의 바로 이전 노드에 삽입하려는 노드를 연결시킨다.
        return head;
    }

    @Override
    public ListNode remove(ListNode head, int positionToRemove) {// positionToRemove == index
        // positionToRemove가 가능범위를 벗어난 경우
        if (!isAvailablePosition(head, positionToRemove) ||
                positionToRemove == getSize(head)) throw new IndexOutOfBoundsException();
        
        // 연결리스트가 null인 경우
        if (head == null) return null;

        // head를 제거하는 경우
        if (positionToRemove == 0) return head.next;

        ListNode prevNode = getNodeAtThePosition(head, positionToRemove - 1);// 제거하려는 위치의 바로 이전 노드를 가져온다.
        prevNode.next = prevNode.next.next;                                         // prevNode에 해당 노드의 다다음 노드를 연결시킨다.
        return head;
    }

    @Override
    public boolean contains(ListNode head, ListNode nodeToCheck) {
        while (head != null) {
            if (head.getData() == nodeToCheck.getData()) {//순회하며 가져온 노드의 값과 nodeToCheck의 값이 일치하는지 확인한다.
                return true;
            }
            head = head.next;//연결리스트를 head부터 tail까지 순회
        }
        return false;
    }

    // position이 허용가능한 인덱스 값인지 확인
    public boolean isAvailablePosition(ListNode head, int position) {
        if (position < 0 || position > (getSize(head))) { // available value : 0 <= position <= linkedlist 크기
            return false;
        }
        return true;
    }

    // 연결리스트의 head를 기준으로 순회하여, position 위치에 있는 노드를 반환
    public ListNode getNodeAtThePosition(ListNode head, int position) {
        for (int i = 0; i < position; i++) {
            head = head.next;
        }
        return head;
    }

    // 현재 노드가 다음 노드와 연결되어 있는지 확인
    public boolean hasNext(ListNode head) {
        if (head.next == null) return false;
        return true;
    }

    // 연결리스트의 크기(노드 개수)를 반환
    public int getSize(ListNode head) {
        if (head == null) return 0;
        if (!hasNext(head)) return 1;
        ListNode node = head.next;
        int size = 2;
        while (hasNext(node)) {
            size++;
            node = node.next;
        }
        return size;
    }

    public String toString(ListNode head) {
        if (head == null) return "";

        List<String> nodes = new ArrayList<>();
        ListNode tmp = head;

        while (head != null) {
            nodes.add(String.valueOf(head.getData()));
            head = head.next;
        }
        head = tmp;
        return String.join(",", nodes);
    }
}



테스트코드

package datastructure.linkedlist;

import org.junit.jupiter.api.*;

import static org.junit.jupiter.api.Assertions.assertAll;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertThrows;

@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@DisplayName("연결리스트 테스트")
public class LinkedListImplTest {
    private LinkedListImpl list;
    private ListNode head;

    @BeforeAll
    void init() {
        list = new LinkedListImpl();
        head = list.add(null, new ListNode(1), 0);
        head = list.add(head, new ListNode(3), 1);
        head = list.add(head, new ListNode(5), 2);
        head = list.add(head, new ListNode(7), 3);
    }

    @Test
    @DisplayName("노드 추가 테스트")
    void addTest() {
        assertAll("노드 추가 오류",
                () -> {// init()에서 생성한 linkedlist를 확인
                    assertEquals("1,3,5,7", list.toString(head));
                },
                () -> {// 허용범위를 벗어난 position에 add하려는 경우
                    Exception exception = assertThrows(IndexOutOfBoundsException.class, () ->
                            list.add(head, new ListNode(9), list.getSize(head) + 1));
                }
        );
    }

    @Test
    @DisplayName("노드 제거 테스트")
    void removeTest() {
        assertAll("노드 제거 오류",
                () -> {// 연결리스트(1,3,5,7)에서 위치가 3(idx 기준)인 node 삭제
                    assertEquals("1,3,5", list.toString(list.remove(head, 3)));
                },
                () -> {// 허용범위를 벗어난 position의 노드를 remove하려는 경우
                    Exception exception = assertThrows(IndexOutOfBoundsException.class, () ->
                            list.remove(head, list.getSize(head)));
                });
    }

    @Test
    @DisplayName("노드 포함 테스트")
    void containsTest() {
        assertAll("노드 포함 오류",
                () -> {// 연결리스트(1,3,5,7)가 1을 포함하는지 확인
                    assertTrue(list.contains(head, new ListNode(1)));
                },
                () -> {// 연결리스트(1,3,5,7)가 11을 포함하지 않는지 확인
                    assertFalse(list.contains(head, new ListNode(11)));
                });
    }
}





:orange_book: References

댓글남기기