싱글톤(Singleton)방식 주의점

jooog·2022년 5월 5일
0

스프링

목록 보기
24/26
post-thumbnail

싱글톤 방식을 사용할때 고려해야할 점

무상태(stateless)로 설계할 것

싱글톤 패턴이나 스프링의 싱글톤 컨테이너를 사용하는 방식으로 설계할 때는 주의해야할 점이 있다. 하나의 객체 인스턴스를 공유하기 때문에 싱글톤 객체는 상태를 유지(stateful)하지 않게 무상태(stateless)로 설계해야 한다.

공유값을 설정하지 말것

또한, 스프링 빈의 필드에 공유 값을 설정하면 큰 장애가 발생할 수 있기 때문에 이 점도 주의해야 한다.

코드로 살펴보기

StatefulService 클래스 생성

public class StatefulService {

    private int price; //상태를 유지하는 필드

    public void order(String name, int price){
        System.out.println("name = " + name);
        System.out.println("price = " + price);

        this.price = price; //문제
    }

    public int getPrice(){
        return price;
    }
}

StatefulServiceTest

스프링 컨테이너에 bean으로 등록된 객체는 싱글톤으로 관리된다. 해당 코드에서는 StatefulService 클래스가 bean으로 등록되어 있다.

public class StatefulServiceTest {

    @Test
    void StatefulServiceTest(){
        //스프링 컨테이너 생성
        ApplicationContext ac = new AnnotationConfigApplicationContext(TestConfig.class);
        StatefulService statefulService1 = ac.getBean(StatefulService.class);
        StatefulService statefulService2 = ac.getBean(StatefulService.class);

        //사용자A 10000원 주문
        statefulService1.order("userA", 10000);

        //사용자B 20000원 주문
        statefulService2.order("userB", 20000);

        //사용자A가 주문금액을 조회한다
        int orderPrice = statefulService1.getPrice();
        System.out.println("orderPrice = " + orderPrice);

    }

스프링 빈으로 등록된 객체에 두번의 요청이 오는데 userA, userB가 각각 다른 주문을 했다고 가정하고 orderPrice를 조회하면 가장 마지막에 설정한 금액인 20000원으로 출력되는 것을 확인할 수 있다.

사용자A의 주문 금액을 조회하면 10000원이 출력될 것으로 예상하지만 userB가 주문한 금액인 20000원이 출력된다.

그 이유는 간단하다. 스프링 컨테이너에 bean으로 등록된 객체는 한번만 생성되기 때문에 statefulService1와 statefulService2는 동일한 객체이고 마지막에 설정한 금액인 20000원으로 설정되기 때문이다.

이 글은 김영한님의 스프링 핵심 원리 강의를 듣고 정리한 내용입니다.

0개의 댓글