Language/Algorithm
[알고리즘 기초] 09_SW 문제해결 응용_2 / Python
migrationArc
2021. 6. 14. 21:36
[알고리즘 기초] 09_SW 문제해결 응용_2 / Python
1. 완전 검색 & 그리디
1.1 반복과 재귀
반복(Iteration)과 재귀(Recursion)
- 반복과 재귀는 유사한 작업을 수행할 수 있다.
- 반복은 수행하는 작업이 완료 될때까지 계속 반복
- 루프(for, while 구조)
- 재귀는 주어는 문제의 해를 구하기 위해 동일하면서 더 작은 문제의 해를 이용하는 하나의 방법
- 하나의 큰 문제를 해결할 수 잇는 (해결하기 쉬운) 더 작은 문제로 쪼개고 그 결과들을 결합한다.
- 재귀 함수로 구현
반복 구조
- 초기화
- 반복되는 명령문을 실행하기 전에 조건 검사에 사용할 변수의 초기값 설정
- 조건검사
- 반복할 명령문 실행
- 업데이트
- 무한루프가 되지 않게 조건이 거짓이 되게 한다
재귀적 알고리즘
- 재귀적 정의는 두 부분으로 나뉜다.
- 하나 또는 그 이상의 기본 경우
- 집합에 포함되어 있는 원소로 induction을 생성하기 위한 seed 역할
- 하나 또는 그 이상의 유도된 경우
- 새로운 집합의 원소를 생성하기 위해 결합되어지는 방법
재귀 함수
- 함수 내부에서 직접 혹은 간접적으로 자기자신을 호출하는 함수
- 일반적으로 재귀적 정의를 이용해서 재귀 함수를 구현한다.
- 따라서, 기본부분과 유도파트로 구성된다.
- 재귀적 프로그램을 작성하는 것은 반복 구조에 비해 간결하고 이해하기 쉽다.
- 함수 호출은 프로그램 메모리 구조에서 스택을 사용한다. 따라서 재귀 호출은 반복적인 스택의 사용을 의미하며 메모리 및 속도에서 성능저하가 발생한다.
반복 또는 재귀
- 해결할 문제를 고려해서 반복이나 재귀의 방법을 선택
- 재귀는 문제 해결을 위한 알고리즘 설계가 간단하고 자연스럽다.
- 추상 자료형의 알고리즘은 재귀적 구현이 가단하고 자연스러운 경우가 많다.
- 일반적으로, 재귀적 알고리즘은 반복 알고리즘보다 더 많은 메모리와 연산을 필요로 한다.
- 입력값 n이 커질수록 재귀 알고리즘은 반복에 비해 비효율적일 수 있다.
1.2 완전검색 기법
완전검색으로 시작하라
- 모든 겨웅의 수를 생성하고 테스트하기 때문에 수행속도는 느리지만. 해답을 찾아내지 못할 확률이 작다.
- 완적 검색은 입력의 크기를 작게 해서 간편하고 빠르게 답을 구하는 프로그램을 작성한다.
- 이를 기반으로 그리디 기법이나 동적 계획법을 이용해서 효율적인 알고리즘을 찾을 수 있다.
- 검정 등에서 주어진 문제를 풀 때, 우선 완전 검색으로 접근하여 해답을 도출한 후, 성능 개선을 위해 다른 알고리즘을 사용하고 해답을 확인하는 것이 바람직하다.
1.3 조합적 문제
순열
- 서로 다른 것들 중 몇개를 뽑아서 한줄로 나열하는 것
- 서로 다른 n개중 r개를 택하는 순열은 아래와 같이 표현한다.
- nPr
- 그리고 nPr은 다음과 같은 식이 성립한다.nPr = n * (n-1) * (n-2) * ... * (n-r+1)
- nPn = n!이라고 표기하며 factorial이라 부른다.
- 다수의 알고리즘 문제들은 순서화된 요소들의 집합에서 최선의 방법을 찾는 것과 관련있다.
- 예) TSP
- N 개의 요소들에 대해서 n! 개의 순열들이 존재한다.
- 12! = 479,001,600
- n > 12인 경우, 시간 복잡도 폭발적으로 up!!
순열 생성 방법
- 사전적 순서
- {1, 2, 3}, n = 3 인 경우 다음과 같이 생성된다.
- [1 2 3] [1 3 2] [2 1 3] [2 3 1] [3 1 2] [3 2 1]
- 최소 변경을 통한 방법
- 각각의 순열들은 이전의 상태에서 단지 두 개의 요소들 교환을 통해 생성
- ['1' 2 '3'] -> ['3' '2' 1] -> [2 '3' '1'] -> ['2' '1' 3] -> ['3' '1' 2] -> [1 3 2]
- 재귀 호출을 통한 순열 생성
부분 집합
- 집합에 포함된 원소들을 선택하는 것이다.
- 다수의 중요 알고리즘들이 원소들의 그륩에서 최적의 부분 집합을 찾는 것이다.
- ex) 배낭 짐 싸기
- N개의 원소를 포함한 집합
- 자기 자신과 고집합 포함한 모든 부분집합의 개수는 2^n개
- 원소의 수가 증가하면 부분집합의 개수는 지수적으로 증가
부분집합 생성방법
- 바이너리 카운팅을 통한 사전적 순서
- 부분집합을 생성하기 위한 가장 자연스러운 방법이다.
- 바이너리 카운팅은 사전적 순서로 생성하기 위한 가장 간단한 방법이다.