[자료구조] 해시테이블이란??

Ko Seoyoung·2021년 5월 5일
3
post-custom-banner

1️⃣ 해시테이블이란?

  • (Key, Value)로 데이터를 저장하는 자료구조 중 하나
  • 빠르게 데이터를 검색할 수 있다

해시 테이블이 빠른 검색속도를 제공하는 이유? 🤔

  • 내부적으로 배열(버킷)을 사용하여 데이터를 저장하기 때문

  • 버킷(bucket)이란?

    해시 테이블은 각각의 Key값에 해시함수를 적용해 배열의 고유한 index를 생성하고, 이 index를 활용해 값을 저장하거나 검색하게 된다. 여기서 실제 값이 저장되는 장소를 버킷 또는 슬롯이라고 한다.

예시:

(Key, Value)가 ("John Smith", "521-1234")인 데이터를 크기가 16인 해시 테이블에 저장하는과정?

(1) index = hash_function("John Smith") % 16 연산을 통해 index 값을 계산한다.

(2) array[index] = "521-1234" 로 전화번호를 저장하게 된다.

이러한 구조로 데이터를 저장하면 Key값으로 데이터를 찾을 때 해시 함수를 1번만 수행하면 되므로 매우 빠르게 데이터를 저장/삭제/조회할 수 있다. 해시테이블의 평균 시간복잡도는 O(1)이다.

해시 함수

  • 해시 함수에서 중요한 것은 고유한 인덱스 값을 설정하는 것이다.
  • 해시 테이블에 사용되는 대표적인 해시 함수 4가지:
    1. Division Method: 나눗셈을 이용하는 방법으로 입력값을 테이블의 크기로 나누어 계산한다.( 주소 = 입력값 % 테이블의 크기) 테이블의 크기를 소수로 정하고 2의 제곱수와 먼 값을 사용해야 효과가 좋다고 알려져 있다.
    2. Digit Folding: 각 Key의 문자열을 ASCII 코드로 바꾸고 값을 합한 데이터를 테이블 내의 주소로 사용하는 방법이다.
    3. Multiplication Method: 숫자로 된 Key값 K와 0과 1사이의 실수 A, 보통 2의 제곱수인 m을 사용하여 다음과 같은 계산을 해준다. h(k)=(kAmod1) × m
    4. Univeral Hashing: 다수의 해시함수를 만들어 집합 H에 넣어두고, 무작위로 해시함수를 선택해 해시값을 만드는 기법이다.

2️⃣ 해시(Hash)값이 충돌하는 경우

그런데 만약 "John Smith"를 해시 함수를 돌려 나온 값과 "Sandra Dee"를 해시 함수를 돌려 나온 값이 동일하다면 어떻게 해야 할까? (중복된 인덱스 값이 생기는 경우)

해시 값 충돌 문제 해결 방법 2가지

  1. 분리 연결법(Separate Chaining)
  2. 개방 주소법(Open Addressing)

분리 연결법(Separate Chaining)

  • Separate Chaining은 동일한 버킷의 데이터에 대해 자료구조를 활용해 추가 메모리를 사용하여 다음 데이터의 주소를 저장한다. 즉, 동일한 버킷으로 접근하는 데이터들을 연결하여 관리한다.
  • 실제로 Java8의 Hash테이블은 Self-Balancing Binary Search Tree 자료구조를 사용해 Chaining 방식을 구현하였다.
  • 장점: 해시 테이블의 확장이 필요없고 간단하게 구현이 가능하고, 손쉽게 삭제할 수 있다.
  • 단점: 데이터의 수가 많아지면 동일한 버킷에 chaining되는 데이터가 많아지며 그에 따라 캐시의 효율성이 감소한다.

개방 주소법(Open Addressing)

  • Open Addressing이란 추가적인 메모리를 사용하는 Chaining 방식과 다르게 비어있는 해시 테이블의 공간을 활용하는 방법이다.
  • Open Addressing을 구현하기 위한 대표적인 방법 3가지
    1. Linear Probing: 현재의 버킷 index로부터 고정폭 만큼씩 이동하여 차례대로 검색해 비어 있는 버킷에 데이터를 저장한다.
    2. Quadratic Probing: 해시의 저장순서 폭을 제곱으로 저장하는 방식이다. 예를 들어 처음 충돌이 발생한 경우에는 1만큼 이동하고 그 다음 계속 충돌이 발생하면 2^2, 3^2 칸씩 옮기는 방식이다.
    3. Double Hashing Probing: 해시된 값을 한번 더 해싱하여 해시의 규칙성을 없애버리는 방식이다. 해시된 값을 한번 더 해싱하여 새로운 주소를 할당하기 때문에 다른 방법들보다 많은 연산을 하게 된다.

  • Open Addressing에서 데이터 삭제 시, 삭제된 공간은 Dummy Space로 활용된다. 따라서 Hash Table을 재정리 해주는 작업이 필요하다.

3️⃣ 해시테이블(HashTable) 시간복잡도

  • 데이터 조회

    평균: O(1) - 각각의 Key값은 해시함수에 의해 고유한 index를 가지므로 데이터에 바로 접근

    최악: O(N) - 데이터의 충돌이 발생한 경우 Chaining에 연결된 리스트들까지 검색을 해야 하므로

  • 충돌을 방지하는 방법들은 데이터의 규칙성(클러스터링)을 방지하기 위한 방식이지만 공간을 많이 사용한다는 치명적인 단점이 있다. 만약 테이블이 꽉 차있는 경우라면 테이블을 확장해주어야 하는데, 이는 매우 심각한 성능의 저하를 불러오기 때문에 가급적이면 확정을 하지 않도록 테이블을 설계해주어야 한다.
    (통계적으로 해시 테이블의 공간 사용률이 70% ~ 80%정도가 되면 해시의 충돌이 빈번하게 발생하여 성능이 저하되기 시작한다고 한다.)

  • 해시 테이블에서 자주 사용하게 되는 데이터를 Cache에 적용하면, 자주 hit하게 되는 데이터를 캐시에서 바로 찾을 수 있으므로 해시 테이블의 성능을 향상시킬 수 있다.


4️⃣ Java의 HashMap(해시맵) vs HashTable(해시테이블)

Java에서 HashTable과 HashMap의 차이는 동기화 지원 여부에 있다.

HashTable은 병렬 프로그래밍을 할 때 동기화를 지원해준다는 것을 의미한다. 따라서 해당 함수를 처리하는 시간이 조금 지연됨을 의미힌다.

  • 병렬 처리를 하면서 자원의 동기화를 고려해야 하는 상황 → 해시테이블(HashTable)
  • 병렬 처리를 하지 않거나 자원의 동기화를 고려하지 않는 상황 → 해시맵(HashMap)

참고자료

https://mangkyu.tistory.com/102

profile
Web Frontend Developer 👩🏻‍💻 #React #Nextjs #ApolloClient
post-custom-banner

0개의 댓글