์น์์ผ์ ํ์ฉํด ์ค์๊ฐ ์ฑํ ๊ตฌํํ๊ธฐ - Spring
์น์์ผ์ ํ์ฉํด ์ค์๊ฐ ์ฑํ
๊ตฌํํ๊ธฐ - Springboot
์น์์ผ์ ํ์ฉํด ์ค์๊ฐ ์ฑํ
๊ตฌํํ๊ธฐ - React
์น์์ผ์ด๋
โ WebSocket์ ์ค์๊ฐ ์๋ฐฉํฅ ํต์ ์ ์ํ ํ๋กํ ์ฝ์ ์ผ์ข ์ด๋ค.
WebSocket์ ๊ธฐ๋ณธ์ ์ผ๋ก TCP ํ๋กํ ์ฝ์ ์ฌ์ฉํ๋ฉฐ, ํธ๋์์ดํฌ ๊ณผ์ ์ ํตํด ์ฐ๊ฒฐ์ ์๋ฆฝํ ํ, ์ง์์ ์ธ ์ฐ๊ฒฐ์ ํตํด ์๋ฐฉํฅ ํต์ ์ด ๊ฐ๋ฅํ๊ฒ ๋๋ค.
์ฃผ๋ก ์ค์๊ฐ ์ฑํ , ๊ฒ์ ๋ฑ์ ๋ง์ด ์ฐ์ธ๋ค
์น์์ผ์ ์ฅ์ ๋ฐ ํน์ง (vs HTTP)
โ ์น์์ผ์ ํต์ฌ ํค์๋๋ ์ง์์ ์ธ ์ฐ๊ฒฐ์ด๋ค.
์น์์ผ์ ์ฅ์ ๊ณผ ํน์ง์ HTTP์ ๋น๊ตํ์ ๋ ๋ช ํํด์ง๋ค.
์ผ๋ฐ์ ์ผ๋ก ์ฌ์ฉ๋๋ HTTP๋ stateless, connectionless๋ผ๋ ํน์ง์ผ๋ก ์ธํด ์ค์๊ฐ์ผ๋ก ์๋ฐฉํฅ ํต์ ์ ๋งบ๊ณ ์ํ ๋ WebSocket์ด ํจ์ฌ ์ ๋ฆฌํ๋ค.
์น์์ผ์ ๋ํ์ ์ธ ์์ ์ธ ์ค์๊ฐ ์ฑํ
์ HTTP๋ก ๊ตฌํํ๋ค๊ณ ํด๋ณด์.
polling์ด๋ ๊ธฐ๋ฒ์ ํ์ฉํด์ผ๋๋๋ฐ ์์ฒญ์ ๋ํ ์๋ต์ด ์ฌ ๋๊น์ง ๊ณ์ํด์ ์์ฒญ์ ๋ณด๋ด๋ ๋ฐฉ์์ด๋ผ ๋น์ฐํ ๋น์ฉ์ด ์์ฒญ ํฌ๋ค
HTTP์์ Polling์ผ๋ก ๊ตฌํํ ์ค์๊ฐ ์ฑํ
๋ -> ์๋ฒ: ์๋
!
๋ -> ์๋ฒ: ๋๋ต (์์ฒญ1)
๋ -> ์๋ฒ: ๋๋ต (์์ฒญ2)
๋ -> ์๋ฒ: ๋๋ต (์์ฒญ3)
...
๋ -> ์๋ฒ: ๋๋ต (์์ฒญN)
์๋ฒ -> ๋: ์๋
~ (๋๋ต ์์ )
๋ฑ๋ด๋ ๋นํจ์จ์ ์ด๋ค...
long polling์ด๋ ๋ฐฉ๋ฒ๋ ์๋๋ฐ ๊ธฐ๋ณธ์ ์ผ๋ก HTTP ์์ฒด๊ฐ ์ค์๊ฐ ํต์ ์ ์ํด ์ค๊ณ๋ ํ๋กํ ์ฝ์ ์๋๊ธฐ ๋๋ฌธ์ ์น์์ผ์ ๋นํ ๋ฐ๋ ๋ชป๋๋ค.
STOMP
โ STOMP(Simple Text Oriented Messaging Protocol)๋ ๊ฐ๋จํ ํ ์คํธ ๊ธฐ๋ฐ ๋ฉ์์ง ํ๋กํ ์ฝ๋ก ๋ฉ์ธ์ง์ ํ์ ๋ฑ์ ์ ์ํ๋ฉฐ sub/pub๋ก ๋ฉ์ธ์ง์ ์ก์์ ์ ๊ด๋ฆฌํ๋ค.
์ฃผ๋ก WebSocket์์ ์ฌ์ฉ๋๋๋ฐ ์ฌ์ค์ ๋ณ๊ฐ์ ํ๋กํ ์ฝ์ด๋ ํท๊ฐ๋ ค์ ์๋ ๊ฒ์ด๋ค.
Spring ์ฝ๋
Spring์ Stomp ์๋ฒ๋ก์ ์ญํ ์ ํ๊ฒ ๋๋ค
์ฌ์ค์ ์น์์ผ ๊ฐ์ฒด๋ฅผ ์์ฑํ๊ณ ์ฐ๊ฒฐ์ ๋งบ๋ ๊ฑด ํ๋ก ํธ์์ ํ๊ฒ ๋๊ธฐ ๋๋ฌธ์ ๋ฐฑ์๋ ์ฝ๋๋ ๋งค์ฐ ๋จ์ํ๋ค.
sub/pub๋ฅผ ์ค์ ํด์ฃผ๊ณ , ์ปจํธ๋กค๋ฌ์ ์น์์ผ ํต์ ์์ฒญ์ด ๋ค์ด์ค๋ฉด ์ค์ ํด๋ ์ฃผ์๋ก ๋ฉ์ธ์ง๋ฅผ ๋ณด๋ด๋ ๊ฒ ์ ๋ถ์ด๋ค.
WebSocketConfig.java
@Configuration
@EnableWebSocketMessageBroker
public class WebSocketConfig implements WebSocketMessageBrokerConfigurer {
@Override
public void configureMessageBroker(MessageBrokerRegistry config){
config.enableSimpleBroker("/sub"); // ๊ตฌ๋
์ฃผ์
config.setApplicationDestinationPrefixes("/pub"); // ๋ฐํ ์ฃผ์
}
@Override
public void registerStompEndpoints(StompEndpointRegistry registry) {
registry.addEndpoint("/ws") // ์น์์ผ ์ฃผ์
.setAllowedOriginPatterns("*"); // ํ์ฉ ๋๋ฉ์ธ
// .withSockJS();
}
}
โ SockJS๋ ๋ธ๋ผ์ฐ์ ์์ ์น์์ผ์ด ์ง์ ์ ๋ ๊ฒฝ์ฐ sockjs๋ก ๋์ด๊ฐ๋๋ก ํ๋ ๊ฑด๋ฐ ํ๋ก ํธ์์ ์ถ๊ฐ์ ์ธ ์ค์ ์ด ํ์ํ๊ณ , sockjs ์ค์ ์ ํ ๊ฒฝ์ฐ ์๋ ํ ์คํธ๊ฐ ์๋๋ฏ๋ก ๊ทธ๋ฅ ์ ํ๋ ๊ฒ์ ์ถ์ฒํ๋ค. ๋ฌด์๋ณด๋ค ์์ฆ์ ๋๋ถ๋ถ ๋ธ๋ผ์ฐ์ ๊ฐ ์น์์ผ ์ง์์ด ๋๋ค.
chatController.java
private final SimpMessageSendingOperations sendingOperations;
@MessageMapping("/send")
public void sendChat(@RequestBody ChatInsertRequestDto requestDto) throws ClassNotFoundException, NotFoundException {
String writerName = userService.getWriterName(requestDto.getWriterId());
chatService.insertChat(requestDto, writerName);
sendingOperations.convertAndSend("/sub/chat/" + requestDto.getGameId() + requestDto.getChatRoomId(), requestDto.toEntity(writerName));
}
โ @MessageMapping
์ @GetMapping
์ด๋ @PostMapping
๊ณผ ๊ฐ์ ์ญํ ์ ํ๋ค. ์์ฒญ์ด ๋ค์ด์์ ๋ ๋งคํํ ์ฃผ์๋ฅผ ์ ์ด์ค๋ค.
- ๋ฉ์ธ์ง ์ ์ก ์์ฒญ์ด ๋ค์ด์จ๋ค๋ ๊ฑด ๋ฐํ ์์ฒญ์ธ๋ฐ config์์ ์ค์ ํ ๋ฐํ ์ฃผ์์ธ 'pub'๋ ์์์ ์ธ์ํด์ฃผ๊ธฐ ๋๋ฌธ์ pub ๋ค์ ์ฌ ์ฃผ์๋ฅผ ์ ์ด์ฃผ๋ฉด ๋๋ค
- ์ ์ฒด์ฃผ์:
ws://{๋๋ฉ์ธ:ํฌํธ๋ฒํธ}/pub/{MessageMapping์ฃผ์}
โ simpMessageSendingOperation
์ ์ฃผ์
๋ฐ์์ ๋ณด๋ผ ์ฃผ์ (sub/{ํ์ ์ฃผ์}
)์ ๋ณด๋ผ ๋ด์ฉ์ ์ธ์๋ก ๋ฃ์ด์ ๋ณด๋ด์ฃผ๋ฉด ๋
ํ ์คํธํด๋ณด๊ธฐ
์น์์ผ์ ํ๋ก ํธ ํด๋ผ์ด์ธํธ๊ฐ ์์ด์ผํด์ ํ ์คํธ๊ฐ ๋งค์ฐ ์ด๋ ค์ด ํธ์ด๋ค.
๋ฉ์ธ์งํ๋ฅผ ๊ตฌํํด ๊ฒ์ฆํ๋ ๋ฐฉ๋ฒ์ด ์์ง๋ง ๋ณ๋์ ํ ์คํธ ์ฝ๋๊ฐ ๊ผญ ํ์ํ ๊ฒฝ์ฐ๊ฐ ์๋๋ผ๋ฉด ์ด๋ ค์์ ๊ถ์ฅ๋์ง ์๋๋ค.
ํฌ๋กฌ ์ต์คํ ์ APIC๋ฅผ ์ด์ฉํ๋ฉด Stomp์ ์น์์ผ์ ํ ์คํธ ํ ์ ์๋ค.
Apic - Complete API solution
The only tool you will ever need for all you API Design, Documentation and Testing needs.
chrome.google.com
'โญ Personal_Study > Spring' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
JPA์์ Wrapper class vs primitive type (0) | 2023.04.17 |
---|---|
Page ๊ฐ์ฒด์ Collections ๋ฉ์๋๋ฅผ ์ ์ฉํ์ ๋ UnsupportedOperationException ๋ฐ์ ์๋ฌ (0) | 2023.03.30 |
๋น ์๋ช ์ฃผ๊ธฐ ์ฝ๋ฐฑ (0) | 2023.01.10 |
์์กด๊ด๊ณ ์๋ ์ฃผ์ (1) | 2023.01.09 |
ComponentScan (0) | 2023.01.07 |
๋๊ธ