반응형
Notice
Recent Posts
Recent Comments
Link
«   2025/01   »
1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31
Archives
Today
Total
01-23 00:02
관리 메뉴

ImJay

[BOJ/Java] 12891. DNA 비밀번호 본문

알고리즘/BOJ - Java

[BOJ/Java] 12891. DNA 비밀번호

ImJay 2024. 2. 4. 16:54
반응형

[BOJ/Java] 12891. DNA 비밀번호

 

12891번: DNA 비밀번호

평소에 문자열을 가지고 노는 것을 좋아하는 민호는 DNA 문자열을 알게 되었다. DNA 문자열은 모든 문자열에 등장하는 문자가 {‘A’, ‘C’, ‘G’, ‘T’} 인 문자열을 말한다. 예를 들어 “ACKA”

www.acmicpc.net


풀이

package edu.ssafy.im.BOJ.S2.No12891;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.StringTokenizer;

public class Main {
    static int s, p, ans = 0; // DNA 문자열의 길이, 부분문자열의 길이, 비밀번호로 사용할 수 있는 종류의 수
    static int arr[]; // DNA 문자열을 저장하는 배열
    static int c[] = new int[4]; // 부분문자열에 포함되어야 할 각 문자의 최소 개수
    static int r[] = new int[4]; // 부분문자열의 각 문자의 개수를 저장하는 배열

    public static void main(String[] args) throws NumberFormatException, IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringBuilder sb = new StringBuilder();
        String input = br.readLine();
        StringTokenizer st = new StringTokenizer(input);
        s = Integer.parseInt(st.nextToken());
        p = Integer.parseInt(st.nextToken());

        input = br.readLine();
        arr = new int[s];
        for (int i = 0; i < s; i++) {
            char c = input.charAt(i);
            // 문자에 따라 숫자로 변환하여 배열에 저장
            switch (c) {
                case 'A':
                    arr[i] = 0;
                    break;
                case 'C':
                    arr[i] = 1;
                    break;
                case 'G':
                    arr[i] = 2;
                    break;
                case 'T':
                    arr[i] = 3;
                    break;
            }
        }

        input = br.readLine();
        st = new StringTokenizer(input);
        for (int i = 0; i < 4; i++) {
            // 각 문자의 최소 개수 입력 받기
            c[i] = Integer.parseInt(st.nextToken());
        }

        // 첫 번째 부분문자열에 대한 각 문자의 개수 계산
        for (int i = 0; i < p; i++) {
            r[arr[i]]++;
        }

        int ans = 0;
        int start = 0, end = 0 + p - 1; // 부분문자열의 시작과 끝 인덱스 설정
        // 모든 가능한 부분문자열에 대해 반복
        for (int i = 0; i < s - p + 1; i++) {
            // 부분문자열에 대해 각 문자의 개수가 최소 개수보다 큰지 확인
            for (int j = 0; j < 4; j++) {
                if (r[j] < c[j]) {
                    break;
                } else if (j == 3) {
                    ans++; // 모든 문자의 개수가 최소 개수 이상인 경우, 비밀번호로 사용 가능한 경우임을 카운트
                }
            }
            try {
                // 다음 부분문자열의 시작과 끝 인덱스를 설정하고 각 문자의 개수 업데이트
                r[arr[start++]]--;
                r[arr[++end]]++;
            } catch (Exception e) {
            }
        }
        sb.append(ans); // 결과 문자열에 비밀번호로 사용 가능한 부분문자열의 개수 추가
        System.out.println(sb); // 결과 출력
    }
}

 

반응형
Comments