새로 생긴 놀이기구는 인기가 매우 많아 줄이 끊이질 않습니다. 이 놀이기구의 원래 이용료는 price원 인데, 놀이기구를 N 번 째 이용한다면 원래 이용료의 N배를 받기로 하였습니다. 즉, 처음 이용료가 100이었다면 2번째에는 200, 3번째에는 300으로 요금이 인상됩니다. 놀이기구를 count번 타게 되면 현재 자신이 가지고 있는 금액에서 얼마가 모자라는지를 return 하도록 solution 함수를 완성하세요. 단, 금액이 부족하지 않으면 0을 return 하세요.
나의 답
1 2 3 4 5 6 7 8 9 10 11
functionsolution(price, money, count) { let sum = 0;
for (let i = 1; i <= count; i++) { sum += price * i; }
다음과 같이 7단계의 순차적인 처리 과정을 통해 신규 유저가 입력한 아이디가 카카오 아이디 규칙에 맞는 지 검사하고 규칙에 맞지 않은 경우 규칙에 맞는 새로운 아이디를 추천해 주려고 합니다. 신규 유저가 입력한 아이디가 new_id 라고 한다면,
1 2 3 4 5 6 7 8
1단계 new_id의 모든 대문자를 대응되는 소문자로 치환합니다. 2단계 new_id에서 알파벳 소문자, 숫자, 빼기(-), 밑줄(_), 마침표(.)를 제외한 모든 문자를 제거합니다. 3단계 new_id에서 마침표(.)가 2번 이상 연속된 부분을 하나의 마침표(.)로 치환합니다. 4단계 new_id에서 마침표(.)가 처음이나 끝에 위치한다면 제거합니다. 5단계 new_id가 빈 문자열이라면, new_id에 "a"를 대입합니다. 6단계 new_id의 길이가 16자 이상이면, new_id의 첫 15개의 문자를 제외한 나머지 문자들을 모두 제거합니다. 만약 제거 후 마침표(.)가 new_id의 끝에 위치한다면 끝에 위치한 마침표(.) 문자를 제거합니다. 7단계 new_id의 길이가 2자 이하라면, new_id의 마지막 문자를 new_id의 길이가 3이 될 때까지 반복해서 끝에 붙입니다.
1478 → “one4seveneight” 234567 → “23four5six7” 10203 → “1zerotwozero3” 이렇게 숫자의 일부 자릿수가 영단어로 바뀌어졌거나, 혹은 바뀌지 않고 그대로인 문자열 s가 매개변수로 주어집니다. s가 의미하는 원래 숫자를 return 하도록 solution 함수를 완성해주세요.
로또 6/45(이하 ‘로또’로 표기)는 1부터 45까지의 숫자 중 6개를 찍어서 맞히는 대표적인 복권입니다. 아래는 로또의 순위를 정하는 방식입니다. 1
순위 당첨 내용 1 6개 번호가 모두 일치 2 5개 번호가 일치 3 4개 번호가 일치 4 3개 번호가 일치 5 2개 번호가 일치 6(낙첨) 그 외 로또를 구매한 민우는 당첨 번호 발표일을 학수고대하고 있었습니다. 하지만, 민우의 동생이 로또에 낙서를 하여, 일부 번호를 알아볼 수 없게 되었습니다. 당첨 번호 발표 후, 민우는 자신이 구매했던 로또로 당첨이 가능했던 최고 순위와 최저 순위를 알아보고 싶어 졌습니다. 알아볼 수 없는 번호를 0으로 표기하기로 하고, 민우가 구매한 로또 번호를 담은 배열 lottos, 당첨 번호를 담은 배열 win_nums가 매개변수로 주어집니다. 이때, 당첨 가능한 최고 순위와 최저 순위를 차례대로 배열에 담아서 return 하도록 solution 함수를 완성해주세요.
let minCount = lottos.filter((v) => win_nums.includes(v)).length; let zeroCount = lottos.filter((v) => !v).length;
const maxCount = minCount + zeroCount;
return [rank[maxCount], rank[minCount]]; }
어렵게 푼 기분이 틀리지 않았다. 조건문이 필요가 없다. 다만 rank의 경우의 수가 많아지면 if가 나을지도..??
폰켓몬
문제 설명
당신은 폰켓몬을 잡기 위한 오랜 여행 끝에, 홍 박사님의 연구실에 도착했습니다. 홍 박사님은 당신에게 자신의 연구실에 있는 총 N 마리의 폰켓몬 중에서 N/2마리를 가져가도 좋다고 했습니다. 홍 박사님 연구실의 폰켓몬은 종류에 따라 번호를 붙여 구분합니다. 따라서 같은 종류의 폰켓몬은 같은 번호를 가지고 있습니다. 예를 들어 연구실에 총 4마리의 폰켓몬이 있고, 각 폰켓몬의 종류 번호가 [3번, 1번, 2번, 3번]이라면 이는 3번 폰켓몬 두 마리, 1번 폰켓몬 한 마리, 2번 폰켓몬 한 마리가 있음을 나타냅니다. 이때, 4마리의 폰켓몬 중 2마리를 고르는 방법은 다음과 같이 6가지가 있습니다.
첫 번째(3번), 두 번째(1번) 폰켓몬을 선택
첫 번째(3번), 세 번째(2번) 폰켓몬을 선택
첫 번째(3번), 네 번째(3번) 폰켓몬을 선택
두 번째(1번), 세 번째(2번) 폰켓몬을 선택
두 번째(1번), 네 번째(3번) 폰켓몬을 선택
세 번째(2번), 네 번째(3번) 폰켓몬을 선택
이때, 첫 번째(3번) 폰켓몬과 네 번째(3번) 폰켓몬을 선택하는 방법은 한 종류(3번 폰켓몬 두 마리)의 폰켓몬만 가질 수 있지만, 다른 방법들은 모두 두 종류의 폰켓몬을 가질 수 있습니다. 따라서 위 예시에서 가질 수 있는 폰켓몬 종류 수의 최댓값은 2가 됩니다. 당신은 최대한 다양한 종류의 폰켓몬을 가지길 원하기 때문에, 최대한 많은 종류의 폰켓몬을 포함해서 N/2마리를 선택하려 합니다. N마리 폰켓몬의 종류 번호가 담긴 배열 nums가 매개변수로 주어질 때, N/2마리의 폰켓몬을 선택하는 방법 중, 가장 많은 종류의 폰켓몬을 선택하는 방법을 찾아, 그때의 폰켓몬 종류 번호의 개수를 return 하도록 solution 함수를 완성해주세요.
제한사항
nums는 폰켓몬의 종류 번호가 담긴 1차원 배열입니다.
nums의 길이(N)는 1 이상 10,000 이하의 자연수이며, 항상 짝수로 주어집니다.
폰켓몬의 종류 번호는 1 이상 200,000 이하의 자연수로 나타냅니다.
가장 많은 종류의 폰켓몬을 선택하는 방법이 여러 가지인 경우에도, 선택할 수 있는 폰켓몬 종류 개수의 최댓값 하나만 return 하면 됩니다.
filter()로 순회하던 과정을 제거하고 for문을 통해 소수의 배수(j += i)로 순회하며 값 0을 할당한 뒤, 마지막에 filter()로 모든 0을 삭제한다.
위 세가지 경우에 대해 성능을 검사해보니, 3번이 다른 두 경우보다 확실히 유의미한 성능차를 보였다.
데이터가 커질 수록 자료구조를 순회하는 방법이 중요함을 다시금 느낀다.
소수 만들기
문제 설명
주어진 숫자 중 3개의 수를 더했을 때 소수가 되는 경우의 개수를 구하려고 합니다. 숫자들이 들어있는 배열 nums가 매개변수로 주어질 때, nums에 있는 숫자들 중 서로 다른 3개를 골라 더했을 때 소수가 되는 경우의 개수를 return 하도록 solution 함수를 완성해주세요.
제한사항
nums에 들어있는 숫자의 개수는 3개 이상 50개 이하입니다.
nums의 각 원소는 1 이상 1,000 이하의 자연수이며, 중복된 숫자가 들어있지 않습니다.
나의 답
1 2 3 4 5 6 7 8
const isPrime = (num) => { let cnt = 0; for (let i = 1; i < num; i++) { if (num % i === 0) cnt++; }
// 성능 우세 functionsolution(nums) { const odd = []; const even = [];
for (let i = 0; i < nums.length; i++) { if (nums[i] % 2 === 0) even.push(nums[i]); else odd.push(nums[i]); }
let cnt = 0;
for (let i = 0; i < even.length - 1; i++) { for (let j = i + 1; j < even.length; j++) { for (let k = 0; k < odd.length; k++) { if (isPrime(even[i] + even[j] + odd[k])) cnt++; } } }
for (let i = 0; i < odd.length - 2; i++) { for (let j = i + 1; j < odd.length - 1; j++) { for (let k = j + 1; k < odd.length; k++) { if (isPrime(odd[i] + odd[j] + odd[k])) cnt++; } } }
git API를 가져와서 렌더하는 페이지를 만들고 있다. 처음에는 바닐라로 시작했지만 오늘 typeScript로 바꿨다. template을 지정하지 않고 CRA를 생성했기 때문에 npm install --save typescript @types/node @types/react @types/react-dom @types/jest로 npm 설치 후 진행하였다.
typeScript Generic
tsconfig.json로 수정하고 기존 만든 바닐라 js를 타입 지정하며 ts로 변환 중에, 바닐라 때 만들었던 useFetchDate 커스텀 훅이 가장 말썽이었어서 기록한다.
// for문 사용 functionsolutionFor(d, budget) { let sum = 0; let cnt = 0;
d.sort((a, b) => a - b);
for (cnt = 0; cnt < d.length; cnt++) { if (budget - sum >= d[cnt]) { sum += d[cnt]; } elsereturn cnt; }
return cnt; }
// Array.prototype.reduce 사용 functionsolutionReduce(d, budget) { let cnt = 0; d.sort((a, b) => a - b).reduce((acc, cur) => { if (budget - acc >= cur) { cnt++; return acc + cur; } return acc; }, 0);
return cnt; }
// Array.prototype.filter 사용 functionsolutionFilter(d, budget) { return d .sort((a, b) => a - b) .filter((v) => (budget - v >= 0 ? ((budget -= v), true) : false)).length; }
두 수를 입력받아 두 수의 최대공약수와 최소공배수를 반환하는 함수, solution을 완성해 보세요. 배열의 맨 앞에 최대공약수, 그다음 최소공배수를 넣어 반환하면 됩니다. 예를 들어 두 수 3, 12의 최대공약수는 3, 최소공배수는 12이므로 solution(3, 12)는 [3, 12]를 반환해야 합니다.
functionsolution(n, m) { const GCF = n > m ? calGCF(n, m) : calGCF(m, n); const LCM = GCF * (n / GCF) * (m / GCF); // LCM = n * m / GCF와 동일하지만 보다 명확한 식을 위해
return [GCF, LCM]; }
console.log(solution(3, 12)); // [3, 12]
중등 수학까지 거슬러 올라갔었다…
수학적으로 접근해서 차근차근 풀면 어렵지 않은 문제.
calGCF에서 인수에 직접 접근하여 재할당하는게 마음에 썩 내키지는 않는다.
감탄한 타인의 답
1 2 3 4 5 6 7 8 9 10 11
functiongreatestCommonDivisor(a, b) { return b ? greatestCommonDivisor(b, a % b) : Math.abs(a); } functionleastCommonMultipleOfTwo(a, b) { return (a * b) / greatestCommonDivisor(a, b); } functiongcdlcm(a, b) { return [greatestCommonDivisor(a, b), leastCommonMultipleOfTwo(a, b)]; }
console.log(gcdlcm(3, 12));
1 2 3 4 5
functiongcdlcm(a, b) { var r; for (var ab = a * b; (r = a % b); a = b, b = r) {} return [b, ab / b]; }
문자열로 구성된 리스트 strings와, 정수 n이 주어졌을 때, 각 문자열의 인덱스 n번째 글자를 기준으로 오름차순 정렬하려 합니다. 예를 들어 strings가 [“sun”, “bed”, “car”]이고 n이 1이면 각 단어의 인덱스 1의 문자 “u”, “e”, “a”로 strings를 정렬합니다.
제한사항
strings는 길이 1 이상, 50이하인 배열입니다.
strings의 원소는 소문자 알파벳으로 이루어져 있습니다.
strings의 원소는 길이 1 이상, 100이하인 문자열입니다.
모든 strings의 원소의 길이는 n보다 큽니다.
인덱스 1의 문자가 같은 문자열이 여럿 일 경우, 사전순으로 앞선 문자열이 앞쪽에 위치합니다.
functionsolution(strings, n) { return strings.sort((a, b) => { if (a[n] < b[n]) { return-1; } if (a[n] > b[n]) { return1; } if (a < b) { return-1; } if (a > b) { return1; }
점심시간에 도둑이 들어, 일부 학생이 체육복을 도난당했습니다. 다행히 여벌 체육복이 있는 학생이 이들에게 체육복을 빌려주려 합니다. 학생들의 번호는 체격 순으로 매겨져 있어, 바로 앞번호의 학생이나 바로 뒷번호의 학생에게만 체육복을 빌려줄 수 있습니다. 예를 들어, 4번 학생은 3번 학생이나 5번 학생에게만 체육복을 빌려줄 수 있습니다. 체육복이 없으면 수업을 들을 수 없기 때문에 체육복을 적절히 빌려 최대한 많은 학생이 체육수업을 들어야 합니다.
전체 학생의 수 n, 체육복을 도난당한 학생들의 번호가 담긴 배열 lost, 여벌의 체육복을 가져온 학생들의 번호가 담긴 배열 reserve가 매개변수로 주어질 때, 체육수업을 들을 수 있는 학생의 최댓값을 return 하도록 solution 함수를 작성해주세요.
제한사항
전체 학생의 수는 2명 이상 30명 이하입니다.
체육복을 도난당한 학생의 수는 1명 이상 n명 이하이고 중복되는 번호는 없습니다.
여벌의 체육복을 가져온 학생의 수는 1명 이상 n명 이하이고 중복되는 번호는 없습니다.
여벌 체육복이 있는 학생만 다른 학생에게 체육복을 빌려줄 수 있습니다.
여벌 체육복을 가져온 학생이 체육복을 도난당했을 수 있습니다. 이때 이 학생은 체육복을 하나만 도난당했다고 가정하며, 남은 체육복이 하나이기에 다른 학생에게는 체육복을 빌려줄 수 없습니다.
어떤 문장의 각 알파벳을 일정한 거리만큼 밀어서 다른 알파벳으로 바꾸는 암호화 방식을 시저 암호라고 합니다. 예를 들어 “AB”는 1만큼 밀면 “BC”가 되고, 3만큼 밀면 “DE”가 됩니다. “z”는 1만큼 밀면 “a”가 됩니다. 문자열 s와 거리 n을 입력받아 s를 n만큼 민 암호문을 만드는 함수, solution을 완성해 보세요.
제한사항
공백은 아무리 밀어도 공백입니다.
s는 알파벳 소문자, 대문자, 공백으로만 이루어져 있습니다.
s의 길이는 8000이하입니다.
n은 1 이상, 25이하인 자연수입니다.
나의 답
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
functionsolution(s, n) { return s .split("") .map((v) => v.charCodeAt()) .map((v) => { if (v === 32) returnString.fromCharCode(32); if (v <= 90) { return v + n > 90 ? String.fromCharCode(v + n - 26) : String.fromCharCode(v + n); } return v + n > 122 ? String.fromCharCode(v + n - 26) : String.fromCharCode(v + n); }) .join(""); }