새소식

PS/백준

[백준/2306] 유전자 (Python)

  • -

Problem : https://www.acmicpc.net/problem/2306

 

2306번: 유전자

DNA 서열은 4개의 문자 {a,c,g,t} 로 이루어진 문자열이다. DNA 서열에는 생명의 신비를 풀 수 있는 많은 정보가 들어 있다. 특히 KOI 유전자의 길이는 사람의 키와 깊은 상관 관계가 있다는 것이 알려

www.acmicpc.net

 

Difficulty : Gold 3

 

Status : Solved

 

Time : 00:27:53

 


 

문제 설명

 

더보기

DNA 서열은 4개의 문자 {a,c,g,t} 로 이루어진 문자열이다. DNA 서열에는 생명의 신비를 풀 수 있는 많은 정보가 들어 있다. 특히 KOI 유전자의 길이는 사람의 키와 깊은 상관 관계가 있다는 것이 알려져 있다. 이러한 KOI 유전자는 다음의 조건을 만족한다.

 

  • at 와 gc 는 가장 짧은 길이의 KOI 유전자이다.
  • 어떤 X가 KOI 유전자라면, aXt와 gXc도 KOI 유전자이다. 예를 들어, agct 와 gaattc는 KOI 유전자이나, tgca 와 cgattc는 KOI 유전자가 아니다.
  • 어떤 X와 Y가 KOI 유전자라면, 이 둘을 연결한 XY도 KOI 유전자이다. 예를 들면, aattgc 또는 atat는 KOI 유전자이나 atcg 또는 tata는 KOI 유전자가 아니다.


KOI 유전자는 DNA 서열 중에서 부분 서열로 구성되어 있다. 부분 서열이란 주어진 서열에서 임의의 위치에 있는 0개 이상의 문자들을 삭제해서 얻어지는 서열이다. 예를 들면, DNA 서열 acattgatcg에서 두 번째 문자 c와 마지막 문자 g를 삭제하여 생긴 부분 서열 aattgatc는 길이가 8인 KOI 유전자이다. 그러나 마지막 문자 g를 삭제하여 만들어진 부분 서열 acattgatc는 KOI 유전자가 아니다.

문제는 주어진 DNA 서열의 부분 서열들 중에서 길이가 최대가 되는 KOI 유전자를 찾아 그 길이를 출력하는 것이다.

 

입력 및 출력

 

더보기

입력

 

첫째 줄에는 분석하고자 하는 DNA 서열이 주어진다. DNA 서열의 길이는 최대 500이다.

 

출력

 

입력 DNA 서열로부터 계산된 가장 긴 KOI 유전자의 길이를 첫 번째 줄에 출력한다. 단, KOI 유전자가 없는 경우에는 0을 출력한다.

 

입력 예시

 

acattgagtc

 

출력 예시

 

8

 

 


 

풀이

 

DP 문제. j부터 j+i까지의 최대 KOI DNA 문자열 길이 DP[j][j+i]가 있다고 할 때, 다음이 성립한다.

 

  • DNA[j] 와 DNA[j+i]가 KOI DNA 조건에 맞을 경우, 즉 at 혹은 gc일 경우에는 DP[j][j+i] = max(DP[j][j+i], DP[j+1][j+i-1] + 2)가 성립한다. 즉 가장 앞쪽 문자열과 가장 뒤쪽 문자열을 제한 최장 문자열 + 2가 성립한다.
  • j <= k < j+i 를 만족하는 k에 대해, k를 기준으로 문자열을 둘로 나눌 수 있다. 이 부분 문자열에서의 KOI DNA 문자열 길이합으로 업데이트할 수 있다. 즉 DP[j][j+i] = max(DP[j][j+i], DP[j][k] + DP[k+1][j+i])가 성립한다.

O(N^3)의 시간복잡도를 보이나 N이 최대 500이므로 시간 내에 해결할 수 있다.

 

풀이 코드

DNA = input().strip()
l = len(DNA)
dp = [[0]*l for _ in range(l)]

for i in range(1, l) :
  for j in range(l-i) :
    if (DNA[j] == 'a' and DNA[j+i] == 't') or (DNA[j] == 'g' and DNA[j+i] == 'c') :
      dp[j][j+i] = dp[j+1][j+i-1] + 2
    for k in range(j, j+i) :
      if dp[j][j+i] < dp[j][k] + dp[k+1][j+i] :
        dp[j][j+i] = dp[j][k] + dp[k+1][j+i]

print(dp[0][-1])

풀이 완료!

Contents

포스팅 주소를 복사했습니다

이 글이 도움이 되었다면 공감 부탁드립니다.