<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom"><channel><title>Spring Boot on Korobopolly's Dev Blog</title><link>https://korobopolly.github.io/series/spring-boot/</link><description>Recent content in Spring Boot on Korobopolly's Dev Blog</description><generator>Hugo</generator><language>ko</language><lastBuildDate>Mon, 16 Feb 2026 13:20:00 +0900</lastBuildDate><atom:link href="https://korobopolly.github.io/series/spring-boot/index.xml" rel="self" type="application/rss+xml"/><item><title>Spring Boot WebSocket - STOMP 프로토콜과 실시간 통신 구현</title><link>https://korobopolly.github.io/posts/network-websocket-stomp/</link><pubDate>Mon, 16 Feb 2026 13:20:00 +0900</pubDate><guid>https://korobopolly.github.io/posts/network-websocket-stomp/</guid><description>&lt;h2 id="http-폴링-vs-websocket"&gt;HTTP 폴링 vs WebSocket&lt;/h2&gt;
&lt;p&gt;실시간 통신을 구현하는 방법은 크게 두 가지입니다.&lt;/p&gt;
&lt;h3 id="http-폴링-polling"&gt;HTTP 폴링 (Polling)&lt;/h3&gt;
&lt;p&gt;클라이언트가 주기적으로 서버에 요청을 보내 새 데이터를 확인합니다.&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#cdd6f4;background-color:#1e1e2e;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-javascript" data-lang="javascript"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#6c7086;font-style:italic"&gt;// Short Polling
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;setInterval(() =&amp;gt; {
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; fetch(&lt;span style="color:#a6e3a1"&gt;&amp;#39;/api/messages&amp;#39;&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; .then(res =&amp;gt; res.json())
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; .then(data =&amp;gt; updateUI(data));
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;}, &lt;span style="color:#fab387"&gt;1000&lt;/span&gt;); &lt;span style="color:#6c7086;font-style:italic"&gt;// 1초마다 요청
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;&lt;strong&gt;단점:&lt;/strong&gt;&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;불필요한 요청이 많음 (데이터가 없어도 요청)&lt;/li&gt;
&lt;li&gt;서버 부하 증가 (동시 접속자 1000명 = 초당 1000건 요청)&lt;/li&gt;
&lt;li&gt;실시간성 낮음 (폴링 간격만큼 지연)&lt;/li&gt;
&lt;/ul&gt;
&lt;h3 id="long-polling"&gt;Long Polling&lt;/h3&gt;
&lt;p&gt;서버가 새 데이터가 있을 때까지 응답을 보류합니다.&lt;/p&gt;</description></item><item><title>Spring Boot 캐싱 - Caffeine, @Cacheable, TTL 전략</title><link>https://korobopolly.github.io/posts/spring-boot-caching/</link><pubDate>Mon, 16 Feb 2026 13:19:00 +0900</pubDate><guid>https://korobopolly.github.io/posts/spring-boot-caching/</guid><description>&lt;h2 id="캐싱이란"&gt;캐싱이란&lt;/h2&gt;
&lt;p&gt;캐싱(Caching)은 자주 사용되는 데이터를 빠르게 접근할 수 있는 임시 저장소에 보관하는 기술입니다. 데이터베이스 조회, 외부 API 호출, 복잡한 연산 결과 등을 메모리에 저장하여 동일한 요청에 대해 빠르게 응답할 수 있습니다.&lt;/p&gt;
&lt;h2 id="왜-캐싱이-필요한가"&gt;왜 캐싱이 필요한가&lt;/h2&gt;
&lt;h3 id="성능-향상"&gt;성능 향상&lt;/h3&gt;
&lt;p&gt;데이터베이스 조회는 일반적으로 수십~수백 밀리초가 걸리지만, 메모리 캐시는 마이크로초 단위로 응답합니다.&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#cdd6f4;background-color:#1e1e2e;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-java" data-lang="java"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#6c7086;font-style:italic"&gt;// 캐시 없이 매번 DB 조회&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#f38ba8"&gt;public&lt;/span&gt; User &lt;span style="color:#89b4fa"&gt;getUser&lt;/span&gt;(Long id) {
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#cba6f7"&gt;return&lt;/span&gt; userRepository.&lt;span style="color:#89b4fa"&gt;findById&lt;/span&gt;(id).&lt;span style="color:#89b4fa"&gt;orElseThrow&lt;/span&gt;(); &lt;span style="color:#6c7086;font-style:italic"&gt;// 평균 50ms&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;}
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#6c7086;font-style:italic"&gt;// 캐시 적용&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#89b4fa;font-weight:bold"&gt;@Cacheable&lt;/span&gt;(&lt;span style="color:#a6e3a1"&gt;&amp;#34;users&amp;#34;&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#f38ba8"&gt;public&lt;/span&gt; User &lt;span style="color:#89b4fa"&gt;getUser&lt;/span&gt;(Long id) {
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#cba6f7"&gt;return&lt;/span&gt; userRepository.&lt;span style="color:#89b4fa"&gt;findById&lt;/span&gt;(id).&lt;span style="color:#89b4fa"&gt;orElseThrow&lt;/span&gt;(); &lt;span style="color:#6c7086;font-style:italic"&gt;// 첫 조회 50ms, 이후 0.1ms&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;}
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;h3 id="비용-절감"&gt;비용 절감&lt;/h3&gt;
&lt;p&gt;외부 API 호출이나 복잡한 연산을 캐싱하면 서버 리소스와 비용을 절감할 수 있습니다.&lt;/p&gt;</description></item><item><title>Spring Boot 이벤트 아키텍처 - @EventListener, @Async, 트랜잭션 이벤트</title><link>https://korobopolly.github.io/posts/spring-boot-event-driven/</link><pubDate>Mon, 16 Feb 2026 13:18:00 +0900</pubDate><guid>https://korobopolly.github.io/posts/spring-boot-event-driven/</guid><description>&lt;h2 id="이벤트-기반-아키텍처란"&gt;이벤트 기반 아키텍처란&lt;/h2&gt;
&lt;p&gt;이벤트 기반 아키텍처(Event-Driven Architecture)는 시스템의 컴포넌트들이 이벤트를 통해 상호작용하는 설계 패턴입니다. 특정 작업이 완료되었을 때 이벤트를 발행하고, 관심 있는 컴포넌트가 이를 구독하여 처리하는 방식입니다.&lt;/p&gt;
&lt;p&gt;전통적인 방식에서는 서비스 A가 서비스 B, C, D를 직접 호출했다면, 이벤트 기반 방식에서는 서비스 A가 이벤트만 발행하고, B, C, D가 각자 필요한 이벤트를 구독하여 처리합니다.&lt;/p&gt;
&lt;h2 id="왜-이벤트-기반-아키텍처인가"&gt;왜 이벤트 기반 아키텍처인가&lt;/h2&gt;
&lt;h3 id="서비스-간-결합도-감소"&gt;서비스 간 결합도 감소&lt;/h3&gt;
&lt;p&gt;직접 호출 방식의 문제점:&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#cdd6f4;background-color:#1e1e2e;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-java" data-lang="java"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#89b4fa;font-weight:bold"&gt;@Service&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#f38ba8"&gt;public&lt;/span&gt; &lt;span style="color:#f38ba8"&gt;class&lt;/span&gt; &lt;span style="color:#f9e2af"&gt;UserService&lt;/span&gt; {
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#f38ba8"&gt;private&lt;/span&gt; &lt;span style="color:#f38ba8"&gt;final&lt;/span&gt; EmailService emailService;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#f38ba8"&gt;private&lt;/span&gt; &lt;span style="color:#f38ba8"&gt;final&lt;/span&gt; NotificationService notificationService;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#f38ba8"&gt;private&lt;/span&gt; &lt;span style="color:#f38ba8"&gt;final&lt;/span&gt; AuditService auditService;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#f38ba8"&gt;private&lt;/span&gt; &lt;span style="color:#f38ba8"&gt;final&lt;/span&gt; AnalyticsService analyticsService;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#f38ba8"&gt;public&lt;/span&gt; &lt;span style="color:#f38ba8"&gt;void&lt;/span&gt; &lt;span style="color:#89b4fa"&gt;registerUser&lt;/span&gt;(User user) {
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; userRepository.&lt;span style="color:#89b4fa"&gt;save&lt;/span&gt;(user);
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#6c7086;font-style:italic"&gt;// 모든 서비스를 직접 호출&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; emailService.&lt;span style="color:#89b4fa"&gt;sendWelcomeEmail&lt;/span&gt;(user);
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; notificationService.&lt;span style="color:#89b4fa"&gt;sendPushNotification&lt;/span&gt;(user);
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; auditService.&lt;span style="color:#89b4fa"&gt;logUserRegistration&lt;/span&gt;(user);
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; analyticsService.&lt;span style="color:#89b4fa"&gt;trackUserSignup&lt;/span&gt;(user);
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; }
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;}
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;위 코드는 &lt;code&gt;UserService&lt;/code&gt;가 4개의 서비스에 강하게 결합되어 있습니다. 새로운 기능 추가 시마다 &lt;code&gt;UserService&lt;/code&gt;를 수정해야 합니다.&lt;/p&gt;</description></item><item><title>Spring Boot REST API - Controller, JPA CRUD, 예외 처리</title><link>https://korobopolly.github.io/posts/spring-boot-rest-api/</link><pubDate>Mon, 16 Feb 2026 13:13:00 +0900</pubDate><guid>https://korobopolly.github.io/posts/spring-boot-rest-api/</guid><description>&lt;h2 id="rest-api-설계-원칙"&gt;REST API 설계 원칙&lt;/h2&gt;
&lt;p&gt;RESTful API는 HTTP 프로토콜을 기반으로 자원(Resource)을 URI로 표현하고, HTTP 메서드로 작업을 정의하는 아키텍처 스타일입니다. Spring Boot는 REST API 개발을 위한 강력한 기능을 제공합니다.&lt;/p&gt;
&lt;h3 id="rest의-핵심-원칙"&gt;REST의 핵심 원칙&lt;/h3&gt;
&lt;p&gt;&lt;strong&gt;1. 자원 기반 URI 설계&lt;/strong&gt;&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;명사를 사용하여 자원을 표현: &lt;code&gt;/users&lt;/code&gt;, &lt;code&gt;/products&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;계층 구조 표현: &lt;code&gt;/users/{id}/orders&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;복수형 사용 권장: &lt;code&gt;/users&lt;/code&gt; (O), &lt;code&gt;/user&lt;/code&gt; (X)&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;&lt;strong&gt;2. HTTP 메서드 활용&lt;/strong&gt;&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;GET: 조회&lt;/li&gt;
&lt;li&gt;POST: 생성&lt;/li&gt;
&lt;li&gt;PUT: 전체 수정&lt;/li&gt;
&lt;li&gt;PATCH: 부분 수정&lt;/li&gt;
&lt;li&gt;DELETE: 삭제&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;&lt;strong&gt;3. 상태 코드 활용&lt;/strong&gt;&lt;/p&gt;</description></item><item><title>Spring Boot 시작하기 - 프로젝트 생성, 구조, 첫 실행</title><link>https://korobopolly.github.io/posts/spring-boot-introduction/</link><pubDate>Mon, 16 Feb 2026 13:12:00 +0900</pubDate><guid>https://korobopolly.github.io/posts/spring-boot-introduction/</guid><description>&lt;h2 id="spring-boot란-무엇인가"&gt;Spring Boot란 무엇인가?&lt;/h2&gt;
&lt;p&gt;Spring Boot는 Spring 프레임워크를 기반으로 한 오픈소스 Java 프레임워크입니다. Spring의 강력한 기능을 그대로 사용하면서도, 복잡한 설정 없이 빠르게 프로덕션 레벨의 애플리케이션을 만들 수 있도록 설계되었습니다.&lt;/p&gt;
&lt;h3 id="spring-vs-spring-boot"&gt;Spring vs Spring Boot&lt;/h3&gt;
&lt;p&gt;&lt;strong&gt;전통적인 Spring Framework&lt;/strong&gt;의 경우:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;XML 기반의 복잡한 설정 파일 필요&lt;/li&gt;
&lt;li&gt;의존성 버전 관리를 직접 해야 함&lt;/li&gt;
&lt;li&gt;서버 배포를 위한 WAR 파일 생성 및 Tomcat 설치 필요&lt;/li&gt;
&lt;li&gt;설정 코드가 비즈니스 로직보다 많아지는 경우 발생&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;&lt;strong&gt;Spring Boot&lt;/strong&gt;는 이러한 문제를 해결합니다:&lt;/p&gt;</description></item></channel></rss>