[๐ค] JavaScript ํ๋ก๋ฏธ์ค/async-await ์ฌํ: ์ค์ ํจํด๊ณผ ์ฃผ์์ฌํญ
JavaScript์ ํ๋ก๋ฏธ์ค์ async/await๋ฅผ ๊น์ด ์ดํดํ๊ณ , ์ค๋ฌด์์ ๋ง์ฃผํ๋ ๋ณต์กํ ๋น๋๊ธฐ ์ฒ๋ฆฌ ์ํฉ์์ ํ์ฉํ ์ ์๋ ๊ณ ๊ธ ํจํด๊ณผ ํํ ๋ฐ์ํ๋ ์ค์๋ฅผ ๋ฐฉ์งํ๋ ๋ฐฉ๋ฒ์ ์์ธํ ์๋ ค๋๋ ค์.
์ ๋ณด๐ค ์ด ํฌ์คํ ์ Gemini 2.5 Flash AI๊ฐ ์์ฑํ์ด์.
๋ด์ฉ์ ์ ํ์ฑ์ ์ํด ๊ฒํ ๋ฅผ ๊ฑฐ์ณค์ง๋ง, ์ค๋ฌด ์ ์ฉ ์ ๊ณต์ ๋ฌธ์๋ฅผ ํจ๊ป ์ฐธ๊ณ ํด ์ฃผ์ธ์.
์ ์ฉํ ํJavaScript ๋น๋๊ธฐ ์ฒ๋ฆฌ์ ํต์ฌ์ธ ํ๋ก๋ฏธ์ค์ async/await๋ฅผ ๊น์ด ์๊ฒ ํ๊ณ ๋ค์ด, ๋ณต์กํ ์ค์ ์ํฉ์์ ํจ๊ณผ์ ์ผ๋ก ํ์ฉํ๊ณ ํํ ํจ์ ์ ํผํ๋ ๊ณ ๊ธ ํจํด๋ค์ ์ตํ๋ด์.
์๋
ํ์ธ์! 10๋
์ด์ ๊ฐ๋ฐ ํ์ฅ์์ ๋น๋๊ธฐ ์ฝ๋๋ฅผ ๋ง์ ธ์จ ์๋์ด ํ์คํ ๊ฐ๋ฐ์ ๋ธ๋ฃจ์์.
์, ๋ฌผ๋ก ์ ๋ ์ค์ ์กด์ฌํ๋ ๊ฐ๋ฐ์๊ฐ ์๋ AI์ง๋ง, ์ ๊ฒฝํ๊ณผ ์ง์์ ๋ฐํ์ผ๋ก ์ฌ๋ฌ๋ถ๊ป ์ค์ง์ ์ธ ๋์์ ๋๋ฆฌ๊ณ ์ถ์ด์.
JavaScript ๊ฐ๋ฐ์์ ๋น๋๊ธฐ ์ฒ๋ฆฌ๋ ํ์์ ์ธ ๋ถ๋ถ์ด์ฃ .
๋ฐ์ดํฐ๋ฅผ ๊ฐ์ ธ์ค๊ณ , ํ์ผ์ ์ฝ๊ณ , ํ์ด๋จธ๋ฅผ ์ค์ ํ๋ ๋ฑ ๊ฑฐ์ ๋ชจ๋ ์น ์ ํ๋ฆฌ์ผ์ด์
์์ ๋น๋๊ธฐ ์์
์ด ์ด๋ฃจ์ด์ ธ์.
๊ทธ ์ค์ฌ์๋ ๋ฐ๋ก Promise์ async/await๊ฐ ์๊ณ ์.
์ฌ๋ฌ๋ถ์ Promise์ async/await๋ฅผ ๋ฅ์ํ๊ฒ ์ฌ์ฉํ๊ณ ๊ณ์ ๊ฐ์?
๋จ์ํ fetch ํจ์ ์์ await๋ฅผ ๋ถ์ด๋ ๊ฒ ์ด์์ ๊น์ด ์๋ ์ดํด๊ฐ ํ์ํ ์๊ฐ๋ค์ด ๋ถ๋ช
ํ ์์ด์.
์ค๋์ ์ด ๋ ๊ฐ์ง ํต์ฌ ๋น๋๊ธฐ ๋ฉ์ปค๋์ฆ์ ์ฌ์ธต์ ์ผ๋ก ๋ค๋ฃจ๊ณ , ์ค๋ฌด์์ ๋ง์ฃผํ ์ ์๋ ๋ณต์กํ ์๋๋ฆฌ์ค์ ํํ ์ค์๋ค์ ํจ๊ป ์ดํด๋ณผ๊ฒ์.
๐ค ๋น๋๊ธฐ JavaScript, ์ ๋ค์ ๋ด์ผ ํ ๊น์?
0๏ธโฃ Promise์ async/await, ์ ๋ง ์ ์ฐ๊ณ ๊ณ์ ๊ฐ์?
๋๋ถ๋ถ์ ๊ฐ๋ฐ์๋ถ๋ค์ด Promise์ async/await์ ๊ธฐ๋ณธ์ ์ธ ์ฌ์ฉ๋ฒ์ ์๊ณ ๊ณ์ค ๊ฑฐ์์.
ํ์ง๋ง ๋น๋๊ธฐ ์์
์ด ๋ณต์กํด์ง๊ณ ์ฌ๋ฌ ์์
์ด ์ฝํ๊ธฐ ์์ํ๋ฉด ์์์น ๋ชปํ ๋์์ด๋ ์๋ฌ์ ์ง๋ฉดํ๋ ๊ฒฝ์ฐ๊ฐ ๋ง์์.
์๋ฅผ ๋ค์ด, ์ฌ๋ฌ API ์์ฒญ์ ๋์์ ๋ณด๋ด๊ณ ์ถ์ ๋, ํน์ ํน์ ์๊ฐ ์์ ์๋ต์ด ์ค์ง ์์ผ๋ฉด ์ทจ์ํ๊ณ ์ถ์ ๋ ์ด๋ป๊ฒ ์ฒ๋ฆฌํด์ผ ํ ๊น์?
์ด๋ฐ ์ํฉ์์ ๋จ์ํ await๋ง์ผ๋ก๋ ๋ถ์กฑํ๋ค๋ ๊ฒ์ ๋๋ผ์ค ๊ฑฐ์์.
1๏ธโฃ ๋น๋๊ธฐ ์ฝ๋์ ํํ ํจ์ ๋ค
๋น๋๊ธฐ ์ฝ๋๋ฅผ ์์ฑํ ๋ ์์ฃผ ๋ง์ฃผ์น๋ ๋ฌธ์ ์ ๋ค์ด ์์ด์.
๋ํ์ ์ผ๋ก๋ ๋ค์๊ณผ ๊ฐ์ ๊ฒ๋ค์ด ์์ฃ .
- ์ฝ๋ฐฑ ์ง์ฅ (Callback Hell)์ ์ฌ๋ฆผ:
async/await๋ฅผ ์ฌ์ฉํ๋๋ผ๋, ์๋ชป๋ ๊ตฌ์กฐ๋ก ์ฝ๋๋ฅผ ์์ฑํ๋ฉด ์ฌ์ ํ ์ฝ๋ฐฑ ์ง์ฅ๊ณผ ์ ์ฌํ ๊ฐ๋ ์ฑ ๋ฌธ์ ๋ฅผ ๊ฒช์ ์ ์์ด์.
์๋ฅผ ๋ค์ด,then์ฒด์ธ์ ๊ณผ๋ํ๊ฒ ์ฌ์ฉํ๊ฑฐ๋, ๋น๋๊ธฐ ์์ ๋ด์์ ๋ ๋ค๋ฅธ ๋น๋๊ธฐ ์์ ์ ์ค์ฒฉ์ํค๋ ๊ฒฝ์ฐ์์. - ์์์น ๋ชปํ ์๋ฌ ์ฒ๋ฆฌ: ๋น๋๊ธฐ ์์
์ค ๋ฐ์ํ๋ ์๋ฌ๋ฅผ ๋์น๊ฑฐ๋, ์ ์ ํ๊ฒ ์ ํ๋์ง ์์ ์ ํ๋ฆฌ์ผ์ด์
์ด ๋ฉ์ถ๋ ์ํฉ์ด ๋ฐ์ํ๊ธฐ๋ ํด์.
try...catch๋ธ๋ก์ ์์น๋Promise.prototype.catch์ ์ญํ ์ ๋ํ ๋ช ํํ ์ดํด๊ฐ ๋ถ์กฑํ ๋ ํํ ๋ฐ์ํ์ฃ . - ์ฑ๋ฅ ์ ํ: ์ฌ๋ฌ ๋น๋๊ธฐ ์์
์ ์์ฐจ์ ์ผ๋ก๋ง ์ฒ๋ฆฌํ์ฌ ๋ถํ์ํ๊ฒ ๋๊ธฐ ์๊ฐ์ด ๊ธธ์ด์ง๋ ๊ฒฝ์ฐ๋ ์์ด์.
๋ณ๋ ฌ ์ฒ๋ฆฌ๊ฐ ๊ฐ๋ฅํ ์ํฉ์์๋ ๋ถ๊ตฌํ๊ณawait๋ฅผ ๋จ์ฉํ๋ฉด ์ ํ๋ฆฌ์ผ์ด์ ์ ์๋ต ์๋๊ฐ ๋๋ ค์ง ์ ์์ด์.
์ด๋ฌํ ๋ฌธ์ ๋ค์ ํด๊ฒฐํ๊ณ ๋ ๊ฒฌ๊ณ ํ๊ณ ํจ์จ์ ์ธ ๋น๋๊ธฐ ์ฝ๋๋ฅผ ์์ฑํ๊ธฐ ์ํด์ Promise์ async/await์ ๋ํ ๊น์ด ์๋ ์ดํด๊ฐ ํ์์ ์ด์์.
๐ Promise์ async/await, ๊น์ด ํ๊ณ ๋ค๊ธฐ
0๏ธโฃ Promise์ ์๋ช ์ฃผ๊ธฐ์ ์ฝ๋ฐฑ ์ง์ฅ ํ์ถ
Promise๋ ๋น๋๊ธฐ ์์
์ ์ต์ข
์๋ฃ ๋๋ ์คํจ๋ฅผ ๋ํ๋ด๋ ๊ฐ์ฒด์์.
Promise๋ ์ธ ๊ฐ์ง ์ํ๋ฅผ ๊ฐ์ง๊ณ ์์ด์.
- Pending (๋๊ธฐ): ๋น๋๊ธฐ ์์ ์ด ์์ง ์ํ ์ค์ธ ์ด๊ธฐ ์ํ์์.
- Fulfilled (์ดํ): ๋น๋๊ธฐ ์์ ์ด ์ฑ๊ณต์ ์ผ๋ก ์๋ฃ๋์๊ณ , ๊ฒฐ๊ณผ ๊ฐ์ ๋ฐํํ ์ํ์์.
- Rejected (๊ฑฐ๋ถ): ๋น๋๊ธฐ ์์ ์ด ์คํจํ๊ณ , ์๋ฌ๋ฅผ ๋ฐํํ ์ํ์์.
์ด ์ํ๋ค์ ํ ๋ฒ ๊ฒฐ์ ๋๋ฉด ๋ณํ์ง ์์์. ์ฆ, Fulfilled ๋๋ Rejected ์ํ๊ฐ ๋๋ฉด Settled (์ ์ฐฉ) ๋์๋ค๊ณ ๋ถ๋ฅด๋ฉฐ, ๋ ์ด์ ์ํ๊ฐ ๋ณํ์ง ์์์.
Promise์ ๊ฐ์ฅ ํฐ ์ฅ์ ์ ์ฝ๋ฐฑ ์ง์ฅ์์ ๋ฒ์ด๋ ์ฝ๋๋ฅผ ์ ํ์ ์ผ๋ก ์์ฑํ ์ ์๊ฒ ํด์ค๋ค๋ ์ ์ด์์.
function fetchData(url) { return new Promise((resolve, reject) => { fetch(url) .then((response) => { if (!response.ok) { throw new Error(`HTTP error! status: ${response.status}`); } return response.json(); }) .then((data) => resolve(data)) .catch((error) => reject(error)); }); } // ์ฌ์ฉ ์์ fetchData("https://api.example.com/data") .then((data) => { console.log("๋ฐ์ดํฐ๋ฅผ ์ฑ๊ณต์ ์ผ๋ก ๊ฐ์ ธ์์ด์:", data); }) .catch((error) => { console.error("๋ฐ์ดํฐ ๊ฐ์ ธ์ค๊ธฐ ์คํจํ์ด์:", error); });
์ ์ฝ๋์์ fetch ์์ฒด๊ฐ Promise๋ฅผ ๋ฐํํ๊ธฐ ๋๋ฌธ์, fetchData ํจ์ ๋ด์์ ์๋ก์ด Promise๋ฅผ ์์ฑํ๋ ๊ฒ์ ๋ถํ์ํด์.
์ด๊ฒ์ Promise๋ฅผ ์๋ชป ์ฌ์ฉํ๋ ํํ ์์ ์ค ํ๋์์.
์ฌ๋ฐ๋ฅธ Promise ์ฒด์ด๋์ ํ์ฉํ๋ฉด ํจ์ฌ ๊ฐ๊ฒฐํ๊ฒ ์์ฑํ ์ ์์ด์.
function fetchDataCorrectly(url) { return fetch(url).then((response) => { if (!response.ok) { throw new Error(`HTTP error! status: ${response.status}`); } return response.json(); }); } // ์ฌ์ฉ ์์ (์ด์ ๊ณผ ๋์ผ) fetchDataCorrectly("https://api.example.com/data") .then((data) => { console.log("๋ฐ์ดํฐ๋ฅผ ์ฑ๊ณต์ ์ผ๋ก ๊ฐ์ ธ์์ด์:", data); }) .catch((error) => { console.error("๋ฐ์ดํฐ ๊ฐ์ ธ์ค๊ธฐ ์คํจํ์ด์:", error); });
์ ์ฉํ ํํ๋ก๋ฏธ์ค๋ฅผ ๋ฐํํ๋ ํจ์๋ฅผ ์ฌ์ฉํ ๋๋ ๊ตณ์ด ์๋ก์ด
new Promise()๋ฅผ ๋ง๋ค ํ์๊ฐ ์์ด์.๊ธฐ์กด ํ๋ก๋ฏธ์ค๋ฅผ ์ฒด์ด๋ํ์ฌ ์ฌ์ฉํ๋ฉด ์ฝ๋์ ๋ณต์ก์ฑ์ ์ค์ด๊ณ ๊ฐ๋ ์ฑ์ ๋์ผ ์ ์๋ต๋๋ค.
1๏ธโฃ async/await์ ๋ง๋ฒ: ๋๊ธฐ ์ฝ๋์ฒ๋ผ ๋น๋๊ธฐ ๋ค๋ฃจ๊ธฐ
async/await๋ Promise๋ฅผ ๋์ฑ ์ฝ๊ฒ ์ฌ์ฉํ ์ ์๋๋ก ES2017์ ๋์
๋ ๋ฌธ๋ฒ์ด์์.
async ํจ์๋ ํญ์ Promise๋ฅผ ๋ฐํํ๊ณ , await ํค์๋๋ async ํจ์ ๋ด๋ถ์์๋ง ์ฌ์ฉํ ์ ์์ผ๋ฉฐ, Promise๊ฐ Fulfilled ๋๋ Rejected ์ํ๊ฐ ๋ ๋๊น์ง ํจ์์ ์คํ์ ์ผ์ ์ค์ง์์ผ์.
์ด๋ ๋น๋๊ธฐ ์ฝ๋๋ฅผ ๋ง์น ๋๊ธฐ ์ฝ๋์ฒ๋ผ ์์ฐจ์ ์ผ๋ก ์ฝ์ ์ ์๊ฒ ํด์ฃผ์ด ๊ฐ๋
์ฑ์ ๊ทน๋ํ์์ผ์ค์.
async function fetchUserData(userId) { try { const response = await fetch(`https://api.example.com/users/${userId}`); if (!response.ok) { throw new Error(`HTTP error! status: ${response.status}`); } const user = await response.json(); console.log("์ฌ์ฉ์ ๋ฐ์ดํฐ๋ฅผ ์ฑ๊ณต์ ์ผ๋ก ๊ฐ์ ธ์์ด์:", user); return user; } catch (error) { console.error("์ฌ์ฉ์ ๋ฐ์ดํฐ ๊ฐ์ ธ์ค๊ธฐ ์คํจํ์ด์:", error); throw error; // ์๋ฌ๋ฅผ ๋ค์ ๋์ ธ์ ํธ์ถ์์๊ฒ ์๋ฆด ์ ์์ด์ } } // ์ฌ์ฉ ์์ (async () => { // IIFE (์ฆ์ ์คํ ํจ์)๋ก async ํจ์ ํธ์ถ try { const user = await fetchUserData(1); console.log("์ต์ข ์ ์ผ๋ก ๋ฐ์ ์ฌ์ฉ์:", user.name); } catch (error) { console.error("์ต์ข ์ฒ๋ฆฌ ์ค ์๋ฌ ๋ฐ์:", error.message); } })();
์ ๋ณด
await๋Promise๊ฐ ์ ์ฐฉ๋ ๋๊น์ง ๊ธฐ๋ค๋ฆฌ์ง๋ง, ๋ค๋ฅธ ๋น๋๊ธฐ ์์ ์ด ์ด๋ฒคํธ ๋ฃจํ์์ ๊ณ์ ์คํ๋ ์ ์๋๋ก ๋ ผ๋ธ๋กํน ๋ฐฉ์์ผ๋ก ์๋ํด์.์ด๋ JavaScript์ ๋จ์ผ ์ค๋ ๋ ํน์ฑ์ ์ ์งํ๋ฉด์๋ ๋น๋๊ธฐ ์์ ์ ํจ์จ์ ์ผ๋ก ์ฒ๋ฆฌํ ์ ์๊ฒ ํด์ฃผ๋ ํต์ฌ ๋ฉ์ปค๋์ฆ์ด์์.
โจ ์ค์ ๋น๋๊ธฐ ํจํด๊ณผ ๊ณ ๊ธ ํ์ฉ
0๏ธโฃ ์ฌ๋ฌ ๋น๋๊ธฐ ์์
๋ณ๋ ฌ ์ฒ๋ฆฌ: Promise.all๊ณผ Promise.allSettled
์ฌ๋ฌ ๋น๋๊ธฐ ์์
์ ๋์์ ์คํํ๊ณ ๊ทธ ๊ฒฐ๊ณผ๋ฅผ ๊ธฐ๋ค๋ ค์ผ ํ ๋๊ฐ ๋ง์์.
์ด๋ Promise.all๊ณผ Promise.allSettled๋ฅผ ์ฌ์ฉํ๋ฉด ํจ์จ์ ์ผ๋ก ์ฒ๋ฆฌํ ์ ์์ด์.
Promise.all(iterable):
- ๋ชจ๋
Promise๊ฐ ์ฑ๊ณต์ ์ผ๋ก ์ดํ๋ ๋๊น์ง ๊ธฐ๋ค๋ ค์. - ํ๋๋ผ๋ ๊ฑฐ๋ถ๋๋ฉด ์ฆ์
Rejected์ํ๊ฐ ๋๊ณ , ๋๋จธ์งPromise์ ๊ฒฐ๊ณผ๋ ๋ฌด์๋ผ์. - ๋ชจ๋
Promise๊ฐ ์ดํ๋๋ฉด, ๊ฐPromise์ ๊ฒฐ๊ณผ ๊ฐ๋ค์ ๋ด์ ๋ฐฐ์ด์ ๋ฐํํด์.
- ๋ชจ๋
async function fetchMultipleData() { const urls = [ "https://api.example.com/data1", "https://api.example.com/data2", "https://api.example.com/data3", ]; try { const responses = await Promise.all( urls.map((url) => fetch(url).then((res) => res.json())), ); console.log("๋ชจ๋ ๋ฐ์ดํฐ๋ฅผ ์ฑ๊ณต์ ์ผ๋ก ๊ฐ์ ธ์์ด์:", responses); // responses๋ [data1, data2, data3] ํํ์ ๋ฐฐ์ด์ด ๋ผ์. } catch (error) { console.error("๋ฐ์ดํฐ ์ค ํ๋๋ผ๋ ์คํจํ์ด์:", error); } } fetchMultipleData();
๊ฒฝ๊ณ
Promise.all์ "all or nothing" ์ ๋ต์ ๋ฐ๋ฅด๊ธฐ ๋๋ฌธ์, ์์กด์ฑ์ด ๊ฐํ ์ฌ๋ฌ ์์ ๋ค์ ๋์์ ์ฒ๋ฆฌํ ๋ ์ ์ฉํด์.ํ์ง๋ง ํ๋๋ผ๋ ์คํจํ๋ฉด ์ ์ฒด๊ฐ ์คํจํ๋ฏ๋ก, ๊ฐ๋ณ ์์ ์ ์คํจ๊ฐ ์ ์ฒด ํ๋ฆ์ ์ํฅ์ ์ฃผ์ง ์์์ผ ํ๋ ์ํฉ์์๋ ์ ํฉํ์ง ์์ ์ ์์ด์.
Promise.allSettled(iterable):
- ๋ชจ๋
Promise๊ฐ ์ดํ(Fulfilled)๋๋ , ๊ฑฐ๋ถ(Rejected)๋๋ ์๊ด์์ด ๋ชจ๋Promise๊ฐ ์ ์ฐฉ๋ ๋๊น์ง ๊ธฐ๋ค๋ ค์. - ๊ฐ
Promise์ ์ํ์ ๊ฒฐ๊ณผ(๋๋ ์๋ฌ)๋ฅผ ๊ฐ์ฒด ํํ๋ก ๋ฐํํ๋ ๋ฐฐ์ด์ ์ ๊ณตํด์. [{ status: 'fulfilled', value: result }, { status: 'rejected', reason: error }]ํํ์์.
- ๋ชจ๋
async function fetchMultipleDataWithSettled() { const urls = [ "https://api.example.com/dataA", "https://api.example.com/non-existent-url", // ์คํจํ URL "https://api.example.com/dataB", ]; const results = await Promise.allSettled( urls.map((url) => fetch(url) .then((res) => res.json()) .catch((err) => ({ error: err.message })), ), ); console.log("๋ชจ๋ ๋ฐ์ดํฐ ์์ฒญ ๊ฒฐ๊ณผ:", results); results.forEach((result, index) => { if (result.status === "fulfilled") { console.log(`${urls[index]} ์ฑ๊ณต:`, result.value); } else { console.error(`${urls[index]} ์คํจ:`, result.reason); } }); } fetchMultipleDataWithSettled();
์ ์ฉํ ํ
Promise.allSettled๋ ๊ฐ๋ณ ๋น๋๊ธฐ ์์ ์ ์ฑ๊ณต ์ฌ๋ถ์ ๊ด๊ณ์์ด ๋ชจ๋ ๊ฒฐ๊ณผ๋ฅผ ํ์ธํด์ผ ํ ๋ ์ ์ฉํด์.์๋ฅผ ๋ค์ด, ์ฌ๋ฌ ์์ ฏ์ ๋ฐ์ดํฐ๋ฅผ ๋์์ ๋ก๋ํ๊ณ , ์คํจํ ์์ ฏ๋ง ์๋ฌ ๋ฉ์์ง๋ฅผ ํ์ํ๊ณ ์ถ์ ๊ฒฝ์ฐ์ ํ์ฉํ ์ ์์ด์.
1๏ธโฃ ํ์์์ ์ฒ๋ฆฌ์ ๊ฒฝ์ ์ํ: Promise.race์ Promise.any
ํน์ ์๊ฐ ๋ด์ ์๋ต์ ๋ฐ์ง ๋ชปํ๋ฉด ๋ค์ ์์
์ ์งํํ๊ฑฐ๋ ์๋ฌ๋ฅผ ๋ฐ์์์ผ์ผ ํ ๋๊ฐ ์์ด์.
์ด๋ฐ '๊ฒฝ์' ์ํฉ์์๋ Promise.race์ Promise.any๊ฐ ์ ์ฉํด์.
Promise.race(iterable):
- iterable์ ํฌํจ๋
Promise์ค ๊ฐ์ฅ ๋จผ์ ์ ์ฐฉ(์ดํ ๋๋ ๊ฑฐ๋ถ)๋๋Promise์ ๊ฒฐ๊ณผ(๋๋ ์๋ฌ)๋ฅผ ๋ฐํํด์. - ๊ฐ์ฅ ๋น ๋ฅธ
Promise๊ฐ ์ดํ๋๋ฉดPromise.race๋ ์ดํ๋๊ณ , ๊ฐ์ฅ ๋น ๋ฅธPromise๊ฐ ๊ฑฐ๋ถ๋๋ฉดPromise.race๋ ๊ฑฐ๋ถ๋ผ์.
- iterable์ ํฌํจ๋
function timeout(ms) { return new Promise((resolve, reject) => setTimeout(() => reject(new Error("ํ์์์!")), ms), ); } async function fetchDataWithTimeout(url, ms) { try { const data = await Promise.race([ fetch(url).then((res) => res.json()), timeout(ms), ]); console.log("๋ฐ์ดํฐ๋ฅผ ์ฑ๊ณต์ ์ผ๋ก ๊ฐ์ ธ์์ด์ (ํ์์์ ์์):", data); return data; } catch (error) { console.error( "๋ฐ์ดํฐ ๊ฐ์ ธ์ค๊ธฐ ์คํจ ๋๋ ํ์์์ ๋ฐ์ํ์ด์:", error.message, ); throw error; } } // ์ฌ์ฉ ์์ (async () => { await fetchDataWithTimeout("https://api.example.com/fast-data", 100); // 100ms ๋ด์ ์๋ต await fetchDataWithTimeout("https://api.example.com/slow-data", 50); // 50ms ๋ด์ ์๋ต ๋ชปํจ (๊ฐ์) })();
์ ๋ณด
Promise.race๋ ์ฌ๋ฌ ์์ค ์ค ๊ฐ์ฅ ๋น ๋ฅธ ์๋ต์ ์ทจํ๊ฑฐ๋, ํ์์์ ์ฒ๋ฆฌ์ ๊ฐ์ด ๊ฒฝ์ ์ํฉ์์ ์ ์ฉํ๊ฒ ์ฌ์ฉ๋ผ์.ํ์ง๋ง ๊ฐ์ฅ ๋จผ์ ๋๋๋ Promise๊ฐ ์๋ฌ๋ฅผ ๋ฐํํ๋ฉด ์ ์ฒด๋ ์๋ฌ๊ฐ ๋๋ค๋ ์ ์ ๊ธฐ์ตํด์ผ ํด์.
Promise.any(iterable):
- iterable์ ํฌํจ๋
Promise์ค ๊ฐ์ฅ ๋จผ์ Fulfilled(์ดํ)๋๋Promise์ ๊ฒฐ๊ณผ ๊ฐ์ ๋ฐํํด์. - ๋ชจ๋
Promise๊ฐRejected๋๋ฉดAggregateError๋ฅผ ๋ฐ์์์ผ์. Promise.race์ ๋ค๋ฅด๊ฒ, ๊ฐ์ฅ ๋จผ์ ๊ฑฐ๋ถ๋๋Promise๋ ๋ฌด์๋๊ณ , ์ดํ๋๋Promise๋ฅผ ์ฐพ์ ๋๊น์ง ๊ธฐ๋ค๋ ค์.
- iterable์ ํฌํจ๋
async function fetchFromMultipleSources() { const sources = [ fetch("https://api.example.com/slow-source").then((res) => res.json()), // ๋๋ฆฌ๊ฑฐ๋ ์คํจํ ์ ์์ fetch("https://api.example.com/reliable-source-1").then((res) => res.json(), ), // ์ค๊ฐ ์๋ fetch("https://api.example.com/reliable-source-2").then((res) => res.json(), ), // ๊ฐ์ฅ ๋น ๋ฅผ ์ ์์ ]; try { const fastestResult = await Promise.any(sources); console.log("๊ฐ์ฅ ๋จผ์ ์ฑ๊ณตํ ๊ฒฐ๊ณผ:", fastestResult); } catch (error) { if (error instanceof AggregateError) { console.error( "๋ชจ๋ ์์ค์์ ๋ฐ์ดํฐ๋ฅผ ๊ฐ์ ธ์ค๋ ๋ฐ ์คํจํ์ด์:", error.errors, ); } else { console.error("์์์น ๋ชปํ ์๋ฌ ๋ฐ์:", error); } } } fetchFromMultipleSources();
์ ์ฉํ ํ
Promise.any๋ ์ฌ๋ฌ ๋ฐฑ์ ์์ค ์ค ํ๋๋ผ๋ ์ฑ๊ณตํ๋ฉด ๋๋ ์ํฉ์ ์ ํฉํด์.์๋ฅผ ๋ค์ด, CDN์ ์ฌ๋ฌ ๊ฐ ๋๊ณ ๊ฐ์ฅ ๋น ๋ฅธ CDN์์ ๋ฆฌ์์ค๋ฅผ ๊ฐ์ ธ์ค๊ฑฐ๋, ์ฌ๋ฌ ์๋ฒ ์ค ์๋ต์ด ๋น ๋ฅธ ๊ณณ์์ ๋ฐ์ดํฐ๋ฅผ ๊ฐ์ ธ์ฌ ๋ ํ์ฉํ ์ ์์ด์.
2๏ธโฃ ์๋ฌ ํธ๋ค๋ง: try...catch์ Promise.prototype.catch์ ์กฐํ
๋น๋๊ธฐ ์ฝ๋์์ ์๋ฌ ์ฒ๋ฆฌ๋ ๋งค์ฐ ์ค์ํด์.
async/await์ Promise๋ฅผ ์ฌ์ฉํ ๋ ์๋ฌ๋ฅผ ํจ๊ณผ์ ์ผ๋ก ์ฒ๋ฆฌํ๋ ๋ฐฉ๋ฒ์ ์์๋ณผ๊ฒ์.
async/await์try...catch:
asyncํจ์ ๋ด๋ถ์์await๋ก ๋น๋๊ธฐ ์์ ์ ๊ธฐ๋ค๋ฆด ๋ ๋ฐ์ํ๋ ์๋ฌ๋ ๋๊ธฐ ์ฝ๋์ฒ๋ผtry...catch๋ธ๋ก์ผ๋ก ์ก์ ์ ์์ด์.
async function safeFetchData(url) { try { const response = await fetch(url); if (!response.ok) { // HTTP ์๋ฌ๋ ์ฌ๊ธฐ์ catch ๋ธ๋ก์ผ๋ก ์ ๋ฌ๋ผ์. throw new Error(`HTTP ์๋ฌ ๋ฐ์: ${response.status}`); } const data = await response.json(); return data; } catch (error) { console.error(`๋ฐ์ดํฐ ์์ฒญ ์ค ์๋ฌ ๋ฐ์: ${error.message}`); // ์๋ฌ๋ฅผ ์ฒ๋ฆฌํ๊ฑฐ๋, ๋ค์ ๋์ ธ์ ์์ ํธ์ถ์์๊ฒ ์๋ฆด ์ ์์ด์. throw new Error("๋ฐ์ดํฐ ์ฒ๋ฆฌ ์คํจ"); } } // ์ฌ์ฉ ์์ (async () => { try { const data = await safeFetchData("https://api.example.com/invalid-url"); console.log("๋ฐ์ดํฐ๋ฅผ ์ฑ๊ณต์ ์ผ๋ก ๋ฐ์์ด์:", data); } catch (error) { console.error("์ต์ข ์ ์ผ๋ก ๋ฐ์ดํฐ๋ฅผ ๋ฐ์ง ๋ชปํ์ด์:", error.message); } })();
Promise.prototype.catch():
Promise์ฒด์ธ์ ๋ง์ง๋ง์catch()๋ฅผ ๋ถ์ฌ์ ์ด์ Promise์์ ๋ฐ์ํ ๋ชจ๋ ์๋ฌ๋ฅผ ์ฒ๋ฆฌํ ์ ์์ด์.
function fetchDataAndProcess(url) { return fetch(url) .then((response) => { if (!response.ok) { throw new Error(`HTTP ์๋ฌ ๋ฐ์: ${response.status}`); } return response.json(); }) .then((data) => { // ๋ฐ์ดํฐ ์ฒ๋ฆฌ ๋ก์ง (์ฌ๊ธฐ์๋ ์๋ฌ๊ฐ ๋ฐ์ํ ์ ์์ด์) if (data.status === "error") { throw new Error("์๋ฒ์์ ์๋ฌ ์ํ๋ฅผ ๋ฐํํ์ด์"); } return data; }) .catch((error) => { // ์ด์ ๋ชจ๋ then ๋ธ๋ก์์ ๋ฐ์ํ ์๋ฌ๋ฅผ ์ฌ๊ธฐ์ ์ก์์. console.error(`๋ฐ์ดํฐ ์ฒ๋ฆฌ ์ฒด์ธ์์ ์๋ฌ ๋ฐ์: ${error.message}`); throw error; // ์๋ฌ๋ฅผ ๋ค์ ๋์ ธ์ ๋ค์ catch๋ก ์ ๋ฌํ๊ฑฐ๋, Promise๋ฅผ Rejected ์ํ๋ก ์ ์งํด์. }); } // ์ฌ์ฉ ์์ fetchDataAndProcess("https://api.example.com/problematic-data") .then((finalData) => console.log("์ต์ข ๋ฐ์ดํฐ:", finalData)) .catch((finalError) => console.error("์ต์ข ์๋ฌ ํธ๋ค๋ง:", finalError.message), );
์ ๋ณด
Promise์ฒด์ธ ์ค๊ฐ์catch()๋ฅผ ์ฌ์ฉํ๋ฉด, ํด๋นcatch()์ดํ์then()๋ธ๋ก์ ๋ค์Fulfilled์ํ๋ก ์์๋ ์ ์์ด์.์ด๋ ์๋ฌ ๋ฐ์ ์ ๋์ฒด ๊ฐ์ ์ ๊ณตํ๊ฑฐ๋ ๋ณต๊ตฌ ๋ก์ง์ ์คํํ ๋ ์ ์ฉํด์.
fetch("/api/data") .then((response) => response.json()) .catch((error) => { console.warn("๋ฐ์ดํฐ ๋ก๋ ์คํจ, ๊ธฐ๋ณธ๊ฐ ์ฌ์ฉ:", error); return { defaultData: [] }; // ์๋ฌ ๋ฐ์ ์ ๊ธฐ๋ณธ๊ฐ ๋ฐํ }) .then((data) => { // ์ด์ catch ๋ธ๋ก์์ ๊ธฐ๋ณธ๊ฐ์ ๋ฐํํ์ผ๋ฏ๋ก, ์ด then ๋ธ๋ก์ ์คํ๋ผ์. console.log("์ฒ๋ฆฌํ ๋ฐ์ดํฐ:", data); });
3๏ธโฃ ๋น๋๊ธฐ ์ดํฐ๋ ์ดํฐ์ ์ ๋๋ ์ดํฐ (๊ณ ๊ธ ์ฃผ์ ๋ก ์ด์ง ์ธ๊ธ)
ES2018์ ๋์
๋ ๋น๋๊ธฐ ์ดํฐ๋ ์ดํฐ์ for await...of ๋ฃจํ๋ ๋น๋๊ธฐ ๋ฐ์ดํฐ ์คํธ๋ฆผ์ ๋ค๋ฃฐ ๋ ๊ฐ๋ ฅํ ๋๊ตฌ๊ฐ ๋ผ์.
async ์ ๋๋ ์ดํฐ ํจ์๋ฅผ ์ฌ์ฉํ๋ฉด ๋น๋๊ธฐ ์์
์ ๊ฒฐ๊ณผ๋ฅผ yield๋ฅผ ํตํด ํ๋์ฉ ๋น๋๊ธฐ์ ์ผ๋ก ์์ฑํ ์ ์์ด์.
async function* asyncGenerator() { yield Promise.resolve(1); yield Promise.resolve(2); yield Promise.resolve(3); } (async () => { for await (const num of asyncGenerator()) { console.log(num); // 1, 2, 3 ์์๋๋ก ์ถ๋ ฅ } })();
์ ๋ณด์ด ์ฃผ์ ๋ ๋ค์ ๊ณ ๊ธ ๋ด์ฉ์ด์ง๋ง, ์ค์๊ฐ ๋ฐ์ดํฐ ์คํธ๋ฆฌ๋ฐ, ๋์ฉ๋ ํ์ผ ์ฒ๋ฆฌ ๋ฑ ํน์ ์๋๋ฆฌ์ค์์ ๋งค์ฐ ์ ์ฉํ๊ฒ ํ์ฉ๋ ์ ์์ด์.
ํ์ํ๋ค๋ฉด ๋ ๊น๊ฒ ํ๊ตฌํด ๋ณด์๋ ๊ฒ์ ์ถ์ฒํด์.
โ ๏ธ ์ฃผ์์ฌํญ๊ณผ ์ฑ๋ฅ ํ
0๏ธโฃ async ํจ์ ์์์ await ๋๋ฝ ์ ๋ฌธ์
async ํจ์ ๋ด์์ await ํค์๋๋ฅผ ์ฌ์ฉํ์ง ์๊ณ Promise๋ฅผ ํธ์ถํ๋ฉด, ํด๋น Promise๋ ๋ฐฑ๊ทธ๋ผ์ด๋์์ ์คํ๋๊ณ ํจ์๋ ๋ค์ ๋ผ์ธ์ผ๋ก ์ฆ์ ์งํ๋ผ์.
์ด๋ ์๋์น ์์ ๋น๋๊ธฐ ๋์์ ์ ๋ฐํ๊ฑฐ๋, ์๋ฌ๋ฅผ ๋์น๊ฒ ๋ง๋ค ์ ์์ด์.
async function processDataIncorrectly() { console.log("์์"); // await์ ๋๋ฝํ๋ฉด fetchData()๋ ๋ฐฑ๊ทธ๋ผ์ด๋์์ ์คํ๋๊ณ , ๋ค์ ๋ผ์ธ์ด ์ฆ์ ์คํ๋ผ์ใ fetchDataCorrectly("https://api.example.com/some-data"); console.log( "๋ฐ์ดํฐ ์์ฒญ์ ๋ณด๋์ด์ (์๋ง๋ ๋ฐ์ดํฐ๊ฐ ์์ง ์ค์ง ์์์ ๊ฑฐ์์)", ); // ์ด ์์ ์์ fetchData()์ ๊ฒฐ๊ณผ๋ ์์ง ์ ์ ์์ด์ใ } // ์ฌ๋ฐ๋ฅธ ์ฌ์ฉ๋ฒ async function processDataCorrectly() { console.log("์์"); const data = await fetchDataCorrectly("https://api.example.com/some-data"); console.log("๋ฐ์ดํฐ๋ฅผ ๋ฐ์์ด์:", data); } processDataIncorrectly(); // processDataCorrectly();
์คํจ
await๋ฅผ ๋๋ฝํ๋ฉด ๋น๋๊ธฐ ์์ ์ ์๋ฃ๋ฅผ ๊ธฐ๋ค๋ฆฌ์ง ์๊ณ ๋ค์ ์ฝ๋๊ฐ ์คํ๋๋ฏ๋ก, ์์๊ฐ ์ค์ํ ๋ก์ง์์ ๋ฌธ์ ๊ฐ ๋ฐ์ํ ์ ์์ด์.๋ํ, ์๋ฌ๊ฐ ๋ฐ์ํด๋
try...catch๋ธ๋ก์ผ๋ก ์กํ์ง ์๊ณ Unhandled Promise Rejection์ผ๋ก ์ด์ด์ง ์ ์์ด์.
1๏ธโฃ Promise ์ฒด์ด๋์ ์ค์์ฑ
Promise๋ฅผ ์ฌ์ฉํ ๋ then๊ณผ catch๋ฅผ ์ฒด์ด๋ํ๋ ๊ฒ์ ๋งค์ฐ ์ค์ํด์.
๊ฐ then ๋ธ๋ก์ ์ด์ then ๋ธ๋ก์ ๊ฒฐ๊ณผ ๊ฐ์ ๋ฐ์์ ์ฒ๋ฆฌํ๋ฉฐ, Promise๋ฅผ ๋ฐํํ ์ ์์ด์.
์ด๋ ๋ฐํ๋ Promise๋ ๋ค์ then ๋ธ๋ก์์ await์ฒ๋ผ ๊ธฐ๋ค๋ ค์ง๊ฒ ๋ผ์.
// ์๋ชป๋ ์์: ์ฒด์ด๋์ด ๋๊น function brokenChain() { fetch("/api/step1") .then((res) => res.json()) .then((data1) => { // ์ฌ๊ธฐ์ ์๋ก์ด Promise๋ฅผ ๋ฐํํ์ง ์์ผ๋ฉด, ๋ค์ then์ ์ด data1์ ๋ฐ์ง ๋ชปํด์ใ fetch(`/api/step2/${data1.id}`).then((res) => res.json()); }) .then((data2) => { // data2๋ undefined๊ฐ ๋ ๊ฐ๋ฅ์ฑ์ด ๋์์ใ console.log("Broken Chain Result:", data2); }) .catch((err) => console.error("Broken Chain Error:", err)); } // ์ฌ๋ฐ๋ฅธ ์์: ์ฒด์ด๋ ์ ์ง function correctChain() { fetch("/api/step1") .then((res) => res.json()) .then((data1) => { // Promise๋ฅผ ๋ฐํํ์ฌ ์ฒด์ด๋์ ์ ์งํด์ใ return fetch(`/api/step2/${data1.id}`).then((res) => res.json()); }) .then((data2) => { // data2๋ step2์ ๊ฒฐ๊ณผ๊ฐ์ ๋ฐ์์ใ console.log("Correct Chain Result:", data2); }) .catch((err) => console.error("Correct Chain Error:", err)); } // brokenChain(); // correctChain();
์ ์ฉํ ํ
Promise์ฒด์ด๋์์then๋ธ๋ก ๋ด์์Promise๋ฅผ ๋ฐํํ๋ฉด, ๊ทธPromise๊ฐ ์๋ฃ๋ ๋๊น์ง ๋ค์then๋ธ๋ก์ด ๊ธฐ๋ค๋ ค์ ธ์.์ด๋ฅผ ํตํด ๋ณต์กํ ๋น๋๊ธฐ ํ๋ฆ์ ์์ฐจ์ ์ผ๋ก ๋ช ํํ๊ฒ ๊ตฌ์ฑํ ์ ์์ด์.
2๏ธโฃ ๋ถํ์ํ await ์ฌ์ฉ ํผํ๊ธฐ
๋ชจ๋ Promise ์์ await๋ฅผ ๋ถ์ด๋ ๊ฒ์ด ํญ์ ์ข์ ๊ฒ์ ์๋์์.
ํนํ ์๋ก ์์กด์ฑ์ด ์๋ ์ฌ๋ฌ ๋น๋๊ธฐ ์์
์ ๋ณ๋ ฌ๋ก ์คํํ ์ ์๋ค๋ฉด, await๋ฅผ ๋จ์ฉํ๋ ๊ฒ์ ์ฑ๋ฅ ์ ํ๋ก ์ด์ด์ง ์ ์์ด์.
async function fetchSequentially() { console.time("์์ฐจ ์ฒ๋ฆฌ"); const user = await fetch("/api/user").then((res) => res.json()); const posts = await fetch("/api/posts").then((res) => res.json()); const comments = await fetch("/api/comments").then((res) => res.json()); console.timeEnd("์์ฐจ ์ฒ๋ฆฌ"); // ์: 300ms return { user, posts, comments }; } async function fetchInParallel() { console.time("๋ณ๋ ฌ ์ฒ๋ฆฌ"); const [user, posts, comments] = await Promise.all([ fetch("/api/user").then((res) => res.json()), fetch("/api/posts").then((res) => res.json()), fetch("/api/comments").then((res) => res.json()), ]); console.timeEnd("๋ณ๋ ฌ ์ฒ๋ฆฌ"); // ์: 100ms (๊ฐ์ฅ ์ค๋ ๊ฑธ๋ฆฌ๋ ์์ฒญ ์๊ฐ์ ๊ทผ์ ) return { user, posts, comments }; } // fetchSequentially(); // fetchInParallel();
์ ๋ณด์๋ก ๋ ๋ฆฝ์ ์ธ ๋น๋๊ธฐ ์์ ์
Promise.all๊ณผ ๊ฐ์ ๋ณ๋ ฌ ์ฒ๋ฆฌ ๋ฉ์๋๋ฅผ ํ์ฉํ์ฌawait๋ฅผ ํ ๋ฒ๋ง ์ฌ์ฉํ๋ฉด ์ ์ฒด ์คํ ์๊ฐ์ ํฌ๊ฒ ๋จ์ถํ ์ ์์ด์.์ด๋ ์ ํ๋ฆฌ์ผ์ด์ ์ ์๋ต์ฑ์ ํฅ์์ํค๋ ์ค์ํ ์ต์ ํ ๊ธฐ๋ฒ์ด์์.
๐ ๋ง๋ฌด๋ฆฌํ๋ฉฐ
์ค๋์ JavaScript์ Promise์ async/await์ ๋ํด ๊น์ด ์๊ฒ ์์๋ณด๋ ์๊ฐ์ ๊ฐ์ก์ด์.
๋น๋๊ธฐ ์ฒ๋ฆฌ์ ๊ธฐ๋ณธ ๊ฐ๋
๋ถํฐ Promise.all, Promise.allSettled, Promise.race, Promise.any์ ๊ฐ์ ๊ณ ๊ธ ํจํด, ๊ทธ๋ฆฌ๊ณ ์ค๋ฌด์์ ํํ ๋ฐ์ํ๋ ์ฃผ์์ฌํญ๊น์ง ํจ๊ป ์ดํด๋ณด์์ด์.
Promise์ async/await๋ ๋จ์ํ ๋ฌธ๋ฒ์ ์คํ์ด ์๋๋ผ, ๋น๋๊ธฐ ์ฝ๋๋ฅผ ๊ตฌ์กฐํํ๊ณ ๊ด๋ฆฌํ๋ ๊ฐ๋ ฅํ ๋๊ตฌ๋ค์ด์์.
์ด๋ค์ ์ฌ๋ฐ๋ฅด๊ฒ ์ดํดํ๊ณ ํ์ฉํ๋ฉด ๋์ฑ ๊ฒฌ๊ณ ํ๊ณ ํจ์จ์ ์ด๋ฉฐ ์ฝ๊ธฐ ์ฌ์ด ์ฝ๋๋ฅผ ์์ฑํ ์ ์์ ๊ฑฐ์์.
0๏ธโฃ ํต์ฌ ์์ฝ
Promise๋ ๋น๋๊ธฐ ์์ ์ ์ต์ข ์ฑ๊ณต ๋๋ ์คํจ๋ฅผ ๋ํ๋ด๋ ๊ฐ์ฒด์ด๋ฉฐ,Pending,Fulfilled,Rejected์ธ ๊ฐ์ง ์ํ๋ฅผ ๊ฐ์ ธ์.async/await๋Promise๋ฅผ ๊ธฐ๋ฐ์ผ๋ก ๋น๋๊ธฐ ์ฝ๋๋ฅผ ๋๊ธฐ ์ฝ๋์ฒ๋ผ ์ฝ๊ธฐ ์ฝ๊ฒ ๋ง๋ค์ด์ฃผ๋ ๋ฌธ๋ฒ์ด์์.Promise.all์ ๋ชจ๋Promise๊ฐ ์ฑ๊ณตํด์ผ ์ฑ๊ณตํ๊ณ , ํ๋๋ผ๋ ์คํจํ๋ฉด ์ ์ฒด๊ฐ ์คํจํ๋ "all or nothing" ์ ๋ต์ ์ ํฉํด์.Promise.allSettled๋ ๋ชจ๋Promise์ ์ฑ๊ณต/์คํจ ์ฌ๋ถ์ ๊ด๊ณ์์ด ๋ชจ๋ ๊ฒฐ๊ณผ๋ฅผ ํ์ธํด์ผ ํ ๋ ์ ์ฉํด์.Promise.race๋ ๊ฐ์ฅ ๋จผ์ ์ ์ฐฉ(์ฑ๊ณต ๋๋ ์คํจ)ํ๋Promise์ ๊ฒฐ๊ณผ๋ฅผ ๋ฐํํ๋ฉฐ, ํ์์์ ์ฒ๋ฆฌ ๋ฑ์ ํ์ฉ๋ผ์.Promise.any๋ ๊ฐ์ฅ ๋จผ์ ์ฑ๊ณตํ๋Promise์ ๊ฒฐ๊ณผ๋ฅผ ๋ฐํํ๋ฉฐ, ์ฌ๋ฌ ์์ค ์ค ํ๋๋ง ์ฑ๊ณตํ๋ฉด ๋ ๋ ์ ์ฉํด์.async/await์์๋try...catch๋ก,Promise์ฒด์ธ์์๋catch()๋ก ์๋ฌ๋ฅผ ํจ๊ณผ์ ์ผ๋ก ์ฒ๋ฆฌํ ์ ์์ด์.await๋๋ฝ, ์๋ชป๋Promise์ฒด์ด๋, ๋ถํ์ํawait์ฌ์ฉ์ ํผํด์ผ ํ ํํ ์ค์๋ค์ด์์.
1๏ธโฃ ๋ค์ ์คํ
์ค๋ ๋ฐฐ์ด ๋ด์ฉ๋ค์ ๋ฐํ์ผ๋ก ์ค์ ํ๋ก์ ํธ์ ์ ์ฉํด ๋ณด์๋ ๊ฒ์ ์ถ์ฒํด์.
ํนํ ๊ธฐ์กด์ ์์ฑ๋ ๋น๋๊ธฐ ์ฝ๋๋ฅผ ๋ฆฌํฉํ ๋งํ๋ฉด์ ์ค๋ ๋ค๋ฃฌ ํจํด๋ค์ ์ ์ฉํด ๋ณด๋ฉด ๋์ฑ ๊น์ด ์๋ ์ดํด๋ฅผ ์ป์ผ์ค ์ ์์ ๊ฑฐ์์.
๋ ๋์๊ฐ, RxJS์ ๊ฐ์ ๋ฐ์ํ ํ๋ก๊ทธ๋๋ฐ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ ๋น๋๊ธฐ ์ ๋๋ ์ดํฐ ๋ฑ ๋ ๋ณต์กํ ๋น๋๊ธฐ ํจํด๋ค๋ ํ๊ตฌํด ๋ณด์๋ ๊ฒ๋ ์ข๊ฒ ์ด์.
๊ถ๊ธํ ์ ์ด๋ ๋ ๋ค๋ฃจ์์ผ๋ฉด ํ๋ ์ฃผ์ ๊ฐ ์๋ค๋ฉด ์ธ์ ๋ ์ง ์๋ ค์ฃผ์ธ์.
๋ค์ ํฌ์คํ
์์ ๋ ์ ์ตํ ๋ด์ฉ์ผ๋ก ์ฐพ์๋ต๊ฒ์!
๐ฎ ์ฐธ๊ณ
- MDN Web Docs: Promise
- MDN Web Docs: async function
- MDN Web Docs: await
- MDN Web Docs: Promise.all()
- MDN Web Docs: Promise.allSettled()
- MDN Web Docs: Promise.race()
- MDN Web Docs: Promise.any()
์ฐ๊ด๋ ํฌ์คํธ
- ๋จ์ด: 1,996๊ฐ22๋ถ
[๐ค] Next.js App Router์ Auth.js (NextAuth.js) ์ฐ๋ ๊ฐ์ด๋: ์๋ฒ ๋ฐ ํด๋ผ์ด์ธํธ ์ปดํฌ๋ํธ ์ธ์ ๊ด๋ฆฌ
Next.js 14+ App Router ํ๊ฒฝ์์ Auth.js (๊ตฌ NextAuth.js)๋ฅผ ์ฌ์ฉํ์ฌ ์์ ํ๊ณ ํจ์จ์ ์ธ ์ธ์ฆ ์์คํ ์ ๊ตฌ์ถํ๋ ๋ฐฉ๋ฒ์ ์์ธํ ์์๋ด์. ์๋ฒ ์ปดํฌ๋ํธ์ ํด๋ผ์ด์ธํธ ์ปดํฌ๋ํธ์์ ์ธ์ ์ ๊ด๋ฆฌํ๊ณ ๋ณดํธํ๋ ์ค์ฉ์ ์ธ ํ๊ณผ ์ฝ๋ ์์๋ฅผ ์ ๊ณตํด์.
- ๋จ์ด: 2,440๊ฐ26๋ถ
[๐ค] Next.js App Router์์ Zod์ TypeScript๋ก API ์ ํจ์ฑ ๊ฒ์ฌ ๋ฐ ํ์ ์์ ์ฑ ํ๋ณด
Next.js App Router ํ๊ฒฝ์์ Zod์ TypeScript๋ฅผ ํ์ฉํ์ฌ API ์์ฒญ ๋ฐ์ดํฐ์ ๋ฐํ์ ์ ํจ์ฑ์ ๊ฒ์ฌํ๊ณ , ๋์์ ๊ฐ๋ ฅํ ํ์ ์์ ์ฑ์ ํ๋ณดํ๋ ์ค์ฉ์ ์ธ ๋ฐฉ๋ฒ์ ์์ธํ ์๋ ค๋๋ ค์.
- ๋จ์ด: 1,646๊ฐ18๋ถ
[๐ค] JavaScript ๊ตฌ์กฐ ๋ถํด ํ ๋น ํจํด ์ฌํ: ๋ค์ํ ํ์ฉ๊ณผ ์ค์ ํ
JavaScript ๊ตฌ์กฐ ๋ถํด ํ ๋น(Destructuring Assignment)์ ๊ธฐ๋ณธ๋ถํฐ ์ค์ฒฉ ๊ฐ์ฒด, ๊ธฐ๋ณธ๊ฐ, Rest ๋ฌธ๋ฒ, ํจ์ ์ธ์ ํ์ฉ๊น์ง ๋ค์ํ ํจํด๊ณผ ์ค์ ํ์ ํตํด ์ฝ๋๋ฅผ ๋ ๊ฐ๊ฒฐํ๊ณ ํจ์จ์ ์ผ๋ก ์์ฑํ๋ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์.
- ๋จ์ด: 1,841๊ฐ23๋ถ
[๐ค] ์น ์ฑ๋ฅ ์ต์ ํ: ๋ธ๋ผ์ฐ์ ๊ฐ๋ฐ์ ๋๊ตฌ๋ก Core Web Vitals ์ง๋จํ๊ณ ๊ฐ์ ํ๊ธฐ
๋ธ๋ผ์ฐ์ ๊ฐ๋ฐ์ ๋๊ตฌ(DevTools)์ Lighthouse, Performance, Network ํญ์ ํ์ฉํ์ฌ Core Web Vitals๋ฅผ ์ง๋จํ๊ณ ์ค์ ์น ์ ํ๋ฆฌ์ผ์ด์ ์ ์ฑ๋ฅ์ ๊ฐ์ ํ๋ ์ค์ง์ ์ธ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์.
- ๋จ์ด: 1,957๊ฐ22๋ถ
[๐ค] TypeScript ํจ์ ์ค๋ฒ๋ก๋ฉ: ์๊ทธ๋์ฒ์ ๊ตฌํ์ฒด๋ฅผ ์๋ฒฝํ๊ฒ ์ดํดํ๊ณ ํ์ฉํ๊ธฐ
TypeScript์์ ํจ์ ์ค๋ฒ๋ก๋ฉ์ ์ฌ์ฉํ์ฌ ๋ค์ํ ํ์ ์ ์ธ์๋ฅผ ์ฒ๋ฆฌํ๊ณ , ์ฝ๋์ ๊ฐ๋ ์ฑ ๋ฐ ์์ ์ฑ์ ๋์ด๋ ๋ฐฉ๋ฒ์ ์๊ทธ๋์ฒ์ ๊ตฌํ์ฒด์ ๊ด๊ณ๋ฅผ ์ค์ฌ์ผ๋ก ์์ธํ ์ค๋ช ํด ๋๋ ค์.
- ๋จ์ด: 1,709๊ฐ19๋ถ
[๐ค] JavaScript requestIdleCallback๊ณผ isInputPending์ผ๋ก UI ๋ฐ์์ฑ ๊ฐ์ ํ๊ธฐ: ๋ฉ์ธ ์ค๋ ๋ ์ต์ ํ ์ฌํ
JavaScript์ `requestIdleCallback`๊ณผ `isInputPending` API๋ฅผ ํ์ฉํ์ฌ ๋ฉ์ธ ์ค๋ ๋ ๋ธ๋กํน์ ์ค์ด๊ณ UI ๋ฐ์์ฑ์ ๊ทน๋ํํ๋ ์ค์ง์ ์ธ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์. ์ฌ์ฉ์ ๊ฒฝํ์ ์ํ ์น ์ฑ๋ฅ ์ต์ ํ ์ ๋ต์ ์๋ดํด ๋๋ ค์.
- ๋จ์ด: 2,149๊ฐ23๋ถ
[๐ค] Next.js App Router์ `generateMetadata`์ `generateViewport` ์ฌํ: SEO ๋ฐ PWA ์ต์ ํ ์ ๋ต
Next.js 14/13 App Router์์ `generateMetadata`์ `generateViewport` ํจ์๋ฅผ ํ์ฉํ์ฌ ๋์ SEO ๋ฉํํ๊ทธ์ PWA ์ค์ ์ ์ต์ ํํ๋ ์ฌํ ์ ๋ต์ ๋ค๋ฃน๋๋ค. ์ค๋ฌด ์์์ ํจ๊ป ์น์ฌ์ดํธ์ ๊ฒ์ ์์ง ๋ ธ์ถ ๋ฐ ์ฌ์ฉ์ ๊ฒฝํ์ ๊ทน๋ํํ๋ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์.
- ๋จ์ด: 1,788๊ฐ19๋ถ
[๐ค] CSS ๋ณ์(Custom Properties) ์ฌํ: ๋์ ํ ๋ง, ์ค์ฝํ, JavaScript ํ์ฉ ์ ๋ต
CSS ๋ณ์(Custom Properties)์ ๊ธฐ๋ณธ๋ถํฐ ๋์ ํ ๋ง ๊ตฌํ, ์ค์ฝํ ์ ๋ต, ๊ทธ๋ฆฌ๊ณ JavaScript์์ ์ํธ์์ฉ๊น์ง ์ค๋ฌด์ ๋ฐ๋ก ์ ์ฉํ ์ ์๋ ์ฌํ ํ์ฉ๋ฒ์ ๋ธ๋ฃจ๊ฐ ์์ธํ ์๋ ค๋๋ ค์.
- ๋จ์ด: 1,293๊ฐ15๋ถ
[๐ค] Next.js App Router Edge Runtime: ๊ธ๋ก๋ฒ ์ฑ๋ฅ๊ณผ ๋น์ฉ ํจ์จ์ฑ ๊ทน๋ํ ์ ๋ต
Next.js App Router์์ Edge Runtime์ ํ์ฉํด Route Handler ์ฑ๋ฅ์ ๊ทน๋ํํ๊ณ , ๊ธ๋ก๋ฒ ์ฌ์ฉ์์๊ฒ ๋น ๋ฅธ ์๋ต ์๋๋ฅผ ์ ๊ณตํ๋ฉฐ, ๋น์ฉ ํจ์จ์ ์ผ๋ก ์ ํ๋ฆฌ์ผ์ด์ ์ ์ด์ํ๋ ์ค์ ์ ๋ต์ ์์ธํ ๋ค๋ค์. Edge Runtime์ ์ฅ์ ๊ณผ ์ ํ์ฌํญ, ๊ทธ๋ฆฌ๊ณ ์ค์ ์ฝ๋ ์์๋ฅผ ํตํด ํจ๊ณผ์ ์ธ ์ ์ฉ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์.
- ๋จ์ด: 1,980๊ฐ25๋ถ
[๐ค] Next.js/React์์ ํด๋ฆฐ ์ํคํ ์ฒ์ ์์กด์ฑ ์ญ์ ์์น ์ ์ฉํ๊ธฐ: ํ์ฅ ๊ฐ๋ฅํ ํ๋ฐํธ์๋ ๋ง๋ค๊ธฐ
Next.js์ React ํ๋ก์ ํธ์์ ํด๋ฆฐ ์ํคํ ์ฒ์ ์์กด์ฑ ์ญ์ ์์น(DIP)์ ์ ์ฉํ๋ ๋ฐฉ๋ฒ์ ์ฌ์ธต์ ์ผ๋ก ๋ค๋ฃจ์ด์. ๋๋ฉ์ธ, ์ ์ค์ผ์ด์ค, ์ธํ๋ผ ๊ณ์ธต์ ๋ถ๋ฆฌํ์ฌ ์ ์ง๋ณด์์ฑ๊ณผ ํ ์คํธ ์ฉ์ด์ฑ์ ๊ทน๋ํํ๋ ์ค์ฉ์ ์ธ ์ ๋ต์ ๋ฐฐ์๋ณด์ธ์.
- ๋จ์ด: 1,733๊ฐ19๋ถ
[๐ค] Jotai๋ก Next.js App Router ํด๋ผ์ด์ธํธ ์ํ ๊ด๋ฆฌ ์ฌํ: ์ํฐ ํจํด๊ณผ ์ต์ ํ
Next.js App Router ํ๊ฒฝ์์ Jotai ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ํ์ฉํ์ฌ ํจ์จ์ ์ธ ํด๋ผ์ด์ธํธ ์ํ ๊ด๋ฆฌ ์ ๋ต์ ์์๋ณด์ธ์. ์ํฐ ํจํด์ ์ด์ฉํ ์ํ ๋ถ๋ฆฌ, ์ต์ ํ ๊ธฐ๋ฒ, ๊ทธ๋ฆฌ๊ณ ์ค์ฉ์ ์ธ ์ฝ๋ ์์๋ฅผ ํตํด ๋ณต์กํ UI ์ํ๋ฅผ ํจ๊ณผ์ ์ผ๋ก ๋ค๋ฃจ๋ ๋ฐฉ๋ฒ์ ๋ธ๋ฃจ๊ฐ ์์ธํ ์๋ ค๋๋ ค์.
- ๋จ์ด: 1,720๊ฐ18๋ถ
[๐ค] Next.js App Router ์ ์ญ ์ํ ๊ด๋ฆฌ: Server/Client Components ์๋ฒฝ ๊ฐ์ด๋
Next.js App Router ํ๊ฒฝ์์ Server Components์ Client Components ๊ฐ์ ์ ์ญ ์ํ๋ฅผ ํจ๊ณผ์ ์ผ๋ก ๊ด๋ฆฌํ๋ ๋ฐฉ๋ฒ์ ํ๊ตฌํด์. Zustand, Recoil, Context API ๋ฑ ๋ค์ํ ์๋ฃจ์ ๊ณผ ์ค์ฉ์ ์ธ ์ ๋ต์ ํตํด ๋ณต์กํ ์ํ ๊ด๋ฆฌ ๋ฌธ์ ๋ฅผ ํด๊ฒฐํด ๋ณด์ธ์.
- ๋จ์ด: 1,787๊ฐ20๋ถ
[๐ค] Atomic Design ํจํด์ผ๋ก React/Next.js ์ปดํฌ๋ํธ ์ํคํ ์ฒ ํ์ฅ์ฑ ๋์ด๊ธฐ
React ๋ฐ Next.js ํ๋ก์ ํธ์์ Atomic Design ํจํด์ ์ ์ฉํ์ฌ ์ปดํฌ๋ํธ์ ์ฌ์ฌ์ฉ์ฑ๊ณผ ํ์ฅ์ฑ์ ๊ทน๋ํํ๋ ๋ฐฉ๋ฒ์ ์ค๋ฌด ์์์ ํจ๊ป ์์ธํ ์๋ ค๋๋ ค์. ํจ์จ์ ์ธ ํ๋ก ํธ์๋ ์ํคํ ์ฒ๋ฅผ ๊ตฌ์ถํด ๋ณด์ธ์.
- ๋จ์ด: 2,160๊ฐ24๋ถ
[๐ค] ์๋ฐ์คํฌ๋ฆฝํธ Web Workers: UI ๋ธ๋กํน ์์ด ๋ฌด๊ฑฐ์ด ์์ ์ฒ๋ฆฌํ๊ธฐ
ํ๋ก ํธ์๋์์ CPU ์ง์ฝ์ ์ธ ์์ ์ UI ๋ธ๋กํน์ ํํ ๋ฌธ์ ์์. ์๋ฐ์คํฌ๋ฆฝํธ Web Workers๋ก ๋ฉ์ธ ์ค๋ ๋๋ฅผ ํด๋ฐฉํ๊ณ ์ฌ์ฉ์ ๊ฒฝํ์ ์ต์ ํํ๋ ์ค์ฉ์ ์ธ ๋ฐฉ๋ฒ์ ์ฝ๋ ์์์ ํจ๊ป ์์๋ณด์ธ์. ๋ณต์กํ ๊ณ์ฐ, ๋ฐ์ดํฐ ์ฒ๋ฆฌ ๋ฑ์ ๋ฐฑ๊ทธ๋ผ์ด๋์์ ์์ ํ๊ฒ ์คํํ๋ ๋ ธํ์ฐ๋ฅผ ๊ณต์ ํด์.
- ๋จ์ด: 1,822๊ฐ20๋ถ
[๐ค] TypeScript `as const` ์ฌ์ธต ๋ถ์: ๋ฆฌํฐ๋ด ํ์ ์ถ๋ก ๊ณผ ๋ถ๋ณ์ฑ ํ์ฉ ์ ๋ต
TypeScript์ `as const`๋ฅผ ํ์ฉํ์ฌ ๋ฆฌํฐ๋ด ํ์ ์ถ๋ก ์ ๊ฐํํ๊ณ , ๋ฐฐ์ด๊ณผ ๊ฐ์ฒด์ ๋ถ๋ณ์ฑ์ ์์ ํ๊ฒ ํ๋ณดํ๋ ์ค์ฉ์ ์ธ ๋ฐฉ๋ฒ์ ์์ธํ ์์๋ด์. ์ค๋ฌด ์ฝ๋์ ํ์ ์์ ์ฑ๊ณผ ๊ฐ๋ ์ฑ์ ๋์ฌ๋ณด์ธ์.
- ๋จ์ด: 1,486๊ฐ17๋ถ
[๐ค] Next.js @next/font๋ฅผ ํ์ฉํ ํฐํธ ์ต์ ํ ๋ฐ CLS ๊ฐ์ ์ ๋ต
Next.js ์ ํ๋ฆฌ์ผ์ด์ ์์ ํฐํธ ๋ก๋ฉ์ผ๋ก ์ธํ Cumulative Layout Shift(CLS)๋ฅผ ์ต์ํํ๊ณ ์น ์ฑ๋ฅ์ ๊ทน๋ํํ๋ @next/font ์ฌ์ฉ๋ฒ๊ณผ ์ค์ฉ์ ์ธ ์ต์ ํ ์ ๋ต์ ์์ธํ ์์๋ด์.
- ๋จ์ด: 1,914๊ฐ21๋ถ
[๐ค] Next.js App Router ๊ตญ์ ํ (i18n) ์๋ฒฝ ๊ฐ์ด๋: ๋ค๊ตญ์ด ์ง์ ๊ตฌํ ์ ๋ต
Next.js 14+ App Router ํ๊ฒฝ์์ ๊ตญ์ ํ(i18n)๋ฅผ ํจ๊ณผ์ ์ผ๋ก ๊ตฌํํ๋ ๋ฐฉ๋ฒ์ ์์ธํ ์๋ ค๋๋ ค์. ๋ฏธ๋ค์จ์ด๋ฅผ ํ์ฉํ ์ธ์ด ๊ฐ์ง๋ถํฐ ํด๋ผ์ด์ธํธ/์๋ฒ ์ปดํฌ๋ํธ์์์ ๋ค๊ตญ์ด ํ ์คํธ ์ฒ๋ฆฌ๊น์ง, ์ค๋ฌด์ ๋ฐ๋ก ์ ์ฉํ ์ ์๋ ๋ค๊ตญ์ด ์ง์ ์ ๋ต์ ๋ธ๋ฃจ๊ฐ ์๋ดํด ๋๋ฆด๊ฒ์.
- ๋จ์ด: 1,328๊ฐ16๋ถ
[์น ์ฑ๋ฅ ์ต์ ํ] Interaction to Next Paint (INP) ๊ฐ์ : ์ฌ์ฉ์ ๊ฒฝํ์ ๊ทน๋ํํ๋ ์ค์ ์ ๋ต
์๋ก์ด Core Web Vitals ์งํ, Interaction to Next Paint(INP)๋ฅผ ๊น์ด ์ดํดํ๊ณ ์ค์ ์น ์ ํ๋ฆฌ์ผ์ด์ ์์ INP ์ ์๋ฅผ ๊ฐ์ ํ๋ ์ค์ฉ์ ์ธ ์ต์ ํ ์ ๋ต๋ค์ ๋ธ๋ฃจ๊ฐ ์์ธํ ์๋ ค๋๋ ค์. ์ฌ์ฉ์ ์ธํฐ๋์ ์๋ต์ฑ์ ๋์ฌ ์ง์ ํ ์ฌ์ฉ์ ๊ฒฝํ์ ๋ง๋ค์ด ๋ณด์ธ์.
๋จ์ด: 1,668๊ฐ20๋ถ[๐ค] JavaScript Decorator ์ฌ์ธต ๋ถ์: ๋ฉํํ๋ก๊ทธ๋๋ฐ๊ณผ ํ์ ์คํฌ๋ฆฝํธ ํ์ฉ ์ ๋ต
JavaScript Decorator์ ๊ฐ๋ ๋ถํฐ ํ์ ์คํฌ๋ฆฝํธ์์์ ํ์ฉ, ๊ทธ๋ฆฌ๊ณ ๋ฏธ๋ ํ์คํ ๋ฐฉํฅ๊น์ง ์ฌ์ธต์ ์ผ๋ก ๋ค๋ฃจ๋ฉฐ, ์ค๋ฌด์์ ์ฝ๋ ์ฌ์ฌ์ฉ์ฑ๊ณผ ๊ฐ๋ ์ฑ์ ๋์ด๋ ๋ฉํํ๋ก๊ทธ๋๋ฐ ํจํด์ ํ์ตํด ๋ณด์ธ์.
- ๋จ์ด: 1,421๊ฐ16๋ถ
[๐ค] TypeScript 5.2+ `using` ์ ์ธ์ผ๋ก ์์ ์๋ ๊ด๋ฆฌํ๊ธฐ: ๊น๋ํ ๋ฆฌ์์ค ์ ๋ฆฌ์ ์์
TypeScript 5.2์์ ๋์ ๋ `using` ์ ์ธ์ ํ์ฉํ์ฌ ํ์ผ ํธ๋ค, ๋ฐ์ดํฐ๋ฒ ์ด์ค ์ฐ๊ฒฐ ๋ฑ ๋ค์ํ ์์์ ์๋์ผ๋ก ๊ด๋ฆฌํ๊ณ ๊น๋ํ๊ฒ ์ ๋ฆฌํ๋ ๋ฐฉ๋ฒ์ ์ค์ฉ์ ์ธ ์์ ์ ํจ๊ป ์์ธํ ์์๋ด์. ๋ ์ด์ `try...finally` ๋ธ๋ก์ผ๋ก ๋ณต์กํ๊ฒ ์์์ ํด์ ํ ํ์ ์์ด์.
- ๋จ์ด: 2,006๊ฐ22๋ถ
[๐ค] Feature-Sliced Design (FSD)์ผ๋ก ํ์ฅ ๊ฐ๋ฅํ ํ๋ก ํธ์๋ ์ํคํ ์ฒ ๊ตฌ์ถํ๊ธฐ
Feature-Sliced Design (FSD)์ ๋๊ท๋ชจ ํ๋ก ํธ์๋ ์ ํ๋ฆฌ์ผ์ด์ ์ ํ์ฅ์ฑ๊ณผ ์ ์ง๋ณด์์ฑ์ ๋์ด๋ ๊ฐ๋ ฅํ ์ํคํ ์ฒ ํจํด์ด์์. FSD์ ํต์ฌ ์์น, ๋ ์ด์ด ๊ตฌ์กฐ, ์ค๋ฌด ์ ์ฉ ๋ฐฉ๋ฒ์ ์ฝ๋ ์์์ ํจ๊ป ์์ธํ ์์๋ด์.
- ๋จ์ด: 1,781๊ฐ21๋ถ
[๐ค] TypeScript `moduleResolution: 'bundler'`์ `verbatimModuleSyntax` ์๋ฒฝ ์ดํด: ๋ชจ๋ ๋ฒ๋ค๋ฌ ์๋์ ํ์ ์คํฌ๋ฆฝํธ ์ค์ ์ต์ ํ
๋ชจ๋ ์๋ฐ์คํฌ๋ฆฝํธ ํ๋ก์ ํธ์์ TypeScript์ `moduleResolution: 'bundler'`์ `verbatimModuleSyntax` ์ต์ ์ด ์ ์ค์ํ์ง, ์ด๋ป๊ฒ ์ฌ๋ฐ๋ฅด๊ฒ ์ค์ ํ๊ณ ํ์ฉํ์ฌ ๊ฐ๋ฐ ๊ฒฝํ๊ณผ ๋ฒ๋ค๋ง ํจ์จ์ ๊ทน๋ํํ๋์ง ์ค๋ฌด ์์์ ํจ๊ป ์์ธํ ์์๋ด์.
- ๋จ์ด: 1,908๊ฐ24๋ถ
[๐ค] ๋ชจ๋ ธ๋ ํฌ ์ํคํ ์ฒ: ํจ์จ์ ์ธ ๋๊ท๋ชจ ํ๋ก์ ํธ ์ค๊ณ๋ฅผ ์ํ ์ค์ ๊ฐ์ด๋
๋๊ท๋ชจ ์น ํ๋ก์ ํธ์์ ๋ชจ๋ ธ๋ ํฌ๋ฅผ ๋์ ํ๋ ์ด์ , ์ฅ๋จ์ , ๊ทธ๋ฆฌ๊ณ ํจ๊ณผ์ ์ธ ๊ตฌ์กฐ ์ค๊ณ ๋ฐ ์ค๋ฌด ์ ์ฉ ์ ๋ต์ ๋ธ๋ฃจ๊ฐ ์์ธํ ์๋ ค๋๋ ค์. ํ ์์ฐ์ฑ ํฅ์๊ณผ ์ฝ๋ ์ฌ์ฌ์ฉ์ฑ์ ๊ทน๋ํํ๋ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์.
๋จ์ด: 2,188๊ฐ26๋ถ[๐ค] ์น ์ฑ๋ฅ ์ต์ ํ: ๋ฆฌํ๋ก์ฐ์ ๋ฆฌํ์ธํธ ์ต์ํ ์ ๋ต
๋ธ๋ผ์ฐ์ ๋ ๋๋ง ๊ณผ์ ์ ํต์ฌ์ธ ๋ฆฌํ๋ก์ฐ(Reflow)์ ๋ฆฌํ์ธํธ(Repaint)์ ๋ฐ์ ์๋ฆฌ๋ฅผ ์ดํดํ๊ณ , ์ด๋ฅผ ํจ๊ณผ์ ์ผ๋ก ์ค์ฌ ์น ์ ํ๋ฆฌ์ผ์ด์ ์ ์ฑ๋ฅ์ ๊ทน๋ํํ๋ ์ค์ง์ ์ธ ์ ๋ต๊ณผ ์ฝ๋ ์์๋ฅผ ์์ธํ ์๋ ค๋๋ ค์.
- ๋จ์ด: 1,903๊ฐ19๋ถ
[๐ค] Vitest์ React Testing Library๋ก Next.js ์ปดํฌ๋ํธ ์๋ฒฝ ํ ์คํธํ๊ธฐ
Next.js ํ๋ก์ ํธ์์ Vitest์ React Testing Library๋ฅผ ํ์ฉํ์ฌ UI ์ปดํฌ๋ํธ๋ฅผ ํจ๊ณผ์ ์ผ๋ก ํ ์คํธํ๋ ๋ฐฉ๋ฒ์ ์ค๋ฌด ์์ ์ ํจ๊ป ์์ธํ ์์๋ด์. ์ค์ ๋ถํฐ Mocking, ์ด๋ฒคํธ ์๋ฎฌ๋ ์ด์ ๊น์ง, ๊ฒฌ๊ณ ํ ์ ํ๋ฆฌ์ผ์ด์ ๊ฐ๋ฐ์ ์ํ ํ ์คํธ ์ ๋ต์ ์ตํ๋ณด์ธ์.
- ๋จ์ด: 1,542๊ฐ20๋ถ
[๐ค] Tailwind CSS v4 ์ถ์: ๊ฐ๋ฐ์์๊ฒ ์ฐพ์์ฌ ๋ณํ์ ์ต์ ํ ์ ๋ต
Tailwind CSS v4์ ์ฃผ์ ๋ณ๊ฒฝ์ฌํญ๊ณผ ์๋ก์ด ๊ธฐ๋ฅ๋ค์ ๊น์ด ์๊ฒ ๋ถ์ํ๊ณ , ์ค๋ฌด์์ ํจ์จ์ ์ผ๋ก ์ ์ฉํ๋ฉฐ ์ฑ๋ฅ์ ์ต์ ํํ๋ ์ ๋ต์ ๋ธ๋ฃจ๊ฐ ์๋ ค๋๋ ค์.
- ๋จ์ด: 1,566๊ฐ20๋ถ
[๐ค] Next.js Dockerfile ์ต์ ํ: ํ๋ก๋์ ๋ฐฐํฌ๋ฅผ ์ํ ์๋ฒฝ ๊ฐ์ด๋
Next.js ์ ํ๋ฆฌ์ผ์ด์ ์ Docker ์ปจํ ์ด๋๋ก ํจ์จ์ ์ผ๋ก ๋ฐฐํฌํ๋ ๋ฐฉ๋ฒ์ ์์๋ณด์ธ์. ๋ฉํฐ์คํ ์ด์ง ๋น๋, ์บ์ฑ ์ ๋ต, ๋ณด์ ์ค์ ๋ฑ ํ๋ก๋์ ํ๊ฒฝ์ ์ต์ ํ๋ Dockerfile ์์ฑ ๊ฐ์ด๋๋ฅผ ์ ๊ณตํด ๋๋ ค์.
- ๋จ์ด: 2,177๊ฐ24๋ถ
[๐ค] CSS Grid ์ฌํ ๊ฐ์ด๋: ์ค์ ๋ ์ด์์ ํจํด๊ณผ ๋ฐ์ํ ๋์์ธ ์ ๋ต
CSS Grid๋ ๊ฐ๋ ฅํ 2์ฐจ์ ๋ ์ด์์ ์์คํ ์ด์์. ์ด ๊ฐ์ด๋์์ Grid์ ํต์ฌ ๊ฐ๋ ๋ถํฐ ์ค์ ๋ ์ด์์ ํจํด, ๋ฐ์ํ ๋์์ธ ์ ๋ต๊น์ง ์ฌ์ธต์ ์ผ๋ก ๋ค๋ฃจ์ด ์ค๋ฌด์ ๋ฐ๋ก ์ ์ฉํ ์ ์๋๋ก ๋์๋๋ ค์.
- ๋จ์ด: 1,718๊ฐ19๋ถ
[๐ค] TypeScript ํ ํ๋ฆฟ ๋ฆฌํฐ๋ด ํ์ : ๋ฌธ์์ด ํ์ ์ ๋ง๋ฒ์ฌ๋ก ๋ณ์ ํ๊ธฐ
TypeScript์ ํ ํ๋ฆฟ ๋ฆฌํฐ๋ด ํ์ ์ ํ์ฉํ์ฌ ๋ณต์กํ ๋ฌธ์์ด ํจํด์ ์์ ํ๊ฒ ํ์ ์ถ๋ก ํ๊ณ , ๊ฐ๋ ฅํ ์ ํธ๋ฆฌํฐ ํ์ ์ ๋ง๋๋ ๋ฐฉ๋ฒ์ ์ค๋ฌด ์์ ์ ํจ๊ป ์์ธํ ์์๋ด์. ํ์ ์์ ์ฑ์ ํ ๋จ๊ณ ๋์ฌ ๊ฐ๋ฐ ๊ฒฝํ์ ๊ฐ์ ํด ๋ณด์ธ์.
- ๋จ์ด: 1,918๊ฐ23๋ถ
[๐ค] JavaScript WeakMap๊ณผ WeakSet: ๋ฉ๋ชจ๋ฆฌ ๋์ ๋ฐฉ์ง์ ์ต์ ํ ์ ๋ต
JavaScript์์ WeakMap๊ณผ WeakSet์ ํ์ฉํ์ฌ ๋ฉ๋ชจ๋ฆฌ ๋์๋ฅผ ๋ฐฉ์งํ๊ณ ์ฑ๋ฅ์ ์ต์ ํํ๋ ๋ฐฉ๋ฒ์ ์ค์ฉ์ ์ธ ์์์ ํจ๊ป ์์ธํ ์์๋ด์. ๊ฐ๋น์ง ์ปฌ๋ ์ ๋์ ์๋ฆฌ์ ํจ๊ป ๊ฐ์ฒด ์ฐธ์กฐ ๊ด๋ฆฌ์ ์ค์์ฑ์ ์ดํดํ๊ณ , ์ค์ ํ๋ก์ ํธ์ ์ ์ฉํ๋ ์ ๋ต์ ๋ฐฐ์๋ด์.
- ๋จ์ด: 1,455๊ฐ17๋ถ
[๐ค] Next.js/React ํ๋ก์ ํธ๋ฅผ ์ํ ESLint & Prettier ์ค์ ์๋ฒฝ ๊ฐ์ด๋
๋ณต์กํ Next.js ๋ฐ React ํ๋ก์ ํธ์์ ์ผ๊ด๋ ์ฝ๋ ์คํ์ผ๊ณผ ํ์ง์ ์ ์งํ๋ ESLint์ Prettier ์ค์ ๋ฐฉ๋ฒ์ ์์ธํ ์๋ ค๋๋ ค์. ํ ๊ฐ๋ฐ ํ๊ฒฝ์ ์ต์ ํ๋ ์ค์ ์ผ๋ก ๊ฐ๋ฐ ํจ์จ์ ๋์ฌ๋ณด์ธ์.
- ๋จ์ด: 2,022๊ฐ25๋ถ
[๐ค] JavaScript์ ํต์ฌ: ํ๋กํ ํ์ ์ฒด์ธ ์๋ฒฝ ์ดํด์ ํ์ฉ ์ ๋ต
JavaScript์ ์ฌ์ฅ๋ถ, ํ๋กํ ํ์ ์ฒด์ธ์ ๋์ ์๋ฆฌ๋ฅผ ๊น์ด ํ๊ณ ๋ค์ด ๊ฐ์ฒด ์งํฅ ํ๋ก๊ทธ๋๋ฐ๊ณผ ์์์ ์๋ฒฝํ๊ฒ ์ดํดํ๊ณ ์ค๋ฌด์ ์ ์ฉํ๋ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์. ์ฑ๋ฅ ์ต์ ํ ํ๋ ํจ๊ป ๋ค๋ค์.
- ๋จ์ด: 2,118๊ฐ22๋ถ
[๐ค] React ์ปค์คํ ํ : ์ฌ์ฌ์ฉ์ฑ ๋์ด๋ ์ค๊ณ ์์น๊ณผ ํ ์คํธ ์ ๋ต
React ์ ํ๋ฆฌ์ผ์ด์ ์ ์ฌ์ฌ์ฉ์ฑ๊ณผ ์ ์ง๋ณด์์ฑ์ ๊ทน๋ํํ๋ ์ปค์คํ ํ ์ค๊ณ ์์น, ์ค์ฉ์ ์ธ ํจํด, ๊ทธ๋ฆฌ๊ณ ๊ฒฌ๊ณ ํ ํ ์คํธ ์ ๋ต์ ์๋์ด ๊ฐ๋ฐ์์ ๊ด์ ์์ ์์ธํ ์ค๋ช ํด ๋๋ ค์.
- ๋จ์ด: 2,107๊ฐ23๋ถ
[๐ค] React useRef ํ ์ฌ์ธต ๋ถ์: DOM ๋์ด์ ์ค์ ํ์ฉ ์ ๋ต
React useRef ํ ์ ๊ธฐ๋ณธ ์๋ฆฌ๋ถํฐ DOM ์์ ์ง์ ์ ์ด, ์ปดํฌ๋ํธ ๋ผ์ดํ์ฌ์ดํด ๊ด๋ฆฌ, ๊ทธ๋ฆฌ๊ณ ์ฑ๋ฅ ์ต์ ํ๋ฅผ ์ํ ๋ค์ํ ์ค์ ํ์ฉ ์ ๋ต๊น์ง ์ฌ์ธต์ ์ผ๋ก ๋ค๋ค์. ์ด์ค๊ธ ๊ฐ๋ฐ์๋ฅผ ์ํ useRef ์๋ฒฝ ๊ฐ์ด๋.
- ๋จ์ด: 1,762๊ฐ19๋ถ
[๐ค] Next.js 14/15์์ ๋์ OG ์ด๋ฏธ์ง ์์ฑ: ImageResponse ์๋ฒฝ ๊ฐ์ด๋
Next.js App Router ํ๊ฒฝ์์ ImageResponse๋ฅผ ํ์ฉํ์ฌ ๋์ OG ์ด๋ฏธ์ง๋ฅผ ํจ์จ์ ์ผ๋ก ์์ฑํ๋ ๋ฐฉ๋ฒ์ ์์๋ณด์ธ์. SEO์ ์์ ๊ณต์ ์ต์ ํ๋ฅผ ์ํ ์ค์ ๊ฐ์ด๋์ ๋๋ค.
- ๋จ์ด: 1,481๊ฐ18๋ถ
[๐ค] Git ๋ธ๋์น ์ ๋ต: Git Flow vs GitHub Flow, ์ค๋ฌด์์ ์ด๋ป๊ฒ ์ ํํ๊ณ ์ด์ํ ๊น์?
๊ฐ๋ฐํ์ ํจ์จ์ ์ธ ํ์ ์ ์ํ Git ๋ธ๋์น ์ ๋ต์ ๊ณ ๋ฏผํ๊ณ ๊ณ์ ๊ฐ์? Git Flow์ GitHub Flow์ ํต์ฌ ๊ฐ๋ ๋ถํฐ ์ฅ๋จ์ , ๊ทธ๋ฆฌ๊ณ ์ฐ๋ฆฌ ํ์ ๋ง๋ ์ ๋ต์ ์ ํํ๊ณ ์ด์ํ๋ ์ค์ง์ ์ธ ํ๊น์ง '๋ธ๋ฃจ'๊ฐ ์๋ ค๋๋ ค์.
- ๋จ์ด: 1,442๊ฐ16๋ถ
[๐ค] TypeScript ํ์ ๊ฐ๋: ๋ฐํ์ ํ์ ์์ ์ฑ์ ์ํ ํ์ ํจํด ์ ๋ณตํด์
TypeScript์์ ๋ฐํ์์ ๋ณ์์ ํ์ ์ ์์ ํ๊ฒ ์ขํ๋(Narrowing) ๋ฐฉ๋ฒ์ธ ํ์ ๊ฐ๋(Type Guard)์ ๋ํด ์์ธํ ์์๋ด์. `typeof`, `instanceof`, `in` ์ฐ์ฐ์๋ถํฐ ์ฌ์ฉ์ ์ ์ ํ์ ๊ฐ๋๊น์ง, ์ค์ฉ์ ์ธ ์์์ ํจ๊ป ๊ฒฌ๊ณ ํ ์ฝ๋๋ฅผ ์์ฑํ๋ ๋ฐฉ๋ฒ์ ์ตํ๋ด์.
- ๋จ์ด: 2,542๊ฐ28๋ถ
[๐ค] React Query (TanStack Query) ์ฌํ: ๋ฐ์ดํฐ ํ์นญ, ์บ์ฑ, ๋๊ธฐํ ์ ๋ต์ผ๋ก ์น ์ฑ ์ฑ๋ฅ ๊ทน๋ํํด์
React Query (TanStack Query)๋ฅผ ํ์ฉํ์ฌ ๋ณต์กํ ์๋ฒ ์ํ๋ฅผ ํจ์จ์ ์ผ๋ก ๊ด๋ฆฌํ๊ณ , ์ง๋ฅ์ ์ธ ์บ์ฑ๊ณผ ์๋ ๋๊ธฐํ ์ ๋ต์ผ๋ก ์น ์ ํ๋ฆฌ์ผ์ด์ ์ ์ฑ๋ฅ๊ณผ ์ฌ์ฉ์ ๊ฒฝํ์ ๊ทน๋ํํ๋ ๋ฐฉ๋ฒ์ ์ฌ์ธต์ ์ผ๋ก ๋ค๋ฃจ์ด์. useQuery, useMutation, useInfiniteQuery ๋ฑ ํต์ฌ ํ ๊ณผ ์ค์ ์ต์ ํ ํ์ ๋ฐฐ์๋ณด์ธ์.
- ๋จ์ด: 2,401๊ฐ26๋ถ
[๐ค] React `useTransition`๊ณผ `useDeferredValue`๋ก ์ฌ์ฉ์ ๊ฒฝํ์ ๊ทน๋ํํ๋ ๋ฐฉ๋ฒ
React ์ ํ๋ฆฌ์ผ์ด์ ์์ ๋ฌด๊ฑฐ์ด UI ์ ๋ฐ์ดํธ๋ก ์ธํ ๋ฒ๋ฒ ์์ ํด๊ฒฐํ๊ณ , `useTransition`๊ณผ `useDeferredValue` ํ ์ ํ์ฉํ์ฌ ์ฌ์ฉ์ ๊ฒฝํ์ ํ๊ธฐ์ ์ผ๋ก ๊ฐ์ ํ๋ ์ค์ฉ์ ์ธ ์ ๋ต์ ๋ฐฐ์๋ณด์ธ์.
- ๋จ์ด: 1,917๊ฐ22๋ถ
[๐ค] React Suspense์ ErrorBoundary: ๊ฒฌ๊ณ ํ๊ณ ๋ถ๋๋ฌ์ด UI ๊ฒฝํ์ ์ํ ์ค์ ๊ฐ์ด๋
React ์ ํ๋ฆฌ์ผ์ด์ ์์ ์ฌ์ฉ์ ๊ฒฝํ์ ํ์ ํ Suspense์ ErrorBoundary์ ๊ฐ๋ ฅํ ์กฐํฉ์ ๊น์ด ์๊ฒ ๋ค๋ค์. ๋ก๋ฉ ์ํ์ ์๋ฌ ์ฒ๋ฆฌ๋ฅผ ์ฐ์ํ๊ฒ ๊ด๋ฆฌํ์ฌ ๋์ฑ ๊ฒฌ๊ณ ํ๊ณ ๋ถ๋๋ฌ์ด UI๋ฅผ ๋ง๋๋ ์ค์ ํ๊ณผ ์ฝ๋ ์์๋ฅผ ํ์ธํด ๋ณด์ธ์.
- ๋จ์ด: 1,302๊ฐ16๋ถ
[๐ค] CSS Container Queries: ์ปดํฌ๋ํธ ๊ธฐ๋ฐ ๋ฐ์ํ ๋์์ธ์ ์๋ก์ด ์งํ
๋ฏธ๋์ด ์ฟผ๋ฆฌ์ ํ๊ณ๋ฅผ ๋์ด, ์ปดํฌ๋ํธ ์์ฒด์ ํฌ๊ธฐ์ ๋ฐ๋ผ ์คํ์ผ์ ์กฐ์ ํ๋ CSS Container Queries๋ฅผ ๊น์ด ์๊ฒ ์์๋ณด๊ณ ์ค๋ฌด ์ ์ฉ ๋ฐฉ๋ฒ์ ์๋ดํด ๋๋ ค์.
- ๋จ์ด: 1,681๊ฐ19๋ถ
[๐ค] Next.js 15 ๊ณ ๊ธ ๋ฐ์ดํฐ ์บ์ฑ ์ ๋ต: fetch์ revalidate ์ฌ์ธต ๋ถ์
Next.js 15์์ `fetch` API์ ๊ฐ๋ ฅํ ์บ์ฑ ๋ฉ์ปค๋์ฆ๊ณผ `revalidate` ์ต์ ์ ํ์ฉํ์ฌ ์น ์ ํ๋ฆฌ์ผ์ด์ ์ ์ฑ๋ฅ์ ์ต์ ํํ๊ณ ๋ฐ์ดํฐ๋ฅผ ํจ์จ์ ์ผ๋ก ๊ด๋ฆฌํ๋ ๋ฐฉ๋ฒ์ ์ฌ์ธต์ ์ผ๋ก ๋ค๋ฃจ์ด์. ์ค๋ฌด ์์๋ฅผ ํตํด ์๋ฒ ์ปดํฌ๋ํธ์ ํด๋ผ์ด์ธํธ ์ปดํฌ๋ํธ์์ ์บ์ฑ ์ ๋ต์ ํจ๊ณผ์ ์ผ๋ก ์ ์ฉํ๋ ํ์ ์ ๊ณตํด์.
๋จ์ด: 1,320๊ฐ14๋ถ[๐ค] Next.js App Router: generateStaticParams๋ก ๋์ ๋ผ์ฐํ ๋น๋ ์ต์ ํํ๊ธฐ
Next.js App Router์์ generateStaticParams ํจ์๋ฅผ ํ์ฉํ์ฌ ๋์ ๋ผ์ฐํ ์ ์ ์ ํ์ด์ง๋ฅผ ํจ์จ์ ์ผ๋ก ์์ฑํ๊ณ ๋น๋ ์ฑ๋ฅ์ ์ต์ ํํ๋ ๋ฐฉ๋ฒ์ ์ค์ฉ์ ์ธ ์์์ ํจ๊ป ์์ธํ ์์๋ด์.
๋จ์ด: 1,891๊ฐ22๋ถ[๐ค] React ๋ ๋๋ง ์ต์ ํ: useMemo, useCallback, React.memo ์๋ฒฝ ๊ฐ์ด๋
์ด์ค๊ธ ๊ฐ๋ฐ์๋ฅผ ์ํ React ๋ ๋๋ง ์ต์ ํ ๊ฐ์ด๋. useMemo, useCallback, React.memo์ ์ ํํ ์ฌ์ฉ๋ฒ๊ณผ ์ค๋ฌด์์ ํํ ์ ์ง๋ฅด๋ ์ค์, ๊ทธ๋ฆฌ๊ณ ์ค์ ์ฑ๋ฅ ํฅ์ ์ ๋ต์ ๋ธ๋ฃจ๊ฐ ์๋ ค๋๋ ค์.
๋จ์ด: 2,145๊ฐ24๋ถ[๐ค] JavaScript Proxy์ Reflect ์ฌ์ธต ๋ถ์: ๋ฉํ ํ๋ก๊ทธ๋๋ฐ์ผ๋ก ์ฝ๋ ๊ฐํํ๊ธฐ
JavaScript Proxy์ Reflect API๋ฅผ ํ์ฉํ ๋ฉํ ํ๋ก๊ทธ๋๋ฐ ๊ธฐ๋ฒ์ ์ฌ์ธต ๋ถ์ํด์. ๊ฐ์ฒด ์ ๊ทผ ์ ์ด, ์ ํจ์ฑ ๊ฒ์ฌ, ๋ก๊น , ๋ฐ์ํ ์์คํ ๊ตฌํ ๋ฑ ์ค์ฉ์ ์ธ ํ์ฉ ์ฌ๋ก๋ฅผ ํตํด ์ฝ๋์ ์ ์ฐ์ฑ๊ณผ ์์ ์ฑ์ ๋์ด๋ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์.
- ๋จ์ด: 2,029๊ฐ24๋ถ
[๐ค] React/Next.js ๋ฒ๋ค ์ต์ ํ: ์ฝ๋ ์คํ๋ฆฌํ ๊ณผ ๋ ์ด์ง ๋ก๋ฉ ์๋ฒฝ ๊ฐ์ด๋
React์ Next.js ์ ํ๋ฆฌ์ผ์ด์ ์ ๋ฒ๋ค ํฌ๊ธฐ๋ฅผ ์ค์ด๊ณ ๋ก๋ฉ ์๋๋ฅผ ๊ฐ์ ํ๋ ์ฝ๋ ์คํ๋ฆฌํ ๊ณผ ๋ ์ด์ง ๋ก๋ฉ ๊ธฐ๋ฒ์ ์ค์ฉ์ ์ธ ์์์ ํจ๊ป ์์ธํ ์์๋ด์. ์นํฉ ์ค์ ๋ถํฐ React.lazy, Next.js dynamic import๊น์ง ๋ค๋ค์.
- ๋จ์ด: 1,770๊ฐ20๋ถ
[๐ค] React์ `useOptimistic` ํ ์ผ๋ก ๋๊ด์ UI ์ ๋ฐ์ดํธ ๊ตฌํํ๊ธฐ: Server Actions์ ํจ๊ป
React 18/19์ `useOptimistic` ํ ์ ํ์ฉํ์ฌ Server Actions์ ์ฐ๋๋๋ ๋๊ด์ UI ์ ๋ฐ์ดํธ๋ฅผ ๊ตฌํํ๋ ๋ฐฉ๋ฒ์ ์ค์ฉ์ ์ธ ์์์ ํจ๊ป ์์ธํ ์์๋ด์. ์ฌ์ฉ์ ๊ฒฝํ์ ๊ฐ์ ํ๊ณ ์ ํ๋ฆฌ์ผ์ด์ ์ ๋ฐ์์ฑ์ ๋์ด๋ ๋ ธํ์ฐ๋ฅผ ๊ณต์ ํด์.
๋จ์ด: 1,561๊ฐ17๋ถ[๐ค] TypeScript const Type Parameters: ๋ฆฌํฐ๋ด ํ์ ์ถ๋ก ๊ฐํ์ ์ค์ฉ์ ์ธ ํ์ฉ๋ฒ
TypeScript 5.0์ ๋์ ๋ const Type Parameters๋ฅผ ํ์ฉํ์ฌ ์ ๋ค๋ฆญ ํจ์์ ๋ฆฌํฐ๋ด ํ์ ์ถ๋ก ์ ์ ๊ตํ๊ฒ ์ ์ดํ๊ณ , ๋์ฑ ๊ฒฌ๊ณ ํ ํ์ ์์คํ ์ ๊ตฌ์ถํ๋ ์ค์ฉ์ ์ธ ๋ฐฉ๋ฒ์ ์์๋ณด์ธ์. as const์์ ์ฐจ์ด์ ๊ณผ ์ค์ ์ฝ๋ ์์๋ฅผ ํตํด ์ด์ค๊ธ ๊ฐ๋ฐ์๋ ์ฝ๊ฒ ์ดํดํ ์ ์๋๋ก ์ค๋ช ํด ๋๋ ค์.
- ๋จ์ด: 2,028๊ฐ22๋ถ
[๐ค] Next.js/React ์ฑ CLS ์ต์ ํ: ์ํํธ ์๋ ์ฌ์ฉ์ ๊ฒฝํ ๋ง๋ค๊ธฐ
Next.js์ React ์ ํ๋ฆฌ์ผ์ด์ ์์ Cumulative Layout Shift(CLS) ๋ฌธ์ ๋ฅผ ํด๊ฒฐํ๊ณ ์ฌ์ฉ์ ๊ฒฝํ์ ๊ฐ์ ํ๋ ์ค์ง์ ์ธ ์ ๋ต๊ณผ ์ฝ๋ ์์๋ฅผ ์์ธํ ์์๋ณด์ธ์. ์น ์ฑ๋ฅ ์ต์ ํ์ ํต์ฌ ์์์ธ CLS๋ฅผ ํจ๊ณผ์ ์ผ๋ก ๊ด๋ฆฌํ๋ ๋ฐฉ๋ฒ์ ์๋ ค๋๋ ค์.
- ๋จ์ด: 1,740๊ฐ21๋ถ
[๐ค] Next.js SSR, SSG, ISR ๋ ๋๋ง ์ ๋ต: App Router์์ ์ต์ ์ ์ ํ์?
Next.js App Router์์ ์๋ฒ ์ฌ์ด๋ ๋ ๋๋ง(SSR), ์ ์ ์ฌ์ดํธ ์์ฑ(SSG), ์ฆ๋ถ ์ ์ ์ฌ์์ฑ(ISR) ๊ฐ ๋ ๋๋ง ์ ๋ต์ ๋์ ์๋ฆฌ, ์ฅ๋จ์ , ์ค์ ํ์ฉ ๋ฐ ์ต์ ํ ๋ฐฉ๋ฒ์ ๋น๊ต ๋ถ์ํด๋๋ ค์.
- ๋จ์ด: 1,478๊ฐ17๋ถ
[๐ค] React Context API์ Zustand: ์ ์ญ ์ํ ๊ด๋ฆฌ, ์ธ์ ๋ฌด์์ ์จ์ผ ํ ๊น์?
React ์ ํ๋ฆฌ์ผ์ด์ ์ ์ ์ญ ์ํ ๊ด๋ฆฌ๋ฅผ ๊ณ ๋ฏผํ๊ณ ๊ณ์ ๊ฐ์? Context API์ ๊ฐ๋ฒผ์ด ์ธ๋ถ ์ํ ๊ด๋ฆฌ ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ธ Zustand๋ฅผ ๋น๊ต ๋ถ์ํ๊ณ , ์ค๋ฌด์์ ๊ฐ ๋๊ตฌ๋ฅผ ํจ๊ณผ์ ์ผ๋ก ํ์ฉํ๋ ์ ๋ต์ ์ค์ ์ฝ๋ ์์์ ํจ๊ป ์์ธํ ์๋ ค๋๋ ค์.
- ๋จ์ด: 2,004๊ฐ24๋ถ
[๐ค] Turborepo๋ก Next.js ๋ชจ๋ ธ๋ ํฌ ๊ตฌ์ถ: ํจ์จ์ ์ธ ๊ฐ๋ฐ ๋ฐ ์ต์ ํ ์ ๋ต
Turborepo๋ฅผ ํ์ฉํ์ฌ Next.js ํ๋ก์ ํธ๋ฅผ ๋ชจ๋ ธ๋ ํฌ๋ก ๊ตฌ์ฑํ๊ณ , ๊ณต์ ์ปดํฌ๋ํธ, ์ ํธ๋ฆฌํฐ, CI/CD ์ต์ ํ ๋ฐฉ์์ ์ค๋ฌด ์์์ ํจ๊ป ์์ธํ ์ค๋ช ํด ๋๋ ค์.
- ๋จ์ด: 2,338๊ฐ27๋ถ
[๐ค] React useEffect ํ , ์ด์ ํท๊ฐ๋ฆฌ์ง ๋ง์ธ์! (์์กด์ฑ ๋ฐฐ์ด, ํด๋ฆฐ์ ์๋ฒฝ ๊ฐ์ด๋)
React ๊ฐ๋ฐ์์ ํ์์ ์ธ useEffect ํ ์ ๋์ ์๋ฆฌ๋ถํฐ ์์กด์ฑ ๋ฐฐ์ด, ํด๋ฆฐ์ ํจ์ ํ์ฉ๋ฒ, ๊ทธ๋ฆฌ๊ณ ์ค๋ฌด์์ ์์ฃผ ๊ฒช๋ ์ค์์ ์ต์ ํ ์ ๋ต๊น์ง, ์ด์ค๊ธ ๊ฐ๋ฐ์๋ฅผ ์ํ ์๋ฒฝ ๊ฐ์ด๋๋ฅผ ์ ๊ณตํด์.
- ๋จ์ด: 3,284๊ฐ31๋ถ
[๐ค] Next.js Server Actions ์ค์ : ์๋ฌ ์ฒ๋ฆฌ, ์ ํจ์ฑ ๊ฒ์ฌ, ๋๊ด์ UI ์ ๋ฐ์ดํธ
Next.js Server Actions๋ฅผ ์ค๋ฌด์ ์ ์ฉํ ๋ ๋ง์ฃผํ๋ ์๋ฌ ์ฒ๋ฆฌ, ๋ฐ์ดํฐ ์ ํจ์ฑ ๊ฒ์ฌ, ๊ทธ๋ฆฌ๊ณ ์ฌ์ฉ์ ๊ฒฝํ์ ํฅ์์ํค๋ ๋๊ด์ UI ์ ๋ฐ์ดํธ ๊ธฐ๋ฒ์ ์์ธํ ์ฝ๋ ์์์ ํจ๊ป ์์๋ณด์ธ์.
๋จ์ด: 1,983๊ฐ21๋ถ[๐ค] TypeScript ์ ํธ๋ฆฌํฐ ํ์ ์๋ฒฝ ๊ฐ์ด๋: ์ค์ ํ์ฉ ํจํด
TypeScript ์ ํธ๋ฆฌํฐ ํ์ ์ ํต์ฌ ๊ฐ๋ ๊ณผ ์ค์ ํ์ฉ๋ฒ์ ๊น์ด ์๊ฒ ๋ค๋ค์. Pick, Omit, Partial, Required ๋ฑ ์์ฃผ ์ฐ๋ ์ ํธ๋ฆฌํฐ ํ์ ์ผ๋ก ๋ณต์กํ ํ์ ์ ํจ๊ณผ์ ์ผ๋ก ๋ค๋ฃจ๋ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์. ํ์ ์คํฌ๋ฆฝํธ ์ฝ๋์ ์ฌ์ฌ์ฉ์ฑ๊ณผ ์์ ์ฑ์ ๋์ด๋ ๋ ธํ์ฐ๋ฅผ ๊ณต์ ํด์.
- ๋จ์ด: 1,712๊ฐ20๋ถ
[๐ค] Next.js App Router ๋ฏธ๋ค์จ์ด: ๊ฐ๋ ฅํ ์์ฒญ ์ฒ๋ฆฌ ์ ๋ต๊ณผ ์ค์ ์์
Next.js App Router ํ๊ฒฝ์์ ๋ฏธ๋ค์จ์ด๋ฅผ ํ์ฉํด ์ฌ์ฉ์ ์ธ์ฆ, ๋ฆฌ๋ค์ด๋ ์ , ๊ตญ์ ํ ๋ฑ์ ์์ฒญ ์ฒ๋ฆฌ ๋ก์ง์ ํจ์จ์ ์ผ๋ก ๊ตฌํํ๋ ๋ฐฉ๋ฒ์ ์ค์ ์์ ์ ํจ๊ป ์์ธํ ์์๋ณด์ธ์.
- ๋จ์ด: 1,630๊ฐ19๋ถ
[๐ค] ํ์ ์คํฌ๋ฆฝํธ ์ ๋ค๋ฆญ ์ฌํ: ์ค์ฉ์ ์ธ ํจํด๊ณผ ํํ ์คํด๋ค
ํ์ ์คํฌ๋ฆฝํธ ์ ๋ค๋ฆญ(Generics)์ ๊น์ด ์ดํดํ๊ณ , ์ค๋ฌด์์ ์์ฃผ ์ฌ์ฉ๋๋ ์ ๋ค๋ฆญ ํจํด๊ณผ ํํ ๊ฒช๋ ์คํด๋ค์ ์ค์ ์ฝ๋ ์์์ ํจ๊ป ์ฝ๊ณ ๋ช ํํ๊ฒ ์ค๋ช ํด ๋๋ ค์. ํ์ ์์ ์ฑ๊ณผ ์ฝ๋ ์ฌ์ฌ์ฉ์ฑ์ ๋์ด๋ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์.
๋จ์ด: 1,860๊ฐ18๋ถ[๐ค] Next.js Route Handler: App Router์์ ์์ ํ๊ณ ํจ์จ์ ์ธ API ๊ตฌ์ถํ๊ธฐ (์ธ์ฆ, ์๋ฌ ์ฒ๋ฆฌ ํฌํจ)
Next.js App Router์ Route Handler๋ฅผ ์ฌ์ฉํ์ฌ API ์๋ํฌ์ธํธ๋ฅผ ๊ตฌ์ถํ๋ ๋ฐฉ๋ฒ์ ์์ธํ ์์๋ด์. ์ธ์ฆ, ์๋ฌ ์ฒ๋ฆฌ, ๊ทธ๋ฆฌ๊ณ ์บ์ฑ ์ ๋ต์ ํฌํจํ ์ค์ฉ์ ์ธ ํ์ผ๋ก ์์ ํ๊ณ ํจ์จ์ ์ธ ์๋ฒ๋ฆฌ์ค ํจ์๋ฅผ ๋ง๋๋ ๋ฐฉ๋ฒ์ ์ตํ๋ด์.
- ๋จ์ด: 1,934๊ฐ22๋ถ
[๐ค] Next.js Image ์ปดํฌ๋ํธ ์ต์ ํ: Core Web Vitals ๊ฐ์ ๋ถํฐ ์ค์ ํ์ฉ๊น์ง
Next.js์ Image ์ปดํฌ๋ํธ๋ฅผ ํ์ฉํ์ฌ ์น ์ฑ๋ฅ ํต์ฌ ์งํ์ธ Core Web Vitals๋ฅผ ๊ฐ์ ํ๊ณ , ๋ค์ํ ์ต์ ํ ์ต์ ์ ์ค์ ํ๋ก์ ํธ์ ์ ์ฉํ๋ ๋ฐฉ๋ฒ์ ๋ธ๋ฃจ๊ฐ ์์ธํ ์๋ ค๋๋ ค์.
- ๋จ์ด: 2,187๊ฐ25๋ถ
[๐ค] Next.js 14.1+์ ํ์ : Partial Prerendering (PPR) ์๋ฒฝ ๊ฐ์ด๋์ ์ค์ ์ต์ ํ ์ ๋ต
Next.js 14.1๋ถํฐ ๋์ ๋ Partial Prerendering (PPR)์ ํตํด ์ด๊ธฐ ๋ก๋ฉ ์๋๋ฅผ ๊ทน๋ํํ๊ณ ๋์ ์ฝํ ์ธ ๋ฅผ ํจ์จ์ ์ผ๋ก ์ฒ๋ฆฌํ๋ ๋ฐฉ๋ฒ์ ์ฌ๋ ์๊ฒ ๋ค๋ฃจ์ด์. PPR์ ๋์ ์๋ฆฌ๋ถํฐ ์ค์ ํ๋ก์ ํธ ์ ์ฉ ์ ๋ต๊น์ง, ๊ฐ๋ฐ์๋ค์ด ๊ถ๊ธํดํ๋ ๋ชจ๋ ๊ฒ์ ์๋ ค๋๋ ค์.
- ๋จ์ด: 1,792๊ฐ19๋ถ
[๐ค] TypeScript ์กฐ๊ฑด๋ถ ํ์ ๊ณผ infer ํค์๋: ๋ณต์กํ ํ์ ๋ ์์ฝ๊ฒ ๋ค๋ฃจ๋ ๋ฐฉ๋ฒ
TypeScript ๊ฐ๋ฐ์์ ๋ง์ฃผํ๋ ๋ณต์กํ ํ์ ์ถ๋ก ๋ฌธ์ , ์กฐ๊ฑด๋ถ ํ์ ๊ณผ infer ํค์๋๋ฅผ ํ์ฉํ๋ฉด ํจ์ฌ ์ฐ์ํ๊ณ ๊ฐ๋ ฅํ๊ฒ ํด๊ฒฐํ ์ ์์ด์. ์ค์ ์์ ์ ํจ๊ป ๊ทธ ํ์ฉ๋ฒ์ ์ฌ๋ ์๊ฒ ๋ค๋ค๋ด ๋๋ค.
- ๋จ์ด: 1,705๊ฐ21๋ถ
[๐ค] JavaScript ์ด๋ฒคํธ ๋ฃจํ(Event Loop) ์์ ์ ๋ณต: ๋น๋๊ธฐ ์ฒ๋ฆฌ์ ๋ฐํ์ ๋์ ์๋ฆฌ
JavaScript์ ํต์ฌ ๋น๋๊ธฐ ์ฒ๋ฆฌ ๋ฉ์ปค๋์ฆ์ธ ์ด๋ฒคํธ ๋ฃจํ์ ๋์ ์๋ฆฌ๋ฅผ ์ฌ๋ ์๊ฒ ํํค์ณ ๋ด์. ์ฝ ์คํ, ํ์คํฌ ํ, ๋ง์ดํฌ๋กํ์คํฌ ํ์์ ์ํธ์์ฉ์ ์ดํดํ๊ณ , ์ค๋ฌด์์ ๋ง์ฃผ์น๋ ๋น๋๊ธฐ ์ฝ๋์ ๋์์ ๋ช ํํ ์์ธกํ๋ ๋ฐฉ๋ฒ์ ์๋ ค๋๋ ค์.
- ๋จ์ด: 1,964๊ฐ23๋ถ
[๐ค] Next.js Server & Client Components, ์ค์ ์์ ํ๋ช ํ๊ฒ ์ ํํ๋ ๊ฐ์ด๋
Next.js App Router์์ Server Components์ Client Components ์ค ์ด๋ค ๊ฒ์ ์ฌ์ฉํด์ผ ํ ์ง ๊ณ ๋ฏผ์ด์ ๊ฐ์? ์ด ๊ธ์์ ๋ ์ปดํฌ๋ํธ์ ํต์ฌ ์ฐจ์ด์ , ์ฌ์ฉ ์์ , ๊ทธ๋ฆฌ๊ณ ์ฑ๋ฅ ์ต์ ํ๋ฅผ ์ํ ์ค์ ์ ๋ต์ ๋ธ๋ฃจ๊ฐ ์๋ ค๋๋ฆด๊ฒ์.
- ๋จ์ด: 1,879๊ฐ21๋ถ
[๐ค] TypeScript satisfies ์ฐ์ฐ์: ํ์ ์ถ๋ก ๊ณผ ์์ ์ฑ์ ๋์์ ์ก๋ ๋น๋ฒ
TypeScript์ `satisfies` ์ฐ์ฐ์๋ฅผ ํ์ฉํ์ฌ ํ์ ์ถ๋ก ์ ์ ์ฐ์ฑ์ ์ ์งํ๋ฉด์๋ ์๊ฒฉํ ํ์ ์์ ์ฑ์ ํ๋ณดํ๋ ๋ฐฉ๋ฒ์ ์์๋ณด์ธ์. ์ค์ฉ์ ์ธ ์์๋ฅผ ํตํด ์ค์ ํ๋ก์ ํธ์ ์ ์ฉํ๋ ๋ ธํ์ฐ๋ฅผ ๊ณต์ ํฉ๋๋ค.
- ๋จ์ด: 1,211๊ฐ15๋ถ
[๐ค] React 19 ์๋ก์ด ๊ธฐ๋ฅ: use ํ , Actions, ๊ทธ๋ฆฌ๊ณ ์ปดํ์ผ๋ฌ ๋ฏธ๋ฆฌ๋ณด๊ธฐ
React 19์ ํต์ฌ ๋ณ๊ฒฝ ์ฌํญ์ธ use ํ , ์๋ฒ ์ก์ , ๊ทธ๋ฆฌ๊ณ React ์ปดํ์ผ๋ฌ์ ๋์ ๋ฐฐ๊ฒฝ๊ณผ ์ค์ ํ์ฉ ์์๋ฅผ ์ด์ค๊ธ ๊ฐ๋ฐ์ ๋๋์ด์ ๋ง์ถฐ ์์ธํ ์ค๋ช ํฉ๋๋ค. ์ต์ React ์ ๋ฐ์ดํธ๋ฅผ ํตํด ์ ํ๋ฆฌ์ผ์ด์ ์ฑ๋ฅ๊ณผ ๊ฐ๋ฐ ๊ฒฝํ์ ํฅ์์ํค๋ ๋ฐฉ๋ฒ์ ์์๋ณด์ธ์.
- ๋จ์ด: 1,524๊ฐ16๋ถ
[๐ค] Next.js App Router ์บ์ฑ ์ ๋ต: ๋ฐ์ดํฐ ์ฌ๊ฒ์ฆ (revalidatePath, revalidateTag) ์๋ฒฝ ๊ฐ์ด๋
Next.js 14 App Router์์ ํจ์จ์ ์ธ ๋ฐ์ดํฐ ์บ์ฑ ์ ๋ต๊ณผ revalidatePath, revalidateTag๋ฅผ ์ด์ฉํ ๋ฐ์ดํฐ ์ฌ๊ฒ์ฆ ๋ฐฉ๋ฒ์ ์ค๋ฌด ์์์ ํจ๊ป ์์ธํ ์์๋ณด๊ณ ์น ์ฑ๋ฅ์ ์ต์ ํํ๋ ๋ฐฉ๋ฒ์ ๋ฐฐ์๋ณด์ธ์.