본문 바로가기
코딩테스트

[Java] 암호

by codeok 2022. 6. 19.
반응형

설명

현수는 영희에게 알파벳 대문자로 구성된 비밀편지를 매일 컴퓨터를 이용해 보냅니다.

비밀편지는 현수와 영희가 서로 약속한 암호로 구성되어 있습니다.

비밀편지는 알파벳 한 문자마다 # 또는 *이 일곱 개로 구성되어 있습니다.

만약 현수가 “#*****#”으로 구성된 문자를 보냈다면 영희는 현수와 약속한 규칙대로 다음과 같이 해석합니다.

1. “#*****#”를 일곱자리의 이진수로 바꿉니다. #은 이진수의 1로, *이진수의 0으로 변환합니다. 결과는 “1000001”로 변환됩니다.

2. 바뀐 2진수를 10진수화 합니다. “1000001”을 10진수화 하면 65가 됩니다.

3. 아스키 번호가 65문자로 변환합니다. 즉 아스크번호 65는 대문자 'A'입니다.

참고로 대문자들의 아스키 번호는 'A'는 65번, ‘B'는 66번, ’C'는 67번 등 차례대로 1씩 증가하여 ‘Z'는 90번입니다.

현수가 4개의 문자를 다음과 같이 신호로 보냈다면

#****###**#####**#####**##**

이 신호를 4개의 문자신호로 구분하면

#****## --> 'C'

#**#### --> 'O'

#**#### --> 'O'

#**##** --> 'L'

최종적으로 “COOL"로 해석됩니다.

현수가 보낸 신호를 해석해주는 프로그램을 작성해서 영희를 도와주세요.

입력

첫 줄에는 보낸 문자의 개수(10을 넘지 안습니다)가 입력된다. 다음 줄에는 문자의 개수의 일곱 배 만큼의 #또는 * 신호가 입력됩니다.

현수는 항상 대문자로 해석할 수 있는 신호를 보낸다고 가정합니다.

출력

영희가 해석한 문자열을 출력합니다.

예시 입력 1 

4
#****###**#####**#####**##**

예시 출력 1

COOL

 

첫 번째 풀이

문제 접근은 다음과 같이 했다.

  1.  # -> 1, * -> 0으로 변경
    • replace()을 사용해서 "#"은 1로, "*"은 0으로
  2. n만큼 7개로 signal을 분리한다.
  3. Integer.parseInt(value, 2)로 이진수 -> 십진수 변환
  4. 십진수 -> 아스키 코드로 출력

반복되는 int 값은 BINARY_LENGTH 상수로 빼서 처리를 했다.

package section01.E12_암호;

import java.io.*;

public class Main {

    private static final int BINARY_LENGTH = 7;

    public String solution(int n, String signal){

        StringBuilder answer = new StringBuilder();

        signal = signal.replace("#", "1")
                       .replace("*", "0");

        int signalLength = signal.length();

        for (int i = 0; i < signalLength / BINARY_LENGTH; i++){
            int startIndex = i * BINARY_LENGTH;
            int endIndex = BINARY_LENGTH + (BINARY_LENGTH * i);

            String binary = signal.substring(startIndex, endIndex);
            int decimal = Integer.parseInt(binary, 2);
            answer.append((char) decimal);
        }

        return answer.toString();
    }

    public static void main(String[] args) throws IOException {
        Main T = new Main();
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out));

        int n = Integer.parseInt(br.readLine());
        String signal = br.readLine();
        bw.write(T.solution(n, signal));

        bw.flush();
        bw.close();
    }
}

 

두 번째 방법 -  signal을 자를때 substring() 사용

다른 방법들은 위와 동일하다.

 

#, *가 1과 0으로 치환된  String signal = "1000011100111110011111001100"을 for문에서 substring()으로 문자열을 substring(0, 7) 7개씩 자르고 나서 다음 문자를 7번째 인덱스부터 끝까지 자르면서 갱신해줬다.

 

Ex) 1000011100111110011111001100

i singal.substring(0, BINARY_LENGTH) signal.substring(BINARY_LENGTH)
0 1000011 100111110011111001100
1 1001111 10011111001100
2 1001111 1001100
3 1001100  

 

package section01.E12_암호;

import java.io.*;

public class MainA {

    private static final int BINARY_LENGTH = 7;

    public String solution(int n, String signal){

        StringBuilder answer = new StringBuilder();

        signal = signal.replace("#", "1")
                       .replace("*", "0");

        for (int i = 0; i < n; i++){
            String binary = signal.substring(0, BINARY_LENGTH);
            int decimal = Integer.parseInt(binary, 2);
            answer.append((char) decimal);

            signal = signal.substring(BINARY_LENGTH);
        }

        return answer.toString();
    }
}
반응형

'코딩테스트' 카테고리의 다른 글

[Java] 가장 짧은 문자거리  (0) 2022.06.19
[Java] 대소문자 변환  (0) 2022.06.11