
싱글톤 컨테이너
- 싱글톤의 문제점들을 해결한 것
- 스프링 컨테이너는 싱글톤 컨테이너 역할을 하는데 싱글톤 객체를 생성하고 관리하는 기능을 싱글톤 레지스트리라 한다.
- 스프링 컨테이너는 싱글톤 패턴의 문제점을 해결하면서 객체 인스턴스를 싱글톤으로 관리한다
- 스프링 컨테이너는 싱글톤 패턴을 적용하지 않아도 객체 인스턴스를 싱글톤으로 관리한다
- 스프링 컨테이너와 이런 기능 덕분에 싱글턴 패턴의 모든 단점을 해결하면서 객체를 싱글톤으로 유지 할 수 있음
- 싱글톤 패턴을 위한 지저분한 코드가 들어가지 않아도 됨
- DIP, OCP, 테스트, private 생성자로 부터 자유롭게 싱글톤을 사용할 수 있음
싱글톤 방식의 주의점
- 객체 인스턴스를 하나만 생성해서 공유하는 싱글톤 방식은 여러 클라이언트가 하나의 같은 객체 인스턴스를 공유하기 때문에 싱글톤 객체는 상태를 유지하게 설계하면 안됨
- 무상태(stateless)로 설계해야 한다
- 특정 클라이언트에 의존적인 필드가 있으면 안됨
- 특정 클라이언트가 값을 변경할 수 있는 필드가 있으면 안됨
- 가급적 읽기만 가능해야 한다
- 필드 대신에 자바에서 공유되지 않는 지역변수, 파라미터, ThreadLocal 등을 사용해야 한다
- 스프링 빈의 필드에 공유 값을 설정하면 큰 장애가 발생할 수 있음
싱글톤 필드 공유 장애 예시
public class StatefulService {
private int price;
public void order(String name, int price) {
System.out.println("name = "+name+" price = " + price);
this.price = price;
}
public int getPrice() {
return price;
}
}
@Test
void statefulServiceTest() {
ApplicationContext ac = new AnnotationConfigApplicationContext(TestConfig.class);
StatefulService statefulService1 = ac.getBean(StatefulService.class);
StatefulService statefulService2 = ac.getBean(StatefulService.class);
int userAstatefulService1.order("userA", 10000);
statefulService2.order("userB", 20000);
int price = statefulService1.getPrice();
System.out.println("price = "+price);
Assertions.assertThat(statefulService1.getPrice()).isEqualTo(20000);
}
static class TestConfig {
@Bean
public StatefulService statefulService() {
return new StatefulService();
}
}

싱글톤 필드 공유 장애 해결 예시
public class StatefulService {
public int order(String name, int price) {
System.out.println("name = "+name+" price = " + price);
return price;
}
}
@Test
void statefulServiceTest() {
ApplicationContext ac = new AnnotationConfigApplicationContext(TestConfig.class);
StatefulService statefulService1 = ac.getBean(StatefulService.class);
StatefulService statefulService2 = ac.getBean(StatefulService.class);
int userAPrice = statefulService1.order("userA", 10000);
int userBPrice = statefulService2.order("userB", 20000);
System.out.println("price = "+userAPrice);
Assertions.assertThat(userAPrice).isEqualTo(10000);
}
static class TestConfig {
@Bean
public StatefulService statefulService() {
return new StatefulService();
}
}