Skip to content

SpiralMomentum/Algorithms

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

92 Commits
 
 
 
 

Repository files navigation

알고리즘 공부

교재 솔루션 : https://sites.math.rutgers.edu/~ajl213/CLRS/CLRS.html

알고리즘 개념

  1. 알고리즘이란?

주어진 입력을 출력으로 이끄는 잘 정의된 계산 절차
계산 문제를 정의하기 위해 입력과 출력의 관계를 잘 정의해야 하는데, 이러한 관계를 구현할 수 있는 계산과정

  1. 타당한 알고리즘

알고리즘이 모든 입력 사례(Instance)에 대해 항상 올바른 출력을 내고, 종료할 경우를 타당하다고 한다
일부 입력 사례에 종료하지 않거나, 잘못된 답을 도출할 경우 타당하지 않다 일컺는다

  1. 어떤 문제를 알고리즘으로 풀어야할까?

문제들은 공통적으로 "후보는 많지만, 대부분이 문제의 답이 아니다. 최상의 답을 찾는 것을 도전한다", "실용적인 사례를 주변에서 찾을 수 있다"와 같은 특징을 갖는다.

  1. 알고리즘은 왜 필요할까?

컴퓨터는 유한한 성능을 갖는다. 이로인해 공간적, 시간적 제약이 생긴다.
이러한 제약사항을 고려하여, 자원을 보다 효율적으로 사용해야 한다. = 효율적인 알고리즘이 필요하다.

  1. 공부방법

입력과 출력 그리고 사례(인스턴스)를 확실하게 이해한다
사례를 통해 아이디어(기본적인 이해)를 정리한다
아이디어를 유사코드로 만들고, 성능을 분석-파악한다
시간, 공간 복잡도를 이용하여 알고리즘을 효율적으로 최적화한다.


삽입정렬

  • 입력 : n개 수들의 수열 (A Sequence of n numbers <a1, a2, ... , an>)
  • 출력 : a1 ≤ a2 ≤ a3 ≤ ... ≤ an을 만족하는 입력 수열의 순열(재배치) (A Permutation <b1, b2, ... , bn> of the input sequence such that b1 ≤ b2 ≤ ... ≤ bn)
  • 유사코드
for j = 2 to A.length
    key = A[j]
    i = j-1
    while i>0 and A[i] > key
        A[i+1] = A[i]
        i = i-1
    A[i+1] = key
  • 성능 분석하기

 for j = 2 to A.length                  c₁            bn
    key = A[j]                          c₂            n-1
    i = j-1                             c₃            n-1
    while i>0 and A[i] > key            c₄            Σtᴊ (j=2 ~ n)
        A[i+1] = A[i]                   c₅            Σ(tᴊ-1) (j=2 ~ n)
        i = i-1                         c₆            Σ(tᴊ-1) (j=2 ~ n)
    A[i+1] = key                        c₇            n-1
    
    *) tᴊ는 j가 2, 3, .., n 일때, while 루프의 검사가 실행되는 횟수를 의미
> 수행시간은 각 명령문 수행시간의 합  
> T(n) = c₁n + c₂(n-1) + c₃(n-1) + c₄Σtᴊ (j=2 ~ n) + c₅Σ(tᴊ-1) (j=2 ~ n) + c₆Σ(tᴊ-1) (j=2 ~ n) + c₇(n-1)  
> (최악의 경우) T(n) = c₁n + c₂(n-1) + c₃(n-1) + c₄(n(n+1)/2-1) + c₅(n(n-1)/2)+ c₆(n(n-1)/2) + c₇(n-1)  
> (최악의 경우) T(n) = an² + bn + c

합병정렬

  • 입력 : n개 수들의 수열 (A Sequence of n numbers <a1, a2, ... , an>)
  • 출력 : a1 ≤ a2 ≤ a3 ≤ ... ≤ an을 만족하는 입력 수열의 순열(재배치) (A Permutation <b1, b2, ... , bn> of the input sequence such that b1 ≤ b2 ≤ ... ≤ bn)
  • 분할(정렬할 n개 원소의 배열을 n/2개씩 부분 수열 두개로 분할),
    정복(합병정렬을 통해 두 부분배열을 재귀적으로 정렬),
    결합(정렬된 두 부분배열을 병합해 정렬된 배열 하나로 만든다) 세가지 포인트가 존재.
  • 유사코드 (보조 프로시저인 "결합" 과정)
Merge(A,p,q,r)  : A는 배열, p,q,r은 인덱스로 p <= q < r을 만족한다.                  cost           times
    n₁ = q-p+1                                                                      c₁             1
    n₂ = r-q                                                                        c₂             1
    배열 L[1 .. n₁+1]과 R[1 .. n₂+1]을 생성한다                                      c₃             1
    
    for i = 1 to n₁                                                                 c₄             n₁
        L[i] = A[p+i-1]                                                             c₅             1
    
    for j = 1 to n₂                                                                 c₆             n₂
        R[j] = A[q+j]                                                               c₇             1
    
    L[n₁+1] = ∞                                                                     c₈             1
    R[n₂+1] = ∞                                                                     c₉             1
    i = 1                                                                           c₁₀            1
    j = 1                                                   
    
    for k = p to r                                                                  c₁₁            n
        if L[i] <= R[j]                                                             c₁₂            c
            A[k] = L[i]                                                             c₁₃            1
            i = i+1                                                                 c₁₄            1
        else A[k] = R[j]                                                            c₁₅            1
            j = j+1                                                                 c₁₆            1
  > T(n) = c₄n₁ + c₆n₂ + c₁₁n + C
  > T(n) = an + b = Θ(n)
  • 유사코드 (합병정렬 전체 과정)
   Merge-Sort(A,p,r)
      if p < r
          q = ⌊(p+r)/2⌋
          Merge-Sort(A,p,q)
          Merge-Sort(A,q+1,r)
          Merge(A,p,q,r)
  # 분할정복 분석
  > 크기 n이 충분히 작아 n <= c 조건을 만족할 경우, T(n) = Θ(1)
  > 다른 모든 경우엔, T(n) = aT(n/b) + D(n) + C(n)
  *) 지금까지 a=b인 상황을 봤지만, 같지 않은 상황에서도 적용이 가능한 알고리즘도 분석할 예정  
  *) D(n)은 문제를 분할하는데 걸리는 시간, C(n) 부분 문제들의 해를 결합하여 원래 문제의 해를 만드는데 걸리는 시간  
  
  # 합병정렬 분석
  > 분할 : 부분 배열의 중간 위치를 계산하는 과정, 상수 시간이 걸린다. (D(n) = Θ(1))
  > 정복 : 두 개의 부분 문제를 재귀적으로 풀면서, 각 부분 문제는 크기가 n/2로 줄어든다. (2T(n/2))
  > 결합 : 앞서 보았듯, Merge 프로시저는 Θ(n) 시간이 걸린다. (C(n) = Θ(n))
  따라서, T(n) = 2T(n/2) + Θ(n) (n>1) / T(n) = Θ(1) (n=1)

알고리즘의 효율성

  1. 점근적 분석

알고리즘의 효율성을 판단할때, 주로 수행시간이 얼마나 걸리는지를 중심으로 판단한다.
이때, 각 알고리즘별 정확한 수행시간을 중심으로 판단하는 것이 아닌, 입력의 크기가 극한으로 증가할 때 수행시간을 중심으로 판단한다. 이는 입력의 크기가 아주 작은 경우를 제외하고, 대부분의 경우에 좋은 판단기법이 되며 이러한 방식을 점근적 분석이라 한다.
이때, 입력의 크기가 극한으로 커가므로, 수행시간의 증가 차수 중 상수계수나 저차항은 무시된다.

  1. 점근적 표기

위에서 살펴본 점근적 분석을 형식적으로 표기하는 방법을 의미한다.


