배열, 정렬, 문자열
문제 설명
정수 n을 기준으로 n과 가까운 수부터 정렬하려고 합니다. 이때 n으로부터의 거리가 같다면 더 큰 수를 앞에 오도록 배치합니다. 정수가 담긴 배열 numlist와 정수 n이 주어질 때 numlist의 원소를 n으로부터 가까운 순서대로 정렬한 배열을 return하도록 solution 함수를 완성해주세요.
function solution(numlist, n) {
return numlist.sort((a, b) => {
const [aGab, bGab] = [Math.abs(a - n), Math.abs(b - n)]
return aGab === bGab ? b - a : aGab - bGab;
})
}
문제 설명
영어 점수와 수학 점수의 평균 점수를 기준으로 학생들의 등수를 매기려고 합니다. 영어 점수와 수학 점수를 담은 2차원 정수 배열 score가 주어질 때, 영어 점수와 수학 점수의 평균을 기준으로 매긴 등수를 담은 배열을 return하도록 solution 함수를 완성해주세요.
function solution(score) {
let avg = score.map(([a, b]) =>(a + b) / 2);
let sorted = avg.slice().sort((a, b) => b - a);
return avg.map(v => sorted.indexOf(v) + 1);
}
// 다른 사람의 풀이
function solution(score) {
return score.map((el) => {
return (
score.filter((v) => (v[0] + v[1]) / 2 > (el[0] + el[1]) / 2).length + 1
);
});
}
문제 설명
머쓱이는 태어난 지 6개월 된 조카를 돌보고 있습니다. 조카는 아직 "aya", "ye", "woo", "ma" 네 가지 발음을 최대 한 번씩 사용해 조합한(이어 붙인) 발음밖에 하지 못합니다. 문자열 배열 babbling이 매개변수로 주어질 때, 머쓱이의 조카가 발음할 수 있는 단어의 개수를 return하도록 solution 함수를 완성해주세요.
function solution(babbling) {
var answer = 0;
let arr = ["aya", "ye", "woo", "ma"];
for (let i in babbling) {
let word = babbling[i];
for (let j in arr) {
word = word.replaceAll(arr[j], ' ');
console.log(word)
}
if(word.trim().length === 0) {
console.log(word.trim())
answer++;
}
}
return answer;
}
// 다른 사람의 풀이
function solution(babbling) {
var answer = 0;
const regex = /^(aya|ye|woo|ma)+$/;
babbling.forEach(word => {
if (regex.test(word)) answer++;
})
return answer;
}
문자열 관련 문제는 정규식을 잘하면 수월할 것 같음,,
문제 설명
머쓱이는 프로그래머스에 로그인하려고 합니다. 머쓱이가 입력한 아이디와 패스워드가 담긴 배열 id_pw와 회원들의 정보가 담긴 2차원 배열 db가 주어질 때, 다음과 같이 로그인 성공, 실패에 따른 메시지를 return하도록 solution 함수를 완성해주세요.
아이디와 비밀번호가 모두 일치하는 회원정보가 있으면 "login"을 return합니다.
로그인이 실패했을 때 아이디가 일치하는 회원이 없다면 “fail”를, 아이디는 일치하지만 비밀번호가 일치하는 회원이 없다면 “wrong pw”를 return 합니다.
// 처음부터 answer을 'fail'로 잡으면 if 문을 하나 없앨 수 있을 텐데!
function solution(id_pw, db) {
const [id, pw] = id_pw;
let answer = ''
for(let [dbId, dbPw] of db) {
if(dbId === id && dbPw === pw) answer = 'login'
if(dbId === id && dbPw !== pw) answer = 'wrong pw'
if(dbId !== id && dbPw !== pw) answer = 'fail'
}
return answer;
}
// 다른 사람의 풀이
function solution(id_pw, db) {
const [id, pw] = id_pw;
const map = new Map(db);
return map.has(id) ? (map.get(id) === pw ? 'login' : 'wrong pw') : 'fail';
}
브라우저 빌트인 DOM을 사용한 명령형 UI 프로그래밍 예시
async function handleFormSubmit(e) {
e.preventDefault();
disable(textarea);
disable(button);
show(loadingMessage);
hide(errorMessage);
try {
await submitForm(textarea.value);
show(successMessage);
hide(form);
} catch (err) {
show(errorMessage);
errorMessage.textContent = err.message;
} finally {
hide(loadingMessage);
enable(textarea);
enable(button);
}
}
function handleTextareaChange() {
if (textarea.value.length === 0) {
disable(button);
} else {
enable(button);
}
}
function hide(el) {
el.style.display = 'none';
}
function show(el) {
el.style.display = '';
}
function enable(el) {
el.disabled = false;
}
function disable(el) {
el.disabled = true;
}
function submitForm(answer) {
// Pretend it's hitting the network.
return new Promise((resolve, reject) => {
setTimeout(() => {
if (answer.toLowerCase() == 'istanbul') {
resolve();
} else {
reject(new Error('Good guess but a wrong answer. Try again!'));
}
}, 1500);
});
}
let form = document.getElementById('form');
let textarea = document.getElementById('textarea');
let button = document.getElementById('button');
let loadingMessage = document.getElementById('loading');
let errorMessage = document.getElementById('error');
let successMessage = document.getElementById('success');
form.onsubmit = handleFormSubmit;
textarea.oninput = handleTextareaChange;
Step 1: 컴포넌트의 다양한 시각적 상태 식별하기
Step 2: 상태 변화를 촉발하는 요소를 파악하기
Step 3: useState를 사용하여 메모리의 상태를 표현하기
Step 4: 비필수적인 state 변수를 제거하기
Step 5: 이벤트 핸들러를 연결하여 state를 설정하기
모든 이벤트 핸들러가 연결된 최종 form
// App.js
import { useState } from 'react';
export default function Form() {
const [answer, setAnswer] = useState('');
const [error, setError] = useState(null);
const [status, setStatus] = useState('typing');
if (status === 'success') {
return <h1>That's right!</h1>
}
async function handleSubmit(e) {
e.preventDefault();
setStatus('submitting');
try {
await submitForm(answer);
setStatus('success');
} catch (err) {
setStatus('typing');
setError(err);
}
}
function handleTextareaChange(e) {
setAnswer(e.target.value);
}
return (
<>
<h2>City quiz</h2>
<p>
In which city is there a billboard that turns air into drinkable water?
</p>
<form onSubmit={handleSubmit}>
<textarea
value={answer}
onChange={handleTextareaChange}
disabled={status === 'submitting'}
/>
<br />
<button disabled={
answer.length === 0 ||
status === 'submitting'
}>
Submit
</button>
{error !== null &&
<p className="Error">
{error.message}
</p>
}
</form>
</>
);
}
function submitForm(answer) {
// Pretend it's hitting the network.
return new Promise((resolve, reject) => {
setTimeout(() => {
let shouldError = answer.toLowerCase() !== 'lima'
if (shouldError) {
reject(new Error('Good guess but a wrong answer. Try again!'));
} else {
resolve();
}
}, 1500);
});
}
처음 사용자가 빠르게 페이지를 보게 해주고 싶다!
미리 데이터를 불러오고, 페이지를 그려서 HTML 파일로 구워둔다
비용도 적게 들고 성능도 괜찮음
사용자 맞춤 & 데이터를 최신으로 유지하기 힘듦 (조금만 바뀌어도 빌드를 모두 다시 해야 함)
처음에 딱 첫 페이지를 그리고 빠르게 보여주고 검색엔진에도 걸렸으면 좋겠다
데이터가 없어 로딩이 필요 없는 애들. 애초에 가져오는 데이터가 없어서 다 똑같이 보니까 사용자마다 다르지 않음. => 로그인/회원가입, 상세페이지
SSR, SSG
- 사용자가 처음에 볼 것만, 첫 데이터만 넣고 리액트쿼리에게 맡기기
- 얘넨 알아서 불러오고 스크로해야할거나 2페이지는 클라이언트에서 그리기
export default async function handler(req, res) {
// Check for secret to confirm this is a valid request
if (req.query.secret !== process.env.MY_SECRET_TOKEN) {
return res.status(401).json({ message: 'Invalid token' });
}
try {
// this should be the actual path not a rewritten path
// e.g. for "/blog/[slug]" this should be "/blog/post-1"
await res.revalidate('/path-to-revalidate');
return res.json({ revalidated: true });
} catch (err) {
// If there was an error, Next.js will continue
// to show the last successfully generated page
return res.status(500).send('Error revalidating');
}
}