동기 처리란, 작업을 순차적으로 처리하는 것을 말한다.
자바스크립트 코드는 기본적으로 동기적으로 처리된다.
동기적으로 처리되는 코드는 위에서부터 아래로 순차적으로 실행되며, 어떤 작업이 끝나야 다음 작업을 수행할 수 있다.
하지만, 동기적으로 처리되는 코드는 작업이 끝날 때까지 다른 작업을 수행할 수 없다는 단점이 있다.
console.log('은행 1번 번호표 업무 시작');
console.log('은행 1번 번호표 업무 끝');
console.log('은행 2번 번호표 업무 시작');
console.log('은행 2번 번호표 업무 끝');
비동기 처리란, 작업을 동시에 처리하는 것을 말한다.
비동기 처리는 이러한 단점을 보완하기 위해 등장했다.
비동기 처리는 작업을 동시에 처리할 수 있으며, 작업이 끝나지 않아도 다른 작업을 수행할 수 있다.
setTimeout(() => {
console.log('느린 미니언즈 라떼');
}, 1000);
console.log('빠른 아메리카노');
fetch('https://jsonplaceholder.typicode.com/posts/1')
.then(res => res.json())
.then(data => console.log(data));
console.log('데이터 요청 중...');
이 중 가장 많이 사용되는 방식은 Promise 및 async/await입니다. 이에 대해 자세히 알아보겠습니다.
const a = () => console.log("a");
const b = () => console.log("b");
a(); // a
b(); // b
// 비동기 처리를 사용하기 위해 setTimeout 함수를 사용합니다.
// a 함수가 실행되고 1초 후에 b 함수가 실행됩니다.
const a = () => {
setTimeout(() => {
console.log("a");
}, 1000);
};
const b = () => console.log("b");
a(); // a
b(); // b
// 콜백 함수를 사용하여 비동기 작업을 순차적으로 처리합니다.
const a = callback => {
setTimeout(() => {
console.log("a"); // a
callback(); // b
}, 1000);
};
const b = () => console.log("b");
a(b); // a, b
new Promise((resolve, reject) => { ... });
프로미스는 세 가지 상태를 가진다.
프로미스를 사용하여 비동기 처리를 하려면, new Promise()를 사용하여 프로미스 객체를 생성하고, resolve와 reject를 사용하여 프로미스의 상태를 변경한다.
// Promise 객체 생성
// new Promise((resolve, reject) => {...})를 통해 새로운 프로미스 객체 생성
// 이 객체는 비동기 작업을 수행할 콜백 함수를 인자로 받으며,
// 이 콜백 함수는 resolve와 reject라는 두 개의 함수를 매개변수로 가집니다.
// resolve, reject 함수는 프로미스의 상태를 변경하는 역할을 합니다.
const a = () => {
return new Promise((resolve, reject) => {
setTimeout(() => {
// 1초 후에
console.log(1);
resolve(); // 성공 시 호출
}, 1000);
});
};
const b = () => console.log(2);
// 아래 코드는 역시나 콜백 지옥에 빠집니다.
// then 메서드를 사용하여 비동기 작업을 순차적으로 처리할 수 있습니다.
// a 함수가 실행되고 완료된 후에 b 함수가 실행되고, c 함수가 실행됩니다.
a().then(() => {
b()
});
위 코드에서는 a
, b
, c
, d
함수가 각각 1초씩 기다린 후 콘솔에 숫자를 출력하는 비동기 작업을 수행합니다. 각 함수는 Promise
객체를 반환하며, 이 객체는 작업이 성공했을 때 resolve
를 호출합니다.
// b 함수 호출 시 return 키워드로 반환하게 되면
// promise 객체를 반환하게 되어 then 메서드를 연달아 사용할 수 있습니다.
a()
.then(() => {
return b();
})
위 코드와 같이 각 then 메서드 내에서 다음 비동기 함수를 반환하도록 변경하여, then 체인을 형성합니다. 이렇게 하면 콜백 지옥을 피할 수 있습니다.
a()
.then(() => b())
화살표 함수의 축약 형태로, 함수 내부가 한 줄로 표현될 때 중괄호를 생략할 수 있습니다.
// resolve는 하나의 함수 데이터를 받아서 return으로 처리할 수 있습니다.
// 해서 b, c, d 함수 데이터 자체로 전달할 수 있습니다.
a()
.then(b)
.then(() => {
console.log('done');
});
// Promise.all 메서드를 사용하여 여러 개의 비동기 작업을 동시에 처리할 수 있습니다.
Promise.all([a(), b()]).then(() => {
console.log('모든 작업이 완료되었습니다.');
});
JSONPlaceholder에서 제공하는 API를 사용하여 사용자 정보를 가져와 출력해보겠습니다.
fetch(): 네트워크 요청을 보내는 API로, Promise 객체를 반환하여 비동기 작업을 처리합니다.
Resolve: Promise 객체가 성공적으로 처리되었을 때 호출되는 콜백 함수로, 비동기 작업이 완료되면 결과를 반환합니다.
// userId 매개변수를 받아 사용자 정보를 가져오는 함수를 구현합니다.
const getUser = userId => {
// Promise 객체를 반환합니다. resolve 함수를 사용하여 비동기 작업이 완료되면 결과를 반환합니다.
return new Promise(resolve => {
// fetch 함수를 사용하여 JSONPlaceholder API를 호출합니다.
fetch(`https://jsonplaceholder.typicode.com/users/${userId}`)
// fetch 함수가 성공적으로 호출되면 response 객체를 반환합니다.
// json 메서드를 사용하여 response 객체를 JSON 형태로 변환합니다.
.then(res => res.json())
// response 객체가 성공적으로 변환되면 data 객체를 반환합니다.
.then(data => resolve(data));
});
};
// 사용자 데이터를 가져오는 함수를 호출합니다.
getUser(1)
// getUser 함수가 성공적으로 호출되면 data 객체를 반환합니다.
.then(data => {
console.log(`사용자 ID 1의 이름은 ${data.name} 입니다.`);
return getUser(2);
})
.then(data => {
console.log(`사용자 ID 2의 이름은 ${data.name} 입니다.`);
return getUser(3);
})
.then(data => {
console.log(`사용자 ID 3의 이름은 ${data.name} 입니다.`);
});
// async 함수를 사용하여 비동기 작업을 처리합니다.
const a = () => {
return new Promise(resolve => {
setTimeout(() => {
console.log(1);
resolve();
}, 1000);
});
};
// awite 함수를 사용하여 a함수를 실행하고 완료된 후에 b함수를 실행합니다.
// await 키워드는 뒷 부분의 비동기 작업이 완료될 때까지 기다립니다.
// await a();
// b();
// async 함수를 사용하여 비동기 작업을 순차적으로 처리합니다.
// await 키워드를 사용할 때는 async 함수 내부에서만 사용할 수 있기 때문에
// main 함수를 async 함수로 선언합니다.
const main = async () => {
await a(); // 1
b(); // 2
};
main(); // 1, 2
const getUser = userId => {
return new Promise(resolve => {
fetch(`https://jsonplaceholder.typicode.com/users/${userId}`)
.then(res => res.json())
.then(data => resolve(data));
});
};
// await 키워드를 사용하여 비동기 작업을 순차적으로 처리합니다.
// const user1 = await getUser(1);
// console.log(`사용자 ID 1의 이름은 ${user1.name} 입니다.`);
// const user2 = await getUser(2);
// console.log(`사용자 ID 2의 이름은 ${user2.name} 입니다.`);
// const user3 = await getUser(3);
// console.log(`사용자 ID 3의 이름은 ${user3.name} 입니다.`);
// await 키워드를 사용할 때는 async 함수로 묶어주어야 합니다.
const main = async () => {
const user1 = await getUser(1);
console.log(`사용자 ID 1의 이름은 ${user1.name} 입니다.`);
const user2 = await getUser(2);
console.log(`사용자 ID 2의 이름은 ${user2.name} 입니다.`);
const user3 = await getUser(3);
console.log(`사용자 ID 3의 이름은 ${user3.name} 입니다.`);
};
main();
// 사용자 데이터를 가져오는 함수
const getUser = userId => {
// resolve, reject 함수를 사용하여 비동기 작업이 성공 또는 실패했을 때 결과를 반환합니다.
return new Promise((resolve, reject) => {
fetch(`https://jsonplaceholder.typicode.com/users/${userId}`)
.then(response => {
// fetch 함수가 성공적으로 호출되면 response 객체를 반환합니다.
// response 객체가 정상적이지 않을 경우 에러를 반환합니다.
if (!response.ok) {
throw new Error('네트워크 응답이 정상적이지 않습니다: ' + response.statusText);
}
return response.json();
})
.then(data => resolve(data))
// fetch 함수가 실패했을 경우 에러를 반환합니다.
.catch(error => reject(error));
});
};
// 사용자 데이터를 가져오는 함수를 호출합니다.
getUser(1)
.then(data => {
console.log(`사용자 ID 1의 이름은 ${data.name} 입니다.`);
return getUser(2);
})
.then(data => {
console.log(`사용자 ID 2의 이름은 ${data.name} 입니다.`);
return getUser(3);
})
.then(data => {
console.log(`사용자 ID 3의 이름은 ${data.name} 입니다.`);
})
// catch 메서드를 사용하여 에러를 처리합니다.
.catch(error => {
console.error('사용자 정보를 가져오는 중 오류가 발생했습니다.');
})
// finally 메서드를 사용하여 작업이 완료되었을 때 메시지를 출력합니다.
.finally(() => {
console.log('사용자 정보를 가져오는 작업이 완료되었습니다.');
});
// async/await를 사용하여 에러를 처리합니다.
const main = async () => {
try {
const user1 = await getUser(1);
console.log(`사용자 ID 1의 이름은 ${user1.name} 입니다.`);
const user2 = await getUser(2);
console.log(`사용자 ID 2의 이름은 ${user2.name} 입니다.`);
const user3 = await getUser(3);
console.log(`사용자 ID 3의 이름은 ${user3.name} 입니다.`);
} catch (error) {
console.error('사용자 정보를 가져오는 중 오류가 발생했습니다.');
} finally {
console.log('사용자 정보를 가져오는 작업이 완료되었습니다.');
}
};
const axios = require('axios');
// 지정된 ID를 가진 유저에 대한 요청
axios.get('/user?ID=12345')
.then(function (response) {
// 성공 핸들링
console.log(response);
})
.catch(function (error) {
// 에러 핸들링
console.log(error);
})
.finally(function () {
// 항상 실행되는 영역
});
cdn : <script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
const getUser = userId => {
return new Promise((resolve, reject) => {
axios.get(`https://jsonplaceholder.typicode.com/users/${userId}`)
.then(response => resolve(response.data))
.catch(error => reject(error));
});
};
// async/await를 사용하여 에러를 처리
const main = async () => {
try {
const user1 = await getUser(1);
console.log(`사용자 ID 1의 이름은 ${user1.name} 입니다.`);
const user2 = await getUser(2);
console.log(`사용자 ID 2의 이름은 ${user2.name} 입니다.`);
const user3 = await getUser(3);
console.log(`사용자 ID 3의 이름은 ${user3.name} 입니다.`);
} catch (error) {
console.error('사용자 정보를 가져오는 중 오류가 발생했습니다.');
} finally {
console.log('사용자 정보를 가져오는 작업이 완료되었습니다.');
}
};
주요 변경사항:
fetch → axios.get
으로 변경response.json()
과정이 필요 없음 (axios는 자동으로 JSON 변환)response.ok
체크가 필요 없음 (axios는 에러 상태코드를 자동으로 catch로 보냄)response.data
로 바로 데이터에 접근 가능axios가 더 간단하고 편리하죠! 😊
console 창에서 fetch 함수를 사용하여 JSONPlaceholder API를 호출해보겠습니다.
// fetch 함수를 사용하여 JSONPlaceholder API를 호출합니다.
console.log(fetch('https://jsonplaceholder.typicode.com/users'));
콘솔창의 promise instance를 클릭하면
then, catch, finally 메서드를 사용할 수 있도록 지정된 것을 확인할 수 있습니다.
fetch 함수를 사용하여 JSONPlaceholder API를 호출하면, Promise 객체를 반환합니다.
fetch('https://jsonplaceholder.typicode.com/users').then(res => console.log(res));
json 메서드를 사용하여 response 객체를 JSON 형태로 변환할 수 있습니다.
fetch('https://jsonplaceholder.typicode.com/users').then(res => console.log(res.json()));
그러면 json 형태로 변환된 데이터를 확인할 수 있습니다.
fetch 함수를 사용하여 JSONPlaceholder API를 호출하면, JSON 형태로 변환된 데이터를 반환합니다.
fetch('https://jsonplaceholder.typicode.com/users', {
// HTTP 요청 메서드를 지정합니다.
method: 'POST',
// 서버로 전송되는 데이터의 형식을 지정합니다.
headers: {
'Content-Type': 'application/json',
},
// 서버로 전송되는 데이터를 지정합니다.
// body 옵션에 명시하는 데이터는 문자열 형태로 지정해야 합니다.
// JSON.stringify 메서드를 사용하여 JSON 형태로 변환합니다.
body: JSON.stringify({
title: 'foo',
body: 'bar',
userId: 1,
}),
})
.then(res => res.json()) // JSON 형태로 변환된 데이터를 반환합니다.
.then(json => console.log(json)); // JSON 형태로 변환된 데이터를 출력합니다.
// .then(res => {
// return res.json();
// });
fetch 함수가 promise 객체를 반환하므로, async/await를 사용하여 비동기 작업을 처리할 수 있습니다.
const main = async () => {
const res = await fetch('https://jsonplaceholder.typicode.com/users');
const json = await res.json();
console.log(json);
};
main();