점근적 표기

  • 알고리즘의 점근적 수행시간을 나타내는데 사용하는 표기.

  • 정의역의 경우 자연수의 집합 N ={0,1,2 ...}로 정의된다.

    1. O 표기 (빅오 표기)

    명칭 : 주어진 함수 g(n)에 대한 빅오g(n) 또는 오g(n)이라 명명한다. (O(g(n)))
    정의 : O(g(n)) = { f(n) : 모든 n > n₀에 대해 0 <= f(n) <= cg(n)인 양의 상수 c, n₀이 존재한다. }
    : f(n) = O(g(n)): g(n)는 f(n)에 대한 점근 상한을 의미한다.
    함수의 상한을 나타내기 위해 사용되는 개념이다.
    O는 알고리즘의 최악의 실행 시간을 설명하는데 좋다.
    +) 추가이해
    https://ko.wikipedia.org/wiki/%EC%A0%90%EA%B7%BC_%ED%91%9C%EA%B8%B0%EB%B2%95

    1. Ω표기 (오메가 표기)

    명칭 : 주어진 함수 g(n)에 대한 빅 오메가 g(n) 또는 오메가 g(n)이라 명명한다. (Ω(g(n))
    정의 : Ω(g(n)) = { f(n) : 모든 n > n₀에 대해 0 <= cg(n) <= f(n) 인 양의 상수 c, n₀이 존재한다.}
    : f(n) = O(g(n)): g(n)는 f(n)에 대한 점근 하한을 의미한다.
    함수의 하한을 나타내기 위해 사용되는 개념이다.
    Ω은 알고리즘의 최상의 사례 실행 시간을 설명하는데 좋다.

    1. Θ표기 (세타 표기)

    명칭 : 주어진 함수 g(n)에 대해 세타g(n)이라 명명한다. (Θ(g(n))
    정의 : Θ(g(n)) = = { f(n) : 모든 n > n₀에 대해 0 <= c₁g(n) <= f(n) <= c₂g(n)인 양의 상수 c₁, c₂, n₀이 존재한다.}
    :


T(n) 증명하기

  • 수행시간에 대한 점화식은 다양한 형태로 표기된다. 이렇게 "추측"된 점화식을 어떻게 증명할 수 있을까?
  • "치환법(substitution method)", "재귀 트리 방법(recursion-tree method", "마스터 방법(master method)"등의 방법을 통해 증명한다.
  • 보통 등식의 꼴을 띄지만, 간혹 부등식 형태의 점화식도 사용된다.(부등식의 경우 빅오 또는 오메가 표기를 통해 사용한다.)
  • 치환법 -수학적 귀납법을 사용한다.

    1. 초항에 대한 증명을 보여준다.
    2. n = k-1까지의 모든 경우에서 해당 점화식이 참임을 가정한다.
    3. n = k일때 점화식이 참임을 증명한다.
      ex)
     T(n) = O(n*log n)을 증명하라
     1) 초항을 증명하라 / 정의역 n은 1보다 큰 자연수다. & T(1) = 1 & T(n) = 2T(n/2) + n
       n=2 > T(2)        <= 2*c*log₂2
             = 2T(1)+1   <= 2c
             = 4         <= 2c
             ∴ c >= 2이면 항상 참
     2) n = k-1까지 참이라고 가정
     3) n = k일때 참임을 증명하기
       T(k) = 2T(k/2) + k    > T(k/2)는 k-1까지 범위에 들어감. 즉 증명되어 있음.
                               ∴ T(k/2) <= c*(k/2)*log(k/2)
       2T(k/2) + k <= 2*(c*(k/2)*log(k/2)) + k
       T(k) <= c*k*logk - ck + k
       T(k) <= c*k*logk + (1-c)k
         > 1-c 는 항상 0보다 작은 수가 나온다. 
       ∴ T(k) <= c*k*logk + (1-c)k <= c*k*logk
          즉, T(k) <= c*k*logk (O(nlogn))을 만족한다.
    
  • 재귀 트리 방법

    1. 개념 이해하기




    1. 사례로 이해하기
      ex)
        case 1 : 줄어드는 크기가 동일
        T(n) = 2T(n/2)+n
        계층을 i로 두었을 때, 각 층의 노드 갯수는 2^i이다.
        또한 계층이 i일때, T(n)에서 n의 식은 n/2^i로 표현할 수 있다. 
        이때 가장 하층의 T(1)이 위치하게 되는데. 이는 즉 마지막 층 i 일때, n의 식이 1이 나온다를 의미한다. 이를 식으로 정리하면,
        n/2^i = 1 
        > n = 2^i
        > i = log₂n
          ∴ 마지막 층의 노드 갯수는 2^i 즉 2^log₂n = n이다.
          ∴ 각 계층별로의 합들은 해당 계층의 노드들을 다 더한 것과 같고, 전체 시간 T(n)은 모든 노드를 더한 것과 같다.
            = n + n + ... + n (개수는 높이인 i 즉 log₂n)
            = n * log₂n
            <= c * (n * log₂n)
            = O(n * log₂n)을 만족한다. 
      
        case 2 : 줄어드는 크기가 상이
        T(n) = T(n/3) + T(2n/3) + cn
                
                    cn                               cn                                                                 cn
              T(n/3)  T(2n/3)    >        c(n/3)            c(2n/3)              >                            c(n/3)             c(2n/3)
                                     T(n/9)  T(2n/9)    T(2n/9)  T(4n/9)                                c(n/9)  c(2n/9)     c(2n/9)  c(4n/9)
                                                                                                                .                   .
                                                                                                                .                   .
                                                                                                 T(1) ---------------------------------------
                                                                                                      T(1) ------------------------------------
                                                                                                          T(1) ---------------------------------
                                                                                                            .....................................
                                                                                                                                                T(1)
        위와 같이 방향에 따라 T(1)이 등장하는 높이가 상이하다. 
        (가장 크게 작아지는 경우가 가장 빠르게 T(1)에 도달, 반대로 작게 작아지는 경우 가장 느리게 T(1)에 도달)
        1) 이때 가장 빠르게 T(1)이 도달하는 방향의 높이를 k₁, 가장 느리게 T(1)이 도달하는 방향의 높이를 k₂이라고 했을때,
        트리에서 T(1) 등장하는 높이들 중에서 k₁의 높이가 가장 작고, k₂의 높이가 가장 크다.
        
        2) k₁과 k₂을 구하면, 
          n/3^k₁ = 1                 n*(2/3)^k₂ = 1
          3^k₁ = n                   n = (3/2)^k₂
          k₁ = log₃n                 k₂ = log(3/2)n
        
        3) ∴ k1 <= level <= k2
              = log₃n <= level <= log(3/2)n
        
        4) 각 레벨에서 노드의 합은 n,  T(n)은 모든 노트의 합임
           ∴ n*log₃n <= T(n) <= n*log(3/2)n
              = n*(log₂n/log₂3) <= T(n) <= n*(log₂n/log₂(3/2))
              = n*log₂n*c1 <= T(n) <= n*log₂n*c2
        
        5) ∴ T(n) <= c*n*log₂n 
              >T(n) = O(n*log₂n) 
      
      
      * 증명하기 / T(n) = T(n/3) + T(2n/3) + n
      1) base case 
          T(3) = T(1) + T(2) + 3 <= d * 3 * log₂3
          T(1), T(2)는 더 이상 분기할 수 없음(상수 시간 소모)
          ∴ 5 <= d * 3 * log₂3
             5 - 3*log₂3 <= d를 만족하는 d가 1개라도 존재... 증명됨
             
      2) n < k 일때 참이라고 가정
      3) n = k일 때 증명하기
          T(k) = T(k/3) + T(2k/3) + k    
          T(k/3), T(2k/3) 은 모두 n < k인 case. 따라서 전제에 의해 <= d*n*log₂n를 만족
          ∴ T(k/3) + T(2k/3) + k <= d*(1/3)*k*(log₂k - log₂3) + d*(2/3)*k*((1+log₂k) - log₂3) + k
                                  <= d*k*log₂k + k - d*k*(log₂3 - (2/3)) ... <= d*k*log₂k
          ∴ T(n) <= d*n*log₂n
          
          
      

### HEAP
  1. 힙이란?
    가. 완전이진트리로 볼 수 있는 배열 객체 나. 트리의 각 노드는 배열에서 한 원소와 대응된다. 다. 가장 낮은 레벨을 빼고는 완전히 차 있고, 가장 낮은 레벨은 왼쪽부터 채운다. 라. 트리의 루트는 [1]이며, 인덱스 i가 주어졌을 경우 부모의 인덱스는 ⌊i/2⌋ 왼쪽 자식의 인덱스는 2i, 오른쪽 자식의 인덱스는 2i+1이다 마. 이진 힙에는 최대 힙과 최소 힙 두가지 종류가 존재한다. 종류에 따라 힙 특성이 다르고, 힙의 모든 노드들은 힙 특성을 만족한다. 이때 특성은 다음과 같다. "임의의 노드 값은 그 부모의 값보다 클(작을) 수 없다." 바. 힙 트리에서 특정 노드의 높이는, 해당 노드에서 리프노드에 이르는 하향 경로 중 가장 긴 것의 간선 수로 정의 (힙의 높이는 루트 노드의 높이)

  힙의 노드 : O(lg n)
  높이가 k일때, 
  힙이 가질 수 있는 최소 노드 갯수는 (1 + 2 + 2^2 + 2^3 + ...  + 2^k-1 + 1) = 2^k
  힙이 가질 수 있는 최대 노드 갯수는 (1 + 2 + 2^2 + 2^3 + ... + 2^k) = 2^k+1 -1
  즉, 힙의 노드 개수 n은 다음과 같은 범위에 들어온다.
  2^k <= n <= 2^k+1 -1 < 2^k+1
  ∴ k <= lg n < k+1
     k = ⌊lg n⌋  즉, 노드 개수가 n개인 힙의 높이는 ⌊lg n⌋이다.
  1. heap-size와 length
    가. 배열의 인자들 나. length는 배열이 가지는 원소의 개수를, heap-size는 배열의 원소 중 힙에 속하는 원소의 개수를 의미한다. 다. 즉, 1 ~ length까지의 값들이 모두 유용할 순 있지만, 힙에 속하는 원소는 1 ~ heap-size 까지이다.

  2. MAX-HEAPIFY, BUILD-MAX-HEAP, HEAPSORT 가. MAX-HEAPIFY 아이디어

  MAX-HEAPIFY(A,i)
    l = LEFT(i)
    r = RIGHT(i)
    if l <= A.heap-size and A[l] > A[i]
      largest = l
    else largest = i
    if r<= A.heap-size and A[r] > A[i]
      largest = r
    if largest != i
      exchange A[i] with A[largest]
      MAX-HEAPIFY(A, largest)

나. MAX-HEAPIFY Worst Case (1) 왼쪽부터 노드가 차기 때문에, 항상 왼쪽의 노드 수가 오른쪽보다 많거나 같다. (2) 따라서 최악의 경우는, 왼쪽 노드를 타고 탐색하는 것이 최악의 경우이다. (3) 이떄, T(n) <= T(2n/3) + Ω(1)

다. BUILD-MAX-HEAP 아이디어

  BUILD-MAX-HEAP(A)
    A.heap-size = A.length
    for i = ⌊A.length/2⌋ downto 1
      MAX-HEAPIFY(A,i)  

라. BUILD-MAX-HEAP


