[프로그래머스] 숫자 문자열과 영단어 | JavaScript

예구·2023년 7월 13일
0

Algorithm

목록 보기
10/47
post-thumbnail

문제출처

1. 문제

문제 설명

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

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

  • 1478 → "one4seveneight"
  • 234567 → "23four5six7"
  • 10203 → "1zerotwozero3"

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

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

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

제한사항

  • 1 ≤ s의 길이 ≤ 50
  • s가 "zero" 또는 "0"으로 시작하는 경우는 주어지지 않습니다.
  • return 값이 1 이상 2,000,000,000 이하의 정수가 되는 올바른 입력만 s로 주어집니다.

입출력 예

sresult
"one4seveneight"1478
"23four5six7"234567
"2three45sixseven"234567
"123"123

입출력 예 설명

입출력 예 #1

  • 문제 예시와 같습니다.

입출력 예 #2

  • 문제 예시와 같습니다.

입출력 예 #3

  • "three"는 3, "six"는 6, "seven"은 7에 대응되기 때문에 정답은 입출력 예 #2와 같은 234567이 됩니다.
  • 입출력 예 #2와 #3과 같이 같은 정답을 가리키는 문자열이 여러 가지가 나올 수 있습니다.

입출력 예 #4

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



2. 풀이

영단어 배열인 num_wordsfor문을 통해 돌면서 주어진 s에 해당되는 게 있다면 num_words의 인덱스 값으로 바꿔줬다.

여기서 주의할 점은 replaceAll은 원래 문자열을 변경하는 것이 아니라 새로운 문자열을 반환한다는 점이다.
처음에 let res = res.replaceAll(num_words[i], i)처럼 res에 재할당하지 않고 res.replaceAll(num_words[i], i)로 작성했다가 문자열이 변경되지 않아서 당황했다.

function solution(s) {
  let num_words = [
    "zero",
    "one",
    "two",
    "three",
    "four",
    "five",
    "six",
    "seven",
    "eight",
    "nine",
  ];

  let res = s;
  for (let i = 0; i < num_words.length; i++) {
    if (Number(res)) break;  // 영단어로 바뀐 부분이 없다면 break

    if (res.match(num_words[i])) {
      res = res.replaceAll(num_words[i], i);
    }
  }

  return Number(res);
}



3. 공부한 내용

match()

  • 특정 텍스트 안에 검색할 단어, 찾고 싶은 단어가 있는 경우 해당 텍스트가 문구에 포함되어 있는지 확인
  • 단어뿐만 아니라 정규표현식을 사용하여 특정 패턴을 검색하는 것 역시 가능
str.match(regexp)
  • regexp

    • 정규식 개체
    • egExp가 아닌 객체 obj가 전달되면, new RegExp(obj)를 사용하여 암묵적으로 RegExp로 변환
    • 매개변수를 전달하지 않고 match()를 사용하면, 빈 문자열:[""]이 있는 Array가 반환
  • 결과 값

    • 문자열이 정규식과 일치하면, 일치하는 전체 문자열을 첫 번째 요소로 포함하는 Array를 반환한 다음 괄호 안에 캡처된 결과가 옴
    • 일치하는 것이 없으면 null 반환
const str = "red is impressive."
str.match("red");  // red가 있으므로 red 출력

const test  = 'love you. love me. love everything!'
const regExp = /love/gi;
test2 = test.match(regExp);
// ['love', 'love', 'love']
// test2변수에 배열로 모든 love 텍스트가 저장됨

replaceAll()

  • pattern의 모든 일치 항목이 replacement로 대체된 새 문자열 반환
  • pattern은 문자열 또는 RexExp일 수 있으며 replacement는 각 일치 항목에 대해 호출되는 문자열 또는 함수일 수 있음
  • 원래 문자열은 변경되지 않음
replaceAll(pattern, replacement)
const p = 'The quick brown fox jumps over the lazy dog. If the dog reacted, was it really lazy?';

console.log(p.replaceAll('dog', 'monkey'));
// The quick brown fox jumps over the lazy monkey. If the monkey reacted, was it really lazy?

const regex = /Dog/ig;
console.log(p.replaceAll(regex, 'ferret'));
// The quick brown fox jumps over the lazy ferret. If the ferret reacted, was it really lazy?



4. 다른 사람의 풀이

나와 접근은 비슷했지만 split으로 해당하는 문자가 있다면 그 문자를 기준으로 문자열을 나눈 후, join으로 그 사이를 해당하는 숫자로 변경하는 방법이다.

splitjoin을 이렇게 사용하는 건 처음봐서 오늘 또 하나 배웠다.

function solution(s) {
    let numbers = ["zero", "one", "two", "three", "four", "five", "six", "seven", "eight", "nine"];
    var answer = s;

    for(let i=0; i< numbers.length; i++) {
        let arr = answer.split(numbers[i]);
        answer = arr.join(i);
    }

    return Number(answer);
}
``
profile
우당탕탕 FE 성장기

0개의 댓글

관련 채용 정보