본문 바로가기
코딩테스트

[프로그래머스 Lv.1 / Java] 숫자 문자열과 영단어

by 지지 2022. 1. 27.

[문제]

숫자 문자열과 영단어

네오와 프로도가 숫자놀이를 하고 있습니다. 네오가 프로도에게 숫자를 건넬 때 일부 자릿수를 영단어로 바꾼 카드를 건네주면 프로도는 원래 숫자를 찾는 게임입니다.

다음은 숫자의 일부 자릿수를 영단어로 바꾸는 예시입니다.

1478 → "one4seveneight"

234567 → "23four5six7"

10203 → "1zerotwozero3"

이렇게 숫자의 일부 자릿수가 영단어로 바뀌어졌거나, 혹은 바뀌지 않고 그대로인 문자열 s가 매개변수로 주어집니다. s가 의미하는 원래 숫자를 return 하도록 solution 함수를 완성해주세요.

참고로 각 숫자에 대응되는 영단어는 다음 표와 같습니다.

숫자
영단어
0
zero
1
one
2
two
3
three
4
four
5
five
6
six
7
seven
8
eight
9
nine

제한사항

● 1 ≤ s의 길이 ≤ 50

s가 "zero" 또는 "0"으로 시작하는 경우는 주어지지 않습니다.

● return 값이 1 이상 2,000,000,000 이하의 정수가 되는 올바른 입력만 s로 주어집니다.

입출력 예

s
result
"one4seveneight"
1478
"23four5six7"
234567
"2three45sixseven"
234567
"123"
123

입출력 예 설명

입출력 예 #1

문제 예시와 같습니다.

입출력 예 #2

문제 예시와 같습니다.

입출력 예 #3

"three"는 3, "six"는 6, "seven"은 7에 대응되기 때문에 정답은 입출력 예 #2와 같은 234567이 됩니다.

입출력 예 #2와 #3과 같이 같은 정답을 가리키는 문자열이 여러 가지가 나올 수 있습니다.

입출력 예 #4

s에는 영단어로 바뀐 부분이 없습니다.

제한시간 안내

정확성 테스트 : 10초


[풀이]

1. for문을 돌려야할 것을 눈치 챘지만 머릿속으로 잘 그려지지 않아 일단 바꿔야 할 부분을 쭉 써내려가봤다.

2. replace 함수를 이용하여 문자열 영단어를 같은 타입의 숫자로 바꾸어주었고, 그렇게 완성된 숫자로만 구성된 문자열을 Integer.parseInt 함수를 사용해 int 타입으로 바꾸어 주었다.

3. 코드를 실행해보니 모든 테스트에 통과를 했다. 이제 이 코드를 더 효율적으로 바꿔주어야겠지?

class Solution {
    public int solution(String s) {
        int answer = 0;

        s = s.replace("zero", "0");
        s = s.replace("one", "1");
        s = s.replace("two", "2");
        s = s.replace("three", "3");
        s = s.replace("four", "4");
        s = s.replace("five", "5");
        s = s.replace("six", "6");
        s = s.replace("seven", "7");
        s = s.replace("eight", "8");
        s = s.replace("nine", "9");
        
        answer = Integer.parseInt(s);
        
        return answer;
    }
}

↑ 무작정 써내려간 코드

이렇게 쭉 써내려 나가니 어떤식으로 코드를 짜야하는지 머릿속으로 그려지기 시작했다.

↓ 무작정 써내려간 코드를 보고 고민하며 효율적으로 바꾼 코드

class Solution {
    public int solution(String s) {
        int answer = 0;
        String[] num = {"zero", "one", "two", "three", "four", "five", "six", "seven", "eight", "nine"};
        
        for(int i = 0; i<10; i++){
            s = s.replace(num[i], Integer.toString(i));
        }
 
        answer = Integer.parseInt(s);
        
        return answer;
    }
}

1. 일단 String 타입의 배열을 만들어 영단어를 배열에 담아주었다.

2. 그리고 for문을 돌려주었는데, 숫자가 0~9까지로 정해져 있기 때문에(num배열의 길이도 10) 10번 돌린다고 적었지만, 지금 생각해보니 배열의 길이만큼 num.length()로 돌려주는 것이 더 좋을 것 같다는 생각이 든다.

3. for문 안에는 s 문자열안에서 배열 num의 인덱스 번호에 해당하는 문자열을 찾아 replace함수를 통해 i로 바꿔주었다. 이 때, Integer.toString()함수를 사용해 숫자 i 를 String 타입으로 바꿔주었다.

- 이렇게 총 10번을 돌려 (zero ~ nine까지) s에 담긴 모든 문자열을 검사해 String타입의 숫자로 바꿔줄 수 있었다.

4. 그리고 마지막으로 출력할 땐 다시 Integer.parseInt()함수를 사용해 String타입의 숫자로 이루어진 문자열을 int 타입의 숫자로 바꿔주었다.


[느낀점]

코딩테스트 연습을 몇 번 해보진 않았지만 그래도 풀어봤던 문제들 중에서는 쉬운 문제라고 생각이 든다.

그러나 아무리 쉬운 문제라도 머릿속으로 그리지 못하고 for만 써놓고 쩔쩔매고 있다면, 바보같다는 생각이 들더라도 노가다로 쭉 코드를 써보는 수밖에 없다고 생각한다. (손코딩도 좋은 방법인 것 같다.)

바보 같은 코드로 테스트를 통과했다면, 그 코드를 어떻게 더 효율적으로 바꿀 수 있을까 고민해 보는 것이 성장에 많은 도움이 된다고 생각한다. replace함수를 남발한 후 더 나은 코드를 스스로 고민하는 과정에서 뿌듯함과 자신감을 얻었다. (물론 쉬운 문제였지만....)

풀 수 없을 것 같은 문제가 나와도 최소 1시간은 고민해보고 테스트에 불합격 하면서 또 고민을 해보는 시간이 정말 필요하다는 걸 느낀문제였다.

댓글