### 이진 탐색 트리
  • 기본 설명 (1) 트리 하나의 부모의 다수의 자식들이 올 수 있다

    (2) 이진 트리 각 노드마다 0~2의 자식을 갖는 트리

    (3) 검색 트리를 이루는 요소 [1] 검색, 최소, 최대, 직전, 직후, 삽입, 삭제 등 [2] 딕셔너리와 우선순위 큐를 이용 [3] 기본 연산은 트리의 높이에 비례한다

    (4) 트리별 높이 시간 복잡도 [1] 완전 이진 트리 : Ω(lg n) [2] 선형 체인 트리 : Ω(n) [3] 임의로 만들어진 이진 트리 : O(lg n)

    (5) 용어 [1] 루트 : 조상이 없다 [2] 리프 : 자식이 없다 [3] 내부 : 리프가 아닌 노드 [4] 높이 : 루트에서 리프까지의 거리

    (6) 이진 트리의 종류 [1] Degenerate 오직 한명의 자식을 갖는 트리 선형 리스트와 유사 높이 : O(n) (n개의 노드에서) [2] Balanced 거의 두명의 자식을 갖는 트리 검색에 유용한 트리 높이 : O(lg n) (n개의 노드에서) [3] Complete 항상 두명의 자식을 갖는 트리

  • 이진 검색 트리의 개념, 특성 (1) 이진 검색 트리의 개념 key, 데이터, left(왼쪽 자식), right(오른쪽 자식), p(부모) 등의 필드를 갖는다.
    (2) 이진 검색 트리의 특성 x가 이진 검색 트리의 한 노드이다. 이때 y가 x의 왼쪽 서브 트리의 한 노드이면, y.key <= x.key를 만족한다. 그리고 y가 x의 오른쪽 서브 트리의 한 노드이면, y.key >= x.key를 만족한다. (3) 구현 관점의 특성 연결된 데이터 구조로 표현됨(각 노드는 객체를 의미) > 키 + 위성 데이터

  • 이진 검색 트리 순회 트리의 있는 노드들을 특정한 순서로 탐색하는 기술

    트리의 각 노드에서 재귀적으로 호출하므로, n개의 노드로 이루어진 이진 검색 트리에서 Θ(n)을 만족

    (1) 전위 트리 순회

        PREORDER-TREE-WALK(x)
            if x != NIL
                print x.key
                PREORDER-TREE-WALK(x.left)           
                PREORDER-TREE-WALK(x.right)
    

    시간 복잡도

        1. BaseCase
        2. n이 k보다 작을때, 참이라고 가정
        3. n이 k일때, 증명
    
      >  
    

    (2) 중위 트리 순회

        INORDER-TREE-WALK(x)
            if x != NIL
                INORDER-TREE-WALK(x.left)           
                print x.key
                INORDER-TREE-WALK(x.right)
    

    시간 복잡도

        1. BaseCase
        2. n이 k보다 작을때, 참이라고 가정
        3. n이 k일때, 증명
    
      > n개의 노드로 이루어진 서브 트리의 루트에 대하여 호출시, 걸리는 시간을 T(n)이라 한다.
        중위 트리 순회는 n개의 노드를 모두 방문하기에, T(n) = Ω(n)
      > 빈 트리의 경우, 상수 c>0에 대해 T(0) = c (경미한 상수 시간 소모)         
      > n>0의 경우 왼쪽 서브 트리가 k개의 노드를, 오른쪽의 서브 트리가 n-k-1을 갖는다.
        이를 다시 표현하면 T(n) <= T(k) + T(n-k-1) + d
      > 치환을 통해, T(n) <= (c+d)n + c를 증명
        1) n = 0
          (c+d) * 0 + c = c = T(0)
        2) T(n) <= T(k) + T(n-k-1) + d
                <= ((c+d)k + c) + ((c+d)(n-k-1) + c) + d
                <= (c+d)n + c - (c+d) + c + d
                <= (c+d)n + c
    

    (3) 후위 트리 순회

        POSTORDER-TREE-WALK(x)
            if x != NIL
                POSTORDER-TREE-WALK(x.left)           
                POSTORDER-TREE-WALK(x.right)
                print x.key
    
  • 이진 검색 트리 검색

        TREE-SEARCH(x,k)
            if x == NIL or K == x.key
                return x
            if k < x.key
                return TREE-SEARCH(x.left, k)
            else
                return TREE-SEARCH(x.right, k)
    

    시간 복잡도

        1. BaseCase
        2. n이 k보다 작을때, 참이라고 가정
        3. n이 k일때, 증명
    
      > 트리의 높이가 h일때, O(h)
    
  • 이진 검색 트리 최소, 최대 원소 (1) 최소 원소

        TREE-MINIMUM(x)
            while x.left != NIL
                x = x.left
            return x
    

    (2) 최대 원소

        TREE-MAXIMUM(x)
            while x.right != NIL
                x = x.right
            return x
    

    시간 복잡도

        1. BaseCase
        2. n이 k보다 작을때, 참이라고 가정
        3. n이 k일때, 증명
    
      > 트리의 높이가 h일때, O(h)
      > 이진 탐색 트리의 특성이 해당 알고리즘의 정확함을 보장한다. 이는 루트로부터 내려가는 하나의 단순 경로에 존재함을 보장
    
  • 이진 검색 트리 직후, 직전 원소 직후원소란 주어진 노드 x보다 큰 key들 중 가장 작은 값을 의미. (직전은 반대)
    (1) 직후 원소

        TREE-SUCCESSOR(x)
            if x.right != NIL
                return TREE-MINIMUM(x.right)
            
            y = x.p
            
            while y != NIL and x == y.right
                x = y
                y = y.p
            
            return y
    

    (2) 직전 원소

        TREE-PREDECESSOR(x)
            if x.left != NIL
                return TREE-MAXIMUM(x.left)
                
            y = x.p
            
            while y != NIL and x == y.left
                x = y
                y = y.p
                
            return y
    

    시간 복잡도

        1. BaseCase
        2. n이 k보다 작을때, 참이라고 가정
        3. n이 k일때, 증명
    
      > 트리의 높이가 h일때, O(h)
      > 이진 탐색 트리의 특성이 해당 알고리즘의 정확함을 보장한다. 이는 루트로부터 내려가는 하나의 단순 경로에 존재함을 보장
    
  • 이진 검색 트리 삽입

        TREE-INSERT(T,z)
            y = NIL
            x = T.root
            
            while x != NIL
                y = x
                if z.key < x.key
                    x = x.left
                else 
                    x = x.right
            
            z.p = y
            
            if y == NIL
                T.root = z
            else if z.key < y.key
                y.left = z
            else 
                y.right = z
    

    시간 복잡도

        1. BaseCase
        2. n이 k보다 작을때, 참이라고 가정
        3. n이 k일때, 증명
    
      > 트리의 높이가 h일때, O(h)
    
  • 이진 검색 트리 삭제 & TRANSPLANT 이진 탐색 트리에서의 노드 삭제를 할 땐, 3가지의 경우가 존재한다.

    1. 해당 노드의 자식이 없을 경우 해당 노드를, 해당 노드의 자식인 NIL로 대체하도록 수정한다.

    2. 해당 노드가 1명의 자식을 갖는 경우 해당 노드의 자식과 부모 사이에 새로운 연결고리를 만들어서 삭제할 노드를 분리한다.

    3. 해당 노드가 2명의 자식을 갖는 경우 해당 노드의 직후 원소와 새로 이어준다. 이 직후 원소는 자식이 없다.

    (1) TRANSPLANT

        TRANSPLANT(T,u,v)
            if u.p == NIL               -- if u doesn't have a parent => u is the root
                T.root = v              --   then v must replace u as the root of the tree T
             
            else if u == u.p.left       -- if u is a left subtree of its parent
                u.p.left = v            --   then v must replace u as the left
                                        --   subtree of u's parent
            else                        -- otherwise u is a right subtree 
                u.p.right = v           --   (as the tree is binary) and v must replace
                                        --   u as the right subtree of u's parent
            if v != NIL                 -- if v has replaced u (and thus is not NIL)
                v.p = u.p               --   v must have the same parent as u
    
      > 한 서브트리를 다른 서브 트리로 교체하는 루틴
    

    (2) TREE DELETE

        TREE-DELETE(T,z)
            if z.left == NIL
                TRANSPLANT(T,z,z.right)
            else if z.right == NIL
                TRANSPLANT(T,z,z.left)
            else
                y = TREE-MINIMUM(z.right)
                if y.p != NIL
                    TRANSPLANT(T,y,y.right)
                    y.right = z.right
                    y.right.p = y
                TRANSPLANT(T,z,y)
                y.left = z.left
                y.left.p = y
    

    시간 복잡도

        1. BaseCase
        2. n이 k보다 작을때, 참이라고 가정
        3. n이 k일때, 증명
    
      > 트리의 높이가 h일때, O(h)
    




AVL Trees

  • 기본 설명

    1. 이진 탐색 트리의 성질을 기반으로 가짐
    2. 트리 내의 어떤 노드에서든, 노드 기준 좌측 서브트리의 높이와 우측 서브트리의 높이 차이는 최대 1이다.
  • AVL 트리의 특성

    1. 뿌리 왼쪽과 오른쪽 하위 트리의 높이가 최대 1씩 차이가 나고, 오른쪽 및 왼쪽 하위 트리도 AVL 트리인 BST이다.
    2. BST의 특징을 기본으로, 각 노드별 높이 균형을 보장 > 회전 운영
function rotateRight (root):
    exchange left subtree with right subtree of left subtree
    make left subtree a new root

