100개의 계단이 있으며 나는 한 번에 10개의 계단을 오를 수 있는 능력을 가지고 있다.
이 때, 100개의 계단을 올라갈 수 있는 경우의 수는 몇 개인가?
2. 해결 코드
문제를 보고 생각할 수 있는 가장 초기의 코드를 작성한 것이다.
stairs = 100
skill = 10
table = [0 for i in range(stairs+1)]
table[0] = 1
for i in range(1, stairs+1):
s=0
for j in range(1,skill+1):
if i-j<0:
t=0
else:
t= table[i-j]
s = s+t
table[i] = s
print (table[stairs])
2. 초기 코드를 확인 후 줄일 수 있는 부분을 확인 후, 파이썬의 강점을 살려 개선 코드를 작성한 것이다
stairs = 100
skill = 10
table = [0 for i in range(stairs+1)]
table[0] = 1
for i in range(1, stairs+1):
s=0
for j in range(1,skill+1):
t=(table[i-j],0)[i-j<0]
s = s+t
table[i] = s
print (table[stairs])
3. 개선 코드를 작성 후 하나의 함수로 만듦으로써 n 계단 오르기를 구현한 것이다.
def climbing(staris, skill):
table = [0 for i in range(stairs + 1)]
table[0] = 1
for i in range(1, stairs+1):
s=0
for j in range(1,skill+1):
t=(table[i-j],0)[i-j<0]
s = s+t
table[i] = s
return table[stairs]
r = (climbing(100,10))
print (r)
4. 함수의 최적화와 코드의 최적화를 구현한 것이며 아래의 코드는 계단 오르기 문제에 있어 가장 최적화 된 코드이다.
def climbing(n,m):
table=[0 for i in range(n+1)]
table[0]=1
for i in range(1,n+1):
s=(i-m,0)[(i-m)<0]
table[i]=sum(table[s:i])
return table[n]
a=climbing(100,10)
print (a)
2.1 비방향 그래프를 보고 크루스칼 알고리즘을 이용하여 최소 비용 신장 트리를 구하는 과정을 단계별로 보이시오
크루스칼 알고리즘의 핵심
오름차순
가중치를 기준으로 간선을 오름차순 정렬
낮은 가중치의 간선부터 시작해서 하나씩 그래프에 추가
사이클을 형성하는 간선을 추가하지 않음
간선의 수가 정점의 수보다 하나 적을 때 MST가 완성
내림차순
가중치를 기준으로 간선을 내림차순으로 정렬
높은 가중치의 간선부터 시작해서 하나씩 그래프에서 제거
두 정점을 연결하는 다른 경로가 없을 경우 해당 간선은 제거하지 않음
간선의 수가 정점의 수보다 하나 적을 때 MST가 완성
2.2 그래프를 보고 다익스트라 알고리즘을 이용하여 v6를 출발점으로하는 최단 거리를 구하는 과정을 단계별로 보이시오
간선의 가중치를 오름차순으로 정렬한다.
가중치가 낮은 순서대로 간선을 하나씩 그래프에 추가한다.
그래프 내에서 사이클을 형성하는 간선은 추가하지 않는다.
간선의 개수가 정점의 개수보다 '1' 작으면 끝낸다.
3. NP, NP-complete
3.1 NP, NP-hard, NP-complete의 정의
$P$ : 어떤 문제가 주어졌을 때 다항식으로 표현되어 polynomical time 즉, 다항 시간내에 해결 가능한 알고리즘을 의미하며 알고리즘의 복잡도가 $O(n^k)$로 표현되는 문제를 '$P$'라 한다.(복잡도 $O(n^k)$ 이하를 가지는 경우 같은 복잡도 내에 모든 해를 구한다.)
$NP$ : 어떤 문제가 주어졌을 때 다항식으로 표현될 수 있는지의 여부가 결정되지 않은 문제들을 'NP(Non-deterministic polynomial)'라 한다.
$NP-Complete$ : NP이면서 동시에 NP-Hard 에 속한다면 그 문제는 'NP-Complete'라 한다. (전수조사가 답)
$NP-Hard$ : 적어도 모든 NP 문제만큼은 어려운 문제들의 집합이며,만약P-NP 문제가P=NP로 풀린다면 P=NP=NP-Complete이므로 P와 NP는 NP-Hard의 부분집합이 되고, P≠NP인 경우는 P와 NP-Hard는서로소가 된다.*다항시간 : $T(n) = O(n^k), k$ 는 상수 => $k$가 상수로 고정이 된다면 그 문제는 해결하기 쉬운 문제. 다항시간안에 풀 수 있는 문제이므로.
3.2 해밀턴회로 거리 구하기(그래프가 주어질 때 그래프의 모든 점을 정확하게 한번씩만 지나는 경로가 존재하는가?) (NP-hard)
4. 배열 제시 후 소팅 알고리즘을 사용하여 정렬 과정을 보여라
4.1 Merge sort 알고리즘을 사용하여 8개의 숫자를 정렬하는 과정을 보여라
4.2 각종 정렬 기법들의 장단점과 시간복잡도
5. Divide and Conquer 알고리즘과 Dynamic Programming 알고리즘 차이
5.1 분할 정복법이 비효율적인 경우를 설명하라
5.2 분할 정복법과 동적프로그래밍을 이용하여 이항계수를 구하는 알고리즘 작성 후 두방식의 차이점에 대해 적으시오
분할 정복법(Divide and Conquer)
어떠한 큰 문제를 해결하기 위해 큰 문제를 작은 단위의 문제로 나누어 해결하는 방법 (Top-down)
문제의 크기가 충분히 작아질때까지 반복적으로 분할
해결한 작은 문제를 가지고 바로 윗 단계의 문제를 해결해 나가는 방식
대표적인 예로는 Binary Search와 Merge Sort, Quick Sort 등이 있다.
동적 프로그래밍(Dynamic Programming)
문제를 작은 인스턴스들로 나누고 Bottom-up 순서로 풀어나가면서 문제의 값을 정해놓고 필요할 때 가져와 문제를 해결하는 방식이다. "기억하며 풀기"라 생각하면 된다.
대표적인 예로는 TSP 문제, 피보나치 수열, 파스칼의 트리 작성 등이 있다.
분할정복법과 동적 프로그래밍의 공통점과 차이점
공통점 : 문제를 작은 단위로 나누어 해결
차이점 : 분할정복법은 Top-down 방식이며, 동적 프로그래밍은 Bottom-up 방식이다.
5.3 Binomial coefficient를 계산하려할 때 다음을 구해라
Divide-and-Conquer 방법을 사용하는 pseudo code 작성
Dynamic Programming 방법을 사용하는 pseudo code 작성
위 두 접근 방법의 차이점을 서술
5.4 이항계수 수학적 귀납법으로 증명하고, 동적 프로그래밍으로 알고리즘을 구현하라.
bottom-up 방식
int binomial(int n, int k)
{
for (int i=0; i<=n; i++)
{
for (int j=0; j<=k && j<=i; j++)
{
if (k==0 || n==k)
arr[i][j] = 1;
else
arr[i][j] = arr[i-1][j-1] + arr[i-1][j];
}
}
return arr[n][k];
}
6. DFS(Depth First Search)와 BFS(Breadth First Search)의 차이
DFS, BFS의 의사코드를 작성하고 이들의 차이를 제시한 다음, 간단한 트리를 작성하여 순회하는 것을 보이시오.
8. 행렬을 주고 TSP에 관한 문제
9. Strassen 행렬 알고리즘을 사용하여 2*2 행렬 A,B의 곱 C를 구하라(m1,m2,...,m7)을 사용해야함
9.1 Strassen's Matrix Multiplication 알고리즘에 대해 설명하라
11. Floyd 알고리즘을 사용하여 W(=D0) 행렬과 D(=D) 행렬을 구하라. W, D를 구하는 과정을 서술하라
11.1 방향 그래프를 보고 플로이드 알고리즘을 이용하여 D와 P를 단계별로 적으시오.
12. 하노이의 탑 알고리즘을 써라
#include <stdio.h>
void HanoiTowerMove(int num, char from, char by, char to)
{
if (num == 1)
printf("원반 1을 %c에서 %c로 이동 \n", from, to);
else
{
HanoiTowerMove(num - 1, from, to, by);
printf("원반 %d를 %c에서 %c로 이동 \n", num, from, to);
HanoiTowerMove(num - 1, by, from, to);
}
}
int main(void)
{
// 막대 A의 원반 5개를 막대 B를 경유하여 막대 C로 옮기기
HanoiTowerMove(5, 'A', 'B', 'C');
return 0;
}
13. 탐욕적 방법 Greedy Algorithm이 항상 최적의 해를 내지 못하는 경우의 예를 들라
Greedy Algorithm이 사용되는 대표적인 예제로는 "최소 수의 동전으로 거스름돈 거슬러주기" 문제가 있다.
만약 우리가 마트의 캐셔로 일을 하고 있고, 850원을 거슬러 주어야 할 때 500원 동전 1개와 100원 동전 3개와 50원 동전 1개 총 4개의 동전을 건넴으로써 최소한의 동전을 거슬러 줄 수 있다. 하지만 이렇게 하지 않고 10원짜리 동전 85개를 건네주거나 50원짜리 17개를 건네주는 방법과 같이 다양한 경우의 수가 존재하며 이와 같을 경우 반드시 최적의 해를 내는 것은 아니다.
또한 우리나라의 경우 총 4개의 동전 500원, 100원, 50원, 10원이 있다. 하지만 만약 400원 동전이 발행된다면 400원 동전 2개, 50원 동전 1개를 통해 총 3개의 동전을 거슬러 줄 수 있을 것이다. 가장 최적의 방법이지만, 기존 알고리즘에 따르면 500원 1개, 100원 3개, 50원 1개로 거슬러주게 될 것이다. 이와 같이 탐욕 알고리즘은 항상 최적의 결과를 보장하지 못한다.
어떠한 큰 문제를 해결하기 위해 큰 문제를 작은 단위의 문제로 나누어 해결하는 방법 (Top-down)
문제의 크기가 충분히 작아질때까지 반복적으로 분할
해결한 작은 문제를 가지고 바로 윗 단계의 문제를 해결해 나가는 방식
대표적인 예로는 Binary Search와 Merge Sort, Quick Sort 등이 있다.
동적 프로그래밍(Dynamic Programming)
문제를 작은 인스턴스들로 나누고 Bottom-up 순서로 풀어나가면서 문제의 값을 정해놓고 필요할 때 가져와 문제를 해결하는 방식이다. "기억하며 풀기"라 생각하면 된다.
대표적인 예로는 TSP 문제, 피보나치 수열, 파스칼의 트리 작성 등이 있다.
분할정복법과 동적 프로그래밍의 공통점과 차이점
공통점 : 문제를 작은 단위로 나누어 해결
차이점 : 분할정복법은 Top-down 방식이며, 동적 프로그래밍은 Bottom-up 방식이다.
분할정복법의 장점, 단점, 특징
장점 : 문제를 나누어 해결한다는 특징상 병렬적으로 문제를 해결하는 데 큰 강점이 있다.
단점 :함수를 재귀적으로 호출함으로 인해 오버헤드가 발생한다. 또한 스택에 다양한 데이터를 보관하고 있어야 하므로 스택오버플로우가 발생한다.
특징 : 분할정복법은 보통 재귀함수를 이용하여 구현하는 것이 일반적이나, 재귀호출을 사용하지 않고 스택, 큐 등의 자료구조를 이용하여 구현하기도 한다.
이항 계수
이항계수는 분할 정복과 동적 프로그래밍에서 차이점을 서술하기 위해 자주 사용된다.
또한 이항계수란 $n$개의 원소에서 $r$개의 원소를 선택하는 방법의 수를 나타내며 재귀 프로그래밍의 기초가 된다. 이항계수는 수학에서 $_nC_r$로 나타낼 수 있으며, 이항계수 프로그래밍의 해결 방법은 $_nC_r =\ _{n-1}C_{r-1}\ +\ _{n-1}C_r$의 공식으로 이루어진다.
위 공식을 재귀적으로 구현할 경우 다음과 같다.
int binomial(int n, int k) {
if (n == k || k == 0)
return 1;
else
return binomial(n - 1, k) + binomial(n - 1, k - 1);
}
만약 binomical(5,2)를 계산하면 아래와 같이 동일한 문제들을 중복으로 풀게되어, 부분문제의 중복 문제가 발생한다. 즉, 알고리즘의 효율이 떨어지게 된다.
위와 같이 중복 계산으로 인해 효율이 떨어지는 것을 보완하기 위해서는, 하위 문제를 이용하여 최종 결과의 해를 도출하는 동적 프로그래밍 방법으로 접근하여 해결할 수 있다. 이전 계산 결과 값을 메모이제이션을 이용하여 풀면 다음과 같이 작성할 수 있다.
int binomial(int n, int k) {
if (n == k || k == 0)
return 1;
else if (arr[n][k] > -1) /* 배열 arr이 -1로 초기화되어 있다고 가정 */
return arr[n][k];
else {
arr[n][k] = binomial(n-1, k) + binomial(n-1, k-1);
return arr[n][k];
}
}
*메모이제이션(Memoization) : 이차원 배열에 인덱스에 해당하는 중복 조합을 저장 후, 만약 해당 인덱스에 값이 존재할경우(캐싱되었을 경우) 앞서 계산했다고 판단하여 해당 값을 반환함으로써 중복 계산을 줄여주는 기술. 즉, 동일 계산의 반복 수행을 제거한다.
Bottom-up 방식으로 해결할 경우 다음과 같이 작성할 수 있다.
int binomial(int n, int k) {
for (int i=0; i<=n; i++) {
for (int j=0; j<=k && j<=i; j++) {
if (k==0 || n==k)
arr[i][j] = 1;
else
arr[i][j] = arr[i-1][j-1] + arr[i-1][j];
}
}
return arr[n][k];
}