function rotateLeft (root):
    exchange right subtree with left subtree of right subtree
    make right subtree a new root
  • AVL에서의 회전 방식 4가지

    1. Single Left Rotation
    2. Double Left Rotation
    3. Single Right Rotation
    4. Double Right Rotation

      SLR, DLR : left(+2) SRR, DRR : right(-2)

  • AVL에서의 삽입

    1. 기존 이진 탐색 트리 방식대로 노드 삽입.
    2. 삽입한 노드로부터 루트로 올라가면서 먼저 만나게 되는 불균형을 회전을 통해 바로 잡아야 한다.
        function AVLInsert (root, newData):
            if subtree is empty:
                insert newData as root
                return root
            
            if newData < root:
                AVLInsert (left subtree, newData)
                if left subtree is taller:
                    leftBalance (root)
            
            else:
                AVLInsert (right subtree, newData)
                if right subtree is taller:
                    rightBalance (root)
            
            return root
            
        
        
        function leftBalance (root):
            if left tree is higher:
                rotateRight (root)
            else:
                rotateLeft (left subtree)
                rotateRight (root)
      
      
        functino rightBalance (root):
            if right tree is higher:
                rotateLeft (root)
            else:
                rotateRight (right subtree)
                rotateLeft (root)    
    
  • AVL에서의 삭제

    1. BST의 삭제 방법과 동일하게 노드(w) 삭제
    2. 노드(w)로부터 불균형한 첫번째 노드(z)를 찾는다.
    3. 불균형한 노드(z)의 자식노드들 중 깊이가 큰 자식노드(y)를 선정
    4. 깊이가 큰 자식노드(y)의 자식노드들 중 깊이가 큰 노드(x)를 선정
    5. z를 루트로 하는 자식트리를 회전을 통해 재배열.
      function AVLDelete (root, dltKey, success):
          if subtree is empty:
              set success to false
              return null
          
          if dltKey < root:
              set left subtree to AVLDelete (left subtree, dltKey, success)
              if tree is shorter:
                  set root to deleteRightBalance (root)
          
          else if dltKey > root:
              set right subtree to AVLDelete (right subtree, dltKey, success)
              if tree is shorter:
                  set root to deleteLeftBalance (root)
          
          else:
              save root
              if no right subtree:
                  set success to true
                  return left subtree
              
              else if no left subtree:
                  set success to true
                  return right subtree
              
              else:
                  find largest node on left subtree
                  save largest key
                  copy data in largest to root
                  set left subtree to AVLDelete (left subtree, largest key, success)
                  if tree is shorter:
                      set root to deleteRightBalance (root)
          return root
    
    
        function deleteRightBalance (root):
            if tree is not balanced:
                set rightOfRight to right subtree
                if left of rightOfRight is higher:
                    set leftOfRight to left subtree of rightOfRight
                    set right subtree to rotateRight (rightOfRight)
                    set root to rotateLeft (root)
                else:
                    set root to rotateLeft (root)
            return root
    
  • AVL에서의 높이

    1. 전체 높이가 h일때, AVL 트리에 들어갈 수 있는 노드 개수의 최솟값 T(h)라고 명
    2. T(1) = 1 / T(2) = 2 / T(h) = T(h-1) + 1 + T(h-2), h>=3
    3. 증명
        1) h=1,2 일때 성립.
        2) h = k, k+1일 때 성립한다고 가정(k>=1)
        3) T(k+2) = T(k+1) + 1 + T(k)  >=  2^((k+1)/2-1) + 1 + 2^(k/2-1)  >=  2*2^(k/2-1) = 2^((k+2)/2-1) 
           > h = k+2일때도 성립
        4) n >= T(h) >= 2^(h/2-1)
           ∴ h ≤ 2logn + 2
    

## GRAPH
  • 그래프의 표현 / Graph G = (V, E)

    1. V는 노드들의 집합
    2. E는 V의 노드들간을 연결하는 간선들의 집합
  • 그래프의 유형

    1. 무방향 그래프

      1. edge (u,v) = (v,u)를 만족한다.
      2. (v,v)는 E에 들어가지 않는다. 즉, V의 노드들에 있어서 셀프 루프가 존재하지 않는다.
    2. 방향 그래프

      1. (u,v)는 노드 u에서 v로 이어지는 간선으로, u -> v로 표현한다.
      2. 셀프 루프가 허용된다.
    3. 비중 그래프

      1. 각 간선들은 주어진 중량 함수 w에 의해 주어진 중량치를 갖는다.
    4. 밀집 그래프

      1. 간선의 수가 노드수의 제곱과 거의 유사한 그래프
    5. 희소 그래프

      1. 간선의 수가 노드수의 제곱에 비해 한참 적을 때

    +) |E| = O(|V2|)
    6. 오일러 그래프 1. 모든 '간선'를 정확히 한 번 포함하고 초기 꼭지점에서 끝나는 경로를 포함하는 그래프를 오일러 그래프라고 한다.

    1. 해밀턴 그래프
      1. 모든 '노드'를 정확히 한 번 포함하고 초기 꼭지점에서 끝나는 경로를 포함하는 그래프를 해밀턴 그래프라고 한다.
  • 그래프에서의 인접

    1. 만약 (u,v)가 E에 속할 경우, 노드 v는 노드 u와 인접한다.
    2. 인접 관계는 다음과 같은 성질을 갖는다.
      1. G가 무방향 그래프일 경우 대칭이다.
      2. G가 방향 그래프일 경우 반드시 대칭이진 않는다.
  • 그래프가 Connected

    1. 모든 노드쌍에 대해서 노드쌍을 잇는 경로가 반드시 존재한다는 것.
    2. |E| >= |v|-1
    3. 만약 |E| = |v| - 1일 경우 해당 그래프 G는 트리다
  • 트리와 평면 그래프

    1. 각 노드쌍 사이에 하나의 경로만 있는 연결된 그래프를 트리라고 한다. = 트리는 사이클이 없는 연결된 그래프로도 정의할 수 있다
    2. 교차 없이 다시 그릴 수 있는 그래프를 평면 그래프라고 한다.
  • 그래프의 이형성

    1. 두 방향 그래프간 이형성이 존재한다는 것은, 각 방향 그래프들의 기초 그래프들 간 사이에도 이형성이 존재한다는 것이다.
  • Connected와 Strongly Connnected

    1. 만약 방향 그래프 D가 두개의 방향그래프의 결합으로 설명될 수 없다면, D는 connected이다.
      1. 이는 D의 기초 그래프가 connected 그래프라는 것과 같다.
      2. 이때 기초 그래프란 방향 그래프의 간선들의 방향을 제거하여 얻는 그래프를 의미한다.
    2. 그래프 D의 임의의 노드 v, w에 대하여 D가 Strongly connected라는 것은, v에서 w로 가는 경로가 존재한다는 것을 의미한다.
    3. 모든 Strongly connected 그래프는 connected이지만, 반대는 성립하지 않는다.
    4. 그래프 G의 모든 간선이 방향화되어 그래프 G가 Strong connected 될 수 있다면, 해당 그래프 G를 방향화(orientable) 그래프라고 명명한다.
  • 차수

    1. 노드 v의 차수는, v와 만나는 간선의 개수이다.
    2. v의 루프는 2개의 차수로 판단한다.
    3. 차수가 0인 노드는 고립된 노드이다.
    4. 차수가 1인 노드는 말단 노드이다.
    5. "Handshaking Lemma"와 그것의 결과를 기억해야한다.
  • Handshaking Lemma

    1. 어떤 그래프에서라도 모든 노드의 차수 합은 짝수이다. = 간선의 수 * 2
    2. 이것은 어떤 그래프의 홀수 차수를 갖는 노드의 수는 짝수라는 결과를 가져온다.
  • Handshaking Dilemma

    1. 방향 그래프에서 노드 v에 대하여 들어오는 간선의 수를 in degree라고 한다.
    2. 방향 그래프에서 노드 v에서 나가는 간선의 수를 out degree라고 한다.
    3. 그래프 G 모든 노드의 out degree 합은, in degree 합과 같다. = Handshaking Dilemma
  • 공간 요구도

    1. 방향 그래프의 모든 인접 리스트들의 길이 합은, 모든 노드에서의 out degree의 합과 같기 때문에 |E|가 된다. 따라서 총 공간 요구도는 θ(V+E)이다.
    2. 무방향 그래프의 모든 인접 리스트들의 길이 합은, 모든 노드에서의 차수의 합과 같기 때문에 |E|이다. 따라서 총 공간 요구도는 θ(V+E)이다.
  • 인접

    1. 두 개의 노드 v, w가 인접한다는 것은, (v,w) 간선이 존재한다는 것을 의미한다. 그리고 v와 w는 (v,w) 간선과 만난다고 표현한다.
    2. 두 구분되는 간선 e, f가 인접한다는 것은, 그들이 공통된 노드를 갖는 것을 의미한다.
    3. 그래프 G = (V,E) 표현하기 위해서, 크게 두가지 방식으로 진행한다.
    • 인접 리스트

      1. 노드 리스트에 대한 하나의 인접 배열로 구성된다
      2. 하나의 리스트는 각 노드에 대한 대한 것이다
      3. 노드 집합에 속하는 특정 노드(u)에 대한 인접 배열은, 특정 노드에 인접하는 모든 노드들로 구성되어 있다.
      4. 작은 밀도 그래프(|E|가 |V|²보다 훨씬 작을 때)에 대해 효율적
      5. 무방향, 방향 그래프 모두 사용 가능
    • 인접 행렬

      1. 높은 밀도 그래프(|E|가 |V|²과 거의 비슷할 때)에 대해 효율적
      2. 주어진 두 정점을 연결하는 간선이 있는지 빠르게 확인할 때 효율적
      3. 무방향, 방향 그래프 모두 사용 가능
  • 인접 리스트 표현

    • 장점

      1. 희소 그래프일 때, 공간을 효율적으로 사용할 수 있다.
      2. 많은 그래프 변형을 지원한다.
    • 단점

      1. 특정 간선이 그래프내에 존재하는지 결정할 때, 비효율적이다. = 시작 노드에서의 인접 리스트를 검사해야 하며, θ(dev(v))가 된다. 즉, 최악의 경우 θ(v)이다.
    • |V|개 리스트의 배열 Adj로 구성되어 있고, 각 리스트는 V에 들어있는 정점 하나에 대한 것

    • 각 u(∈ V)에 대하여, 인접 리스트 Adj[u]는 간선 (u,v)(∈ E)가 존재하는 모든 정점 v를 포함한다. = Adj[u]는 그래프 G에서 정점 u에 인접해 있는 모든 정점으로

    • 그래프가 방향 그래프의 경우, (u,v) 간선은 Adj[u]에 정점 v가 나타나도록 표현됨.

      모든 인접 리스트들의 길이의 총 합은 |E|

    • 그래프가 무방향 그래프인 경우, (u,v) 간선이 무방향 간선이므로 u가 v의 인접 리스트에 나타나고 v가 u의 인접 리스트에 나타난다.

      모든 인접 리스트들의 길이의 총 합은 2|E|

    • 방향과 무방향 그래프 모두에 대해 필요한 메모리의 양이 Θ(V + E)

    • 주어진 간선(u,v)이 그래프에 있는지 확인하기 위해, 인접 리스트 Adj[u]에서 v를 검색하는 것보다 빠른 방법이 없음

  • 인접 행렬 표현

    • 정점에 임의의 순서로 1,2, ..., |V| 번호가 매겨져 있다 가정

    • |V| * |V|의 행렬 A = (aij)이 된다

      aij = 1 ((i, j) ∈ E) = 0 (그 외의 경우)

    • 공간 및 시간 복잡도

      1. 공간 복잡도 θ(V2)
      2. 모든 인접 노드들을 결정하는 시간 θ(V)
      3. (u,v)간선이 존재하는지 결정하는 시간 θ(1)
    • 그래프에 있는 간선 개수와 상관없이 Θ(V2)의 메모리가 필요

    • 무방향 그래프에선 간선(u,v)와 (v,u)는 같은 간선을 의미하므로, 인접행렬 A는 그 자신의 전치행렬과 같다

    • 인접 리스트에 비해 더 단순하여, 그래프가 어느정도 작을 경우 이를 더 선호

  • 그래프 탐색 알고리즘

    너비 우선 탐색, 깊이 우색 탐색 등 대표적인 방법이 있다.

  • 너비 우선 검색

    1. 인풋
      • 방향 혹은 무방향 Graph G = (V,E)와 시작 노드 s (V에 존재)
    2. 아웃풋
      • d[v] : s부터 특정 노드 v(V에 존재)까지의 거리 (간선들의 가장 작은 합 혹은 최단 경로) s로부터 도달할 수 없다면 무한대
      • π[v] : 최단 경로에서의 도착 노드의 직전 노드.
      • 시작 노드 s를 루트 노드로 하는 너비 우선 탐색 트리(도달 가능한 모든 노드가 포함됨)
    3. 설명
      • 발견된 노드와 아직 발견되지 않은 노드간의 경계를 전체적으로 점차 균일하게 확정해가며 수행한다.
      • 발견된 노드란 탐색 중 처음으로 만난 노드를 말한다.
      • finished된 노드란 해당 노드의 모든 인접 노드들이 발견된 노드를 말한다.
      • 색을 이용하여 노드 상태를 나눈다. 1. 하얀색 - 발견되지 않음 2. 회색 - 발견은 되었으나 끝나진 않음 3. 검은색 - 끝남
    • 그래프 G = (V,E)와 한 개의 구별되는 출발점 s에 대한, 너비 우선 검색은 s로부터 도달할 수 있는 모든 정점을 발견하기 위해 G의 간선들을 체계적으로 탐색
    • s로부터 거리가 k+1인 한 정점을 만나기 전에, s로부터 거리가 k인 정점을 모두 발견하는 알고리즘
    • 탐색한 간선들을 기반으로, s로부터 도달할 수 있는 각 정점까지의 거리를 계산
    • s를 루트로 하고 s에서 닿을 수 있는 모든 정점을 가지는 너비 우선 트리를 만듬
    • s에서 도달할 수 있는 모든 정점 v에 대한 너비 우선 트리에서, s에서 v까지의 단순 경로는 그래프 G의 s에서 v까지의 최단 경로에 해당(가장 적은 간선을 갖는 경로)
    • 방향 및 무방향 그래프 모두 적용
    1. 책내용 정리

        - 진행 정도를 따라가기 위해 각 정점을 흰색, 회색 또는 검은색으로 칠해나간다
        - 모든 정점은 처음에 흰색으로 시작해 회색이 됐다가, 다시 검은색이 된다
        - 정점은 검색 도중에 처음으로 발견되면 흰색이 아닌 다른 색으로 변화된다 (회색과 검은색의 정점은 이미 발견되었음을 의미)
        - 회색과 검은색을 구분하는 것은 검색이 너비 우선 방식으로 수행되도록 보장하기 위함이다.
          = 간선(u,v)(∈ E)이고 정점 u가 검은색이라면, 정점 v는 회색이거나 검은색이다 (검은색 정점에 인접해있는 모든 정점은 이미 발견됨)
            회색 정점은 인접한 흰색 정점을 가질 수 있는데, 이는 발견된 정점과 발견되지 않은 정점 사이의 경계선을 나타낸다
      
        - 너비 우선 탐색은 너비 우선 트리를 만드는데, 이 트리는 출발점 s만을 루트로 갖는다.
        - 이미 발견된 정점 u의 인접 리스트를 스캔하는 도중에 흰색 정점 v가 발견될 때마다, 정점 v와 간선(u,v)가 트리에 더해진다.
          이때, u를 v의 너비 우선 트리의 직전원소 또는 부모라고 한다. (모든 정점은 많아야 한번 발견되므로, 부모 정점은 많아야 각각 한개)
        - 너비 우선 트리에서 조상과 자손의 관계는 일반적인 경우처럼 루트 s에 대해 상대적으로 정의됨.
          = 루트 s에서 정점 v까지의 트리에 있는 단순 경로에 정점 u가 있으면 u는 v의 조상이 되고 v는 u의 자손이 된다.
      
    2. BFS 수도코드

        BFS(G,s)
          for 각각의 정점 u(∈ G.V - {s})
              u.color = WHITE
              u.d = ∞
              u.𝝿 = NIL
      
          s.color = GRAY
          s.d = 0
          s.𝝿 = NIL
          Q = Ø
      
          ENQUEUE(Q,s)
      
          while Q != Ø
              u = DEQUEUE(Q)
              for 각각의 정점 v(∈ G.Adj[u])
                  if v.color == WHITE
                      v.color = GRAY
                      v.d = u.d + 1
                      v.𝝿 = u
                      ENQUEUE(Q,v)
              u.color = BLACK
      
      1. 출발점 s를 제외한 모든 정점을 흰색으로 칠하고, 각 정점 u에 대하여 u.d를 무한대의 값으로 설정하며, 각 정점의 부모 정점을 NIL로 설정
      2. s.d는 0으로 s의 직전 원소를 NIL로 바꾸고 s는 회색으로 만드는데, 이는 프로시저가 처음 시작할 때 이미 s는 발견된 것으로 간주하기 때문
      3. Q는 정점 s만 포함하고 있는 큐로 초기화
      4. while 루프는 회색 정점이 남아있는동안 반복. 회색 정점의 의미는 발견은 되었지만 인접 리스트가 완전히 조사되지 않은 정점임
         > 해당 루프는 Q(큐)에 있는 모든 정점이 회색일때까지 반복
      5. Dequeue는 Q의 헤드에 위치하는 회색 정점 u를 알아내고, Q에서 해당 정점을 제거한다.
      6. for 루프는 u의 인접 리스트에 있는 각 정점 v를 조사하고, v가 흰색인 경우 아직 발견되지 않음을 의미함로 발견과정을 거친다.
      7. u의 인접 리스트에 잇는 모든 정점을 검사하면 u를 검은색으로 칠한다.
      
    3. 시간 복잡도

      • 초기화에 걸리는 시간은 O(V)
      • 루프내부 [1] 초기화 이후, 각 노드에 대해서 큐 삽입 삭제 연산은 O(1)이 걸리므로, 모든 노드에서 큐잉을 처리하는데 걸리는 시간은 O(V) [2] 인접 리스트의 각 노드들은 최대 한번만 탐색된다. 따라서 인접 리스트들의 총 합은 θ(E)
      • 모든 노드에 대해서 시간 복잡도를 합하면 O(V+E)를 만족한다.
    4. 너비 우선 트리

      1. 그래프 G = (V,E)와 시작점 s에서, G의 predecessor 서브 그래프 Gπ = (Vπ, Eπ)가 있다. 이때 Vπ = {v ∈ V: π[v] != NIl } + {s} Eπ = {(π[v], v)∈E, v ∈ Vπ - {s}}
      2. 다음과 같은 특성을 만족할 경우 predecessor 서브 그래프 Gπ는 깊이 우선 트리다. [1] Vπ는 s로부터 도달할 수 있는 노드들로 구성된다. [2] Vπ에 있는 모든 노드에 있어서, Gπ에는 s부터 v까지의 유일한 단순 경로가 존재한다. 또한 이는 s부터 v까지의 최단 경로이다. [3] Eπ에 있는 간선은 트리 간선이라 불린다. |Eπ| = |Vπ| - 1
    • 최단 경로

          중요한 특성
          - G = (V,E)를 방향 또는 무방향 그래프라 하고 s(∈ V)를 임의의 정점이라 하자. 그러면 간선(u,v)(∈E)에 대해 다음이 성립
      
            δ(s,v) <= δ(s,u) + 1
      
              (1) 증명
                  s에서 u로 도달할 수 있다면, v에서도 s에서 도달할 수 있다.
                  이 경우, s에서 v까지의 최단 경로는 s에서 u까지의 최단 경로에 간선(u,v)를 바로 뒤에 넣은 경로보다 멀 수 없으므로 위의 부등식은 성립한다.
                  u가 s에서 도달할 수 없다면 δ(s,u) = ∞이고 부등식은 성립.
              (2) 결론
                  BFS가 각각의 정점 v(∈ V)에 대해 v.d = δ(s,u)를 적절하게 계산함을 보이고 싶다. 이때 위 정리에 의해 v.d가 δ(s,u)의 상한임을 보인다.
      
      
          - G = (V,E)가 방향 또는 무방향 그래프고, BFS를 그래프 G에서 주어진 출발점 s(∈ V)로부터 수행시킨다고 가정. 그러면 BFS가 끝났을 때 각 정점 v(∈ V)에 대해 BFS로 계산한 v.d는 v.d >= δ(s,u)를 만족한다.
      
              (1) 증명
                  ENEQUEUE 연산의 개수에 대해 귀납법을 사용한다. 귀납 가정은 모든 v(∈ V)에 대해 v.d >= δ(s,u)라는 것이다.
                  이 귀납법의 기본은 s가 BFS의 수행 과정 중 큐에 들어갔을 때 상황. 여기서 모든 v(∈ V - {s})에 대해 s.d = 0 = δ(s,s)이고 v.d = ∞ >= δ(s,v)이므로 귀납 가정이 성립한다.
      
                  귀납 단계를 위해 정점 u부터 검색하는 동안 발견된 흰색 정점 v를 생각하였을 때, 귀납 가정은 u.d >=  δ(s,u)를 의미한다. 
                  "v.d = u.d + 1"와 위의 정리를 통해 다음 식이 도출된다.
                  v.d = u.d + 1
                     >= δ(s,v) + 1
                     >= δ(s,v)
      
                  이후 정점 v가 큐에 들어가고 회색이 되며, 이후 동작은 흰색 정점에 대하여 수행되기에 해당 정점은 다시 큐에 들어가지 않는다.
                  > v.d값이 다시 변하지 않고 유지된다.
      
              (2) 결론
                  v.d = δ(s,v)임을 증명하기 위해 BFS를 수행하는 동안 큐가 어떻게 동작하는지 정확이 보여야 되는데, 아래 정리는 큐가 서로 다른 d값을 최대 2개까지 가질 수 있음을 보임
      
      
          - 그래프 G = (V,E)에 대해 BFS를 수행하는 동안 큐(Q)는 정점 <v1, v2, ..., vr>을 가지고, v1은 Q의 헤드에, vr은 꼬리에 위치한다고 가정. 그러면 i = 1, 2, ..., r-1일 때 vr.d <= v1.d + 1이고 vi.d <= vi+1.d다.     
      
              (1) 증명
                  큐 연산의 개수에 대한 귀납법 의한다. 처음에 큐가 s만 가지고 있을때는 보조정리가 당연히 성립
                  귀납 단계에선 큐에 정점을 하나 삽입하거나 삭제한 후에도 위가 성립함을 보여야한다.
                  큐의 헤드인 v1이 삭제된다면 v2가 새로운 헤드가 된다.(큐가 비었으면 이 보조정리는 자동으로 성립)
                  귀납 가정에 의해 v1.d <= v2.d. 이는 vr.d <= vr.d + 1 <= v2.d + 1이므로 나머지 부등식은 영향을 받지 않음
                  따라서 이 보조정리는 v2가 헤드인 경우에도 성립
      
                  큐에 정점을 하나 삽입하면, 이는 vr+1이 되고, 그때 인접 리스트를 스캔하는 중 노드 u는 큐에서 이미 제거되었고 귀납 가정에 의해 새로운 헤드 v1은 v1.d >= u.d가 된다
                  그러므로 vr+1.d <= u.d + 1 = v.d = vr+1 + d이므로 나머지 부등식은 바뀌지 않는다. 
                  따라서 v가 큐에 삽입된 때도 위 정리는 성립
      
              (2) 결론
                  정점이 삽입될 때 d값이 시간이 지남에 따라서 단조증가함을 보이는 것은 아래의 정리가 보여준다.
      
      
          - BFS를 수행하는 도중에 정점 vi와 vj가 큐에 삽입되고, vi가 vj보다 먼저 삽입된다고 가정. 그러면 vj가 들어갈 때, vi.d <= vj.d가 된다.
      
              (1) 증명
                  위의 정리와 BFS를 수행하는 동안 각 정점은 최대 한 번 d값을 받는다는 특성에 의해 바로 증명됨.
      
      
          - 너비 우선 검색의 정확성
      
              G = (V,E)는 방향 또는 무방향 그래프고 주어진 출발점 s(∈ V)로부터 G에서 BFS가 수행된다고 가정. 
              그러면 수행 중에 BFS는 출발점 s로부터 도달할 수 있는 모든 정점 v(∈ V)를 찾고, 끝나면 모든 정점 v(∈ V)에 대해 v.d = δ(s,v)가 된다.
              게다가 s에서 도달할 수 있는 v != s인 어떤 정점 v에 대해서도 s로부터 v까지의 최단 경로 중 하나는 s에서 v.𝝿까지의 최단 경로에 간선 (v.𝝿, v)가 바로 뒤에 붙어있는 경로
      
              (1) 증명
                  모순을 이끌어내기 위해 한 정점이 최단 경로 거리가 아닌 d값을 가진다고 가정.
                  v는 최소 값 δ(s,v)를 가지지만, 틀린 d값을 받는다고 가정.
                  v != s
      
                  위의 정리에 의해 v.d >= δ(s,v)이므로 v.d > δ(s,v)이다. 
                  정점 v가 s로부터 도달할 수 없다면 δ(s,v) = ∞ >= v.d이므로 v는 반드시 s로부터 도달할 수 있어야 한다.
      
                  u를 s에서 v까지의 최단 경로에 있는 v 바로 앞에 있고 δ(s,v) = δ(s,u) + 1을 만족하는 정점이라 하자.
                  δ(s,u) < δ(s,v)이고, v를 고르는 방법 때문에 u.d = δ(s,u)이다. 
      
                  이를 종합하면 v.d > δ(s,v) = δ(s,u) + 1 = u.d + 1
      
      
                  이제 BFS가 Q에서 정점 u를 제거하는 상황을 생각한다.
                  이 경우, v은 흰색, 회색 또는 검은색이다. 
                  [1] v가 흰색이면 v.d = u.d + 1이기에 모순이다.
                  [2] v가 검은색이면 이 정점은 이미 큐에서 삭제되었으므로 v.d <= u.d이 성립되므로 모순.
                  [3] v가 회색이면 Q에서 u보다 먼저 삭제되고 v.d = w.d + 1을 만족하는 어떤 정점 w를 큐에서 꺼내면서 회색으로 칠해진다. 이는 w.d <= u.d 이므로 v.d = w.d + 1 < u.d + 1 이고 이는 역시 모순
                  그러므로 모든 정점 v(∈ V)에 대해 v.d = δ(s,v)라고 결론 내릴 수 있다.
      
                  s로부터 도달할 수 있는 모든 정점은 반드시 발견되어야 하는데, 이는 모두 발견되지 않는다면 이 정점은 ∞ = v.d >= δ(s,v)가 될수도 있기 때문
      
                  + v.𝝿 = u 라면 v.d = u.d + 1임. 이는 s에서 v.𝝿까지의 최단 경로를 구한 뒤 간선(v.𝝿, v)를 탐색하여 s에서 v까지 최단 경로를 얻을 수 있다.
      
    • 너비 우선 트리

      1. 출발점 s를 가지는 그래프 G = (V,E)에 대해 G의 직전원소 부분 그래프는 다음을 만족하는 G𝝿 = (V𝝿, E𝝿)로 정의 V𝝿 = {v ∈ V: v.𝝿 != NIL} U {s} 이고, E𝝿 = {(v.𝝿, v) : v, V𝝿 - {s}}이다
      2. V𝝿가 s에서 도달할 수 있는 정점으로 구성되어 있고, 모든 정점 v(∈ V)에 대해 부분 그래프 G𝝿는 s에서 v까지의 유일한 단순 경로를 가지며, 이 경로가 G의 s에서 v까지의 최단 경로이기도 하다면 직전 원소 부분 그래프 G𝝿는 너비 우선 트리다.
      3. 너비 우선 트리는 연결되어 있고 |E𝝿| = |V𝝿| - 1이므로 사실상 트리다.
      4. E𝝿에 있는 간선을 트리 간선이라 한다.

      아래 정리는 BFS에 의해 만들어진 직전원소 부분 그래프가 너비 우선 트리임을 보여준다.

          BFS를 방향 또는 무방향 그래프 G = (V,E)에 적용했을 때, 부분 그래프 G𝝿 = (V𝝿, E𝝿)가 너비 우선 트리가 되도록 𝝿를 만든다.
          
          증명 
              (u,v)(∈ E)이고 δ(s,v) < ∞라면(s에서 v로 도달할 수 있다면) v.𝝿 = u로 지정되고 그 역도 성립
              따라서 V𝝿는 s에서 도달할 수 있는 V의 정점으로 구성된다.
              G𝝿는 위의 정리에 의해 트리를 형성하므로, G𝝿는 s에서 V𝝿에 있는 각 정점까지 유일한 단순 경로를 포함한다. 
              위 정리를 귀납적으로 적용하면, 이런 형태의 모든 경로는 G에서 최단 경로라 결론 지을 수 있다.                 
      
  • 깊이 우선 탐색

    • 설명

      • 가장 최근에 발견되고 아직 조사하지 않은 간선을 가진 정점 v로부터 나오는 간선을 조사한다
      • v의 모든 간선이 조사되면 그 검색을 v를 발견하게 해준 정점으로부터 나오는 간선을 조사하기 위해 뒤로 되돌아간다.(predecessor)
      • "먼저 가능한 깊이 조사한다."
      • 이 과정은 원래의 출발점으로부터 도달할 수 있는 모든 정점이 발견될 때까지 계속한다.
      • 발견되지 않은 정점이 하나라도 남아있으면, 깊이 우선 탐색은 그 중 하나를 새 출발점으로 선택하고 해당 출발점으로부터 검색을 반복한다.
    • 입력

      1. 방향, 무방향 그래프 G = (V,E) / 시작 정점이 주어지지 않는다.
    • 출력

      1. 각 정점마다 1부터 2|V|사이의 정수인, 2가지의 종류의 시간 기록이 나온다. d[v] = 발견된 시간 (흰색에서 회색이 되는 순간) f[v] = 끝난 시간 (회색에서 검은색이 되는 순간)
      2. 𝝿[v]은 정점 u의 인접 리스트를 조사하는 동안 발견된 정점 v의 predecessor을 의미 (=u)
    • 상태

      1. 흰색 : 발견되지 않음
      2. 회색 : 발견은 되었으나 끝나진 않음
      3. 검은색 : 끝남
    • 의사코드

         DFS(G)
             for each vertex u ∈ V[G]
                 do 
                     color[u] = white
                      𝝿[u] = NIL
             
             time = 0
             
             for each vertex u ∈ V[G]
                 do if color[u] == white
                     then DFS-Visit(u)
                     
                     
                     
         DFS-Visit(u)
             color[u] = GRAY
             time = time + 1
             d[u] = time
             
             for each v ∈ Adj[u]
                 do if color[v] == WHITE
                     then 𝝿[v] = u
                         DFS-Visit(v)
             
             color[u] = Black
             time = time + 1
             f[u] = time
             
      
    • 시간복잡도

      1. DFS의 2개의 for문은 DFS-VISIT에 걸리는 시간을 제외하고, Θ(V)의 시간이 소요된다.
      2. DFS-VISIT은 V의 모든 흰색 정점에서 한번씩 수행되기에(회색으로 칠해지는 시점), for문은 |Adj[v]|번이 수행되며, 총 DFS-VISIT 수행시간은 모든 노드에서의 |Adj[v]|을 더한 것 , 즉 Θ(E)의 시간을 소요한다.
      3. 따라서 DFS 총 소요 시간은 Θ(V+E)
    • 깊이 우선 탐색 트리 (직전 원소 부분 그래프)

      1. 깊이 우선 탐색 트리에서 직전 원소 부분 그래프 G𝝿 = (V, E𝝿)이며, E𝝿 = {𝝿[v], v ∈ V and 𝝿[v] != NIL}이다.

        BFS와 어떻게 다른가?? 직전 원소 부분 그래프는 깊이 우선 포레스트를 형성하는데 이는 여러 개의 깊이 우선 트리로 구성된다. (이때 E𝝿를 트리 간선이라고 부른다.)

  • 간선 분류

    • 트리 간선

      1. 깊이 우선 forest에 존재.
      2. (u,v)를 탐색하면서 발견됨
    • Back 엣지

      1. (u,v): u가 v의 자손인 간선
      2. 깊이 우선 트리에 존재
    • Forward 엣지

      1. v가 u의 자손인 간선
      2. 하지만 트리 간선은 아님
    • Corss 엣지

      1. 그 밖의 모든 엣지들
  • Minimum Spaning Tree

    • 그래프로 보는 MST 모델

      1. 무방향 그래프 G = (V,E)
      2. Weight w(u,v)가 각 간선에 존재. (u,v) ∈ E
      3. 다음을 만족하는 T(⊆E)를 찾아라. [1] T connectes all vertices (T는 spanning 트리이다.) [2] w(T) = 모든 간선의 가중치 합을 최소화한 것.
    • 설명

      1. 모든 스패닝 트리에 걸쳐 가중치가 최소인 스패닝 트리를 최소 스패닝 트리 또는 MST라고 한다.
      2. 같은 가중치를 갖지만, 서로 다른 MST가 존재할 수 있다. (복수 존재 가능)
    • Optimal substructure 특성 MST는 Optimal substructure 특성을 만족하는데, 이는 optimal tree는 optimal 서브트리로 구성된다는 것이다.

      1. T를 그래프 G(중간에 (u,v)간선이 존재하는)의 MST라 하자.
      2. 간선 (u,v)를 제가하여, T를 T1, T2 2개의 트리로 파티셔닝한다.
      3. 여기서 증명할 주장은 다음과 같다. T1은 G1 = (V1, E1)의 MST이다. 그리고 T2는 G2 = (V2, E2)의 MST이다. 그렇다면 w(T) = w(u,v) + w(T1) + w(T2)인가? 이때, T1 또는 T2보다 더 나은 트리가 있을 수 없고, 그렇지 않으면 T가 차선책일 것이다.
    • Growing An MST

      1. MST의 특성 [1] |V|-1개의 간선을 갖는다. [2] cycle을 갖지 않는다. [3] 유일하지 않다.

      2. Building up the solution [1] 간선의 집합 A를 선언 [2] 처음에 A는 비어있다. [3] 루프를 돌면서 A에 간선들을 넣는다. 이때 루프는 A가 MST의 subset을 유지할때까지만 돈다. [4] A에 간선을 넣을땐 위의 조건을 만족해야 하고, A가 MST의 부분집합인 경우 (u,v)간선은 안전하다고 표현한다. = 마찬가지로 A U {(u,v)}가 어떤 MST의 부분집합인 경우 안전하다고 한다. > 우리는 오직 안전한 간선만을 넣는다.

      3. 이론 [1] T를 그래프 G의 MST, 그리고 A를 T의 서브트리라고 하자 [2] (u,v)는 A와 V-A를 잇는 최소 비중치 간선이다. [3] (u,v)는 T에 속하게 된다.

    • 의사코드

      GENERIC-MST(G,w)
             A = Ø
             while A does not form a spanning tree
                do find an edge(u,v) that is safe for A
                   A = A U {(u,v)}
             return A
      

      루프 조건을 이용하여 해당 알고리즘이 유효함을 보이기 [1] initialization 빈 집합은 해당 조건을 그냥 만족 [2] Maintenance 새로운 안전 간선을 추가하기 전까지, A는 어떤 MST의 부분집합임을 만족한다. [3] Termination 모든 간선들이 A에 추가되고 멈추면, A는 MST인 spanning tree이다

    • 안전 간선 찾기

      1. S(⊂ V)는 h는 포함하지만 g는 포함하지 않는 노드들의, 어떤 집합이다. = g는 V-S에 존재
      2. 어떤 MST에서, S와 V-S를 이어주는 간선이 반드시 최소 하나 존재한다.
      3. 정의 - S⊂V, A⊆E일때 [1] cut(S, V-S)은, S와 V-S간을 분리하는 노드들의 파티션이다. [2] (u,v)간선(∈ E)는 cut(S, V-S)를 가로지른다. [3] A cut respects A if and only if no edge in A crosses the cut [4] cut을 가로지르는 간선을 light edge라 한다. & 그리고 light edge의 비중치는 cut을 가로지르는 모든 간선들 중 최소이다.
    • Disjoint-Set Union 문제

      1. Disjoint set을 돕는 데이터 구조를 원함. Disjoint set : 서로 중복되지 않는 부분 집합들로 나눠진 원소들에 대한 정보를 저장하고 조작하는 자료 구조
      2. 다음과 같은 동작이 필요하다 [1] MakeSet(x) : S = S U {{x}} [2] Union(Si, Sj) : S = S - {Si, Sj} U {Si U Sj} [3] FindSet(x) : return Si ∈ S such that x ∈ Si
    • Kruskal 알고리즘

      1. 의사코드
        Kruskal()
            T =  Ø
            for each v ∈ V
                MakeSet(v)
            sort E by increasing edge weight w
            for each (u,v) ∈ E (in sorted order)
                if FindSet(u) != FindSet(v)
                    T = T U {{u,v}}
                    Union(FindSet(u), FindSet(v))
      
      1. 시간복잡도 [1] Sort : O(1) [2] MakeSet() : O(V) [3] FindSet() : O(E) [4] Union() : O(V)
    • Prim 알고리즘

      1. 의사코드
        MST-Prim(G, w, r)
            Q = V[G]
            for each u ∈ A
                key[u] = ∞
            key[r] = 0
            p[r] = NULL
            while (Q not empty)
                u = ExtractMin(Q)
                for each v ∈ Adj[u]
                    if (v ∈ Q and w(u,v) < key[v])
                        p[v] = u
                        key[v] = w(u,v)
      
      1. 시간복잡도 [1] (Q = V[G]), (key[u] = ∞), (key[r] = 0) => θ(V) [2] ( for each v ∈ Adj[u]), (if (v ∈ Q and w(u,v) < key[v])), (p[v] = u), (key[v] = w(u,v)) => u의 차수만큼 진행 이때, key[v] = w(u,v)는 Handshaking Lemma에 의해 θ(E)라는 결론을 내리고 이는 Decrease-key를 내포한다. [3] (2번 과정), ( while (Q not empty)), (u = ExtractMin(Q)) => |V| ==> θ(V)*T(Extract_Min) + θ(E)*T(Decrease-key)

      2. 프림 알고리즘 분석 Q T(EXTRACT-MIN) T(DEREASE-KEY) TOTAL array O(V) O(1) O(V2) binary heap O(lg V) O(lg V) O(ElgV) Fibonacci heap O(lg V) O(1) O(E+VlgV)

  • Single-Source Shortest Path

    • 문제

      1. 상황은 주어진 weighted directed graph G에서, 주어진 시작 노드 s에서 또 다른 노드 v까지의 최소 비용 경로를 찾을때이다.
      2. 최단 경로는 최소 비용이다
      3. 비용은 경로상의 간선들의 비용 합이다.
    • Optimal Substructure

      1. 최단 경로는 최단 서브경로들로 이루어져있다.
      2. 증명은 Contcontradiction을 이용한다. [1] subpath가 최단 경로가 아니라고 가정 [2] 그곳에는 반드시 더 짧은 경로의 subpath가 존재한다. [3] 현재 subpath는, 더 짧은 경로를 갖는 subpath로 대체될 수 있다. [4] 하지만 전체 경로는 이전 상태가 최단 경로였기에, 지금은 최단 경로가 아니다

      모순

    • Shortest Path Properties

      1. δ(u,v)를 u부터 v까지의 최단 경로의 비중치라고 하자
      2. 최단 경로는 Triangle inequality(삼각 부등식)을 만족한다. δ(u,v) <= δ(u,x) + δ(x,v)
      3. 증명은 다음과 같다.
      • Relaxatoin
        1. 최단 경로 알고리즘의 핵심 키는 휴식이다.
        2. 아이디어는, 모든 노드에 있어서 δ(s,v)에 대한 d[v]의 upper bound를 유지하는 것
          Relax(u,v,w) 
              if (d[v] > d[u]+w) 
                  then d[v]=d[u]+w;
        
    • Bellman-Ford Algorithm

      1. 의사코드

            BellmanFord()
                for each v ∈ V
                    d[v] = ∞
                d[s] = 0
                for i=1 to |V|-1
                    for each edge (u,v) ∈ E
                        Relax(u, v, w(u,v))
                for each edge (u,v) ∈ E
                    if (d[v] > d[u] + w(u,v))
                        return no solution
        
      2. Single-Source 최단 경로 문제 [1] 각 노드에 대해서 d[v], π[v] 계산한다. [2] negative edge weights를 허용한다. [3] s로부터 도달할 수 있는 negative-weight가 존재하지 않는다면 true 그렇지 않다면 false를 반환 [4] 아이디어 모든 간선을 순회(|V|-1번), 각 순회마다 각 간선에 대하여 relaxation을 수행

    • Dijkstra's Algorithm

      1. 입력 [1] 방향그래프 G(V,E), 이때 그래프에는 negative weight가 존재하지 않는다. [2] 시작점 src

      2. 출력 [1] 최단 경로의 길이.(시작 지점부터 G의 모든 노드들 간의)

      3. 아이디어 [1] V = {1, 2, 3, ..., n }이고, 1은 시작 노드이다. [2] S : 이미 선택된 노드들의 집합 [3] C : 남아있는 노드들의 집합 [4] D[1, 2, ..., n] : 최단 경로의 비용 [5] 1(소스)부터 S = {1, 2, …, n}까지 거리가 최소인 C의 노드 v를 S에 반복적으로 추가

      4. 의사코드

            Dijkstra ( L[1,…,n,  1,…,n] )
            /* L is cost array, L[i,j] : cost if (i,j) in E  or  : infinity if (i,j) is not in E */
            C <- { 2, 3, …, n }   
                for i <- 2 to n do D[i] <- L[1,i]   
                    repeat   (n-2) times                 
                        v <- a node in C s.t. D[v] = Min{ D[w] } for each w in C                                             C <- C – {v}�                 
                        for each w in C do D[w] <- Min{ D[w], D[v]+L[v,w] }   
            return D
        
      5. Single-source 최단 경로 문제 [1] negatice-weight edge가 없다 [2] 2개의 노드 집합을 유지한다. S = 최종 최단 경로에 존재하는 노드들(weight는 이미 정해짐) C = V-S에 존재하는 노드들 [3] V-S에 존재하는 노드 u를 반복적으로 선택하고, minimum 최단 경로 추정값 d[v]를 함께 선택한다.

    • Shortest Path 문제

      1. 입력 [1] 방향 그래프 G = (V,E) [2] weight 함수 w : E > R

      2. 경로의 weight [1] p = <v0, v1, ..., vk> [2] w(p) = i는 1부터 k까지 w(vi-1, vi)의 합

      3. 최단 경로의 weight [1] δ(u,v) = min {w(p) : u > v까지의 경로가 존재한다면} [2] 최단 경로(u에서 v로 가는)은 다음을 만족하는 어떤 경로 p이다 : w(p) = δ(u,v)

      4. 최단 경로 Representation [1] d[v] = δ(s,v) : 최단 경로 추정 처음엔 d[v] = ∞이지만, 알고리즘이 진행될수록 줄어든다 [2] π[v] = 경로 s에 존재하는 노드 v의 이전 노드 만약 이전 노드가 존재하지 않으면 NIL π는 트리를 점차 줄이는데, 이는 최단 경로 트리이다. [3] 최단 경로와 최단 경로 트리는 유일하지 않다.

    • Relaxation

      1. Relaxing an edge(u,v) [1] u를 통해 지남으로써 지금까지 발견된 v로의 최단 경로를 개선할 수 있는지 테스트 [2] 만약 d[v] > d[u] + w(u,v)이라면, 최단 경로를 개선시킬 수 있다. d[v], π[v]를 업데이트
  • All-Pairs Shortest Paths

    • 각 노드에 대하여 한번씩 Bellman-Ford 수행

      1. O(V2E) 만약 그래프가 밀집이라면(E=θ(V2)), O(V2E)가 된다.
      2. negative-weight 간선이 없다면, Dijkstra 알고리즘을 각 노드에 적용한다. 바이너리 힙에서 O(VElgV)이다. (밀집일 경우 O(V3lg V)이다) 피보나치 힙에선 O(EV+V2lg V)이다. (밀집의 경우 O(V3이다)
    • 입력

      1. 방향 그래프 G = (V,E)
      2. weight 함수 w : E > R
    • 계산

      1. 그래프의 모든 노드에 대하여, 최단 경로
      2. 결과 표현 (n*n 매트릑스에 최단 경로 δ(u,v)를 표시)
    • 설명

      1. 주어진 weight 인접 매트릭스에 해당하는 그래프 G가 존재. 노드 번호는 1부터 n까지이고, W = (wij), n*n 매트릭스, |V| = n가 존재할때 wij는 0(i = j), weight of (i,j)(i != j, (i,j)∈E), ∞ (i != j, (i,j) !∈ E)의 값을 갖는다.
      2. 결과는 n*n의 매트릭스가 나오고, D = (dij), dij = δ(i,j)이다
      3. 동적 프로그래밍을 해결하는데 사용된다
    • Optimal Substructure of a Shortest Path

      1. 성질 [1] All subpaths of a shortest path are shortest paths [2] i부터 j로 가는 최단 경로 p가 m개의 간선을 갖는다고 하자. [3] 만약 i = j라면 w(p) = 0이고, p에는 간선이 없다.
      2. 주장 i != j이고, 중간에 k라는 노드가 존재하며, 만약 i부터 j까지의 p가 존재하며, p'이라는 i부터 중간 노드 k까지의 최단 경로 + w(k,w)와 같다면 [1] p'은 최대 m-1개의 간선을 갖는다. [2] p'은 최단 경로이다. [3] δ(i,j) = δ(i,k) + wkj이다
    • Recursive Solution

      1. lij(m)는 i부터 j까지의 최단 경로의 weight를 의미하는 동시에, 최대 m개의 간선을 갖음을 의미한다.
      2. m = 0 : lij(0) = 0(i = j) or ∞ (i != j) m >= 1 : lij(m) = min {lij(m-1), min{lik(m-1)+wkj(m)}} (1 <= k <= n) = min {lik(m-1)+wkj(m)} (1 <= k <= m)
      3. i부터 j까지의 최단 경로는 최대 m-1개의 간선을 갖는다.
      4. 최대 m개의 간선을 갖는 i부터 j까지의 최단 경로는, j의 가능한 모든 predecessors를 포함한다.
    • Computing the Shortest Paths

      1. m = 1: lij(1) = Wij i부터 j가지의 간선은 1개로 제한된다. L(1) = W
      2. W = (wij)가 주어지고, L(1), L(2), ... , L(n-1)을 계산한다. 이때, L(m) = (lij(m))이다.
      3. L(n-1)은 실제 최단 경로를 포함한다. 주어진 L(m-1)과 W로 L(m)을 계산한다. = 하나의 간선을 추가한 최단 경로로 확장한다,
      4. 만약 그래프가 negative cycle을 갖지 않으면, 모든 단순 최단 경로는 최대 n-1개의 간선을 갖는다. δ(i,j) = lij(n-1) and lij(n), lij(n+1), ...m lij(n-1)
    • 최단 경로 확장

      1. lij(m) = min {lik(m-1)+wkj(m)} (1<=k<=n)
      2. 의사코드
            l_ij(m) = min { l_ik(m-1) + w_kj } (1<=k<=n)
                create L', an nxn matrix
                for i <- 1 to n
                    do for j <- i to n
                        do i_ij' = ∞
                            for k <- 1 to n
                                do i_ij' = min (i_ij', i_ik + w_kj)
                return L'
        
    • Slow-APSP(W,n)

      1. 의사코드
            L(1) = W
            for m = 2 to n-1
                do L(m) = EXTEND(L(m-1), W, n)
            return L(n-1)
        
      2. 시간복잡도 = θ(n4)
    • Running Time 향상시키기

      1. 모든 L(m) 매트릭스를 계산할 필요 없음
      2. 만약 negative-weight cycle이 없다면, L(m) = L(n-1) for all m >= n-1이 성립
      3. L(n-1) 계산은 sequence 계산을 통해 계산할 수 있다.
    • Faster-APSP(W,n)

      1. 의사코드
            L(1) = W
            m = 1
            while m < n-1
                do L(2m) = EXTEND(L(m), L(m), n)
                   m = 2m
            return L(m)
        
      2. 시간복잡도 : θ(n3lg n)
  • Floyd-Warshall 알고리즘

    • 입력

      1. 방향, weighted 그래프 G=(V,E)
      2. Negative-weight 간선이 존재할 수 있다.
      3. Negative-weight cycle이 그래프에 존재하지 않는다.
    • 계산

      1. 그래프의 모든 노드에 대한 최단경로
  • 최단 경로 구조

    • 그래프 G의 노드들 V = {1, 2,..., n}
    • 경로 p = <v1, v2, ..., vl>
    • p의 intermediate 노드는 {v2, v3, ..., v1-1} 중의 어떤 노드이다. -i에서 j로 가는 모든 경로를 고려하고, 이 경로의 intermediate vertices 들은 i부터 n까지 존재하고, 1부터 k까지의 부분 집합이 존재한다. 이때, 이 경로들에 존재하는 최소 비용 경로를 찾는다. "No vertex on these paths has index > k"
    1. 1부터 k까지 다 고려했는데 intermediate가 아닐 경우
      intermediate vertices from {1, 2, …, k}를 갖는 i에서 j로 가는 최단 경로는, intermediate vertices from {1, 2, …, k - 1}를 갖는 i에서 j로 가는 최단 경로이다.

    2. 1부터 k까지 다 고려했는데 intermediate가 맞을 경우

      1. k는 p1, p2의 intermediate가 아니다.
      2. p1은 i부터 k로 향하는(1,2,...,k-1) 최단 경로이다
      3. p2는 k부터 j로 향하는(1,2,...,k-1) 최단 경로이다.
    • Recursive Soulution
      1. dij = intermediary 노드{1, 2,..., k}를 갖는 i부터 j로 가는 최단 경로의 weight
      2. k = 0

        dij(k) = wij

      3. k >= 1

        [1] k는 경로 p의 intermediate 노드가 아닌경우 dij(k) = dij(k-1) [2] k가 경로 p의 intermediate 노드인 경우 dij(k) = dik(k-1) + dkj(k-1)

      4. 최종 결과

        dij(n) = δ(i, j) ∀i, j ∈ V

  • FLOYD-WARSHALL(W)

    1. 의사코드
          n = rows[W]
          D(0) = W
          for k=1 to n
              do for i=1 to n
                  do for j=1 to n
                      do d_ij(k) = min(d_ij(k-1), d_ik(k-1) + d_kj(k-1))
          return D(n)            
      
    2. 시간복잡도 θ(n3)
  • Constructing a Shortest Path

    1. πij(0) = Nil i=j or wij = ∞ 1 i!=j and wij != ∞
    2. πij(k) = πij(k-1) dij(k-1) <= dik(k-1) + dkj(k-1) = πkj(k-1) otherwise

About

알고리즘 공부

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published