반응형

빈 스코프란?

  • 스코프는 빈이 존재할 수 있는 범위를 뜻한다.
  • 스프링은 다음과 같은 다양한 스코프를 지원한다.
    • 싱글톤: 기본 스코프, 스프링 컨테이너의 시작과 종료까지 유지되는 가장 넓은 범위의 스코프이다.
    • 프로토타입: 스프링 컨테이너는 프로토타입 빈의 생성과 의존관계 주입까지만 관여하고 더는 관리하지 않는 매우 짧은 범위의 스코프이다.
    • 웹 스코프
      • request: 웹 요청이 들어오고 나갈때 까지 유지되는 스코프이다.
      • session: 웹 세션이 생성되고 종료될 때 까지 유지되는 스코프이다.
      • application: 웹의 서블릿 컨텍스와 같은 범위로 유지되는 스코프이다.
  • 빈 스코프는 @Scope 를 사용하여 지정할 수 있다.
    • 컴포넌트 스캔 자동 등록
    @Scope("prototype")
    @Component
    public class HelloBean {}
    

    • 수동 등록
    @Scope("prototype")
    @Bean
    PrototypeBean HelloBean() {
     return new HelloBean();
    }

프로토타입 스코프

  • 싱글톤 스코프의 빈을 조회하면 스프링 컨테이너는 항상 같은 인스턴스의 스프링 빈을 반환한다. 반면에 프로토타입 스코프를 스프링 컨테이너에 조회하면 스프링 컨테이너는 항상 새로운 인스턴스를 생성해서 반환한다.
  • 핵심은 스프링 컨테이너는 프로토타입 빈을 생성하고, 의존관계 주입, 초기화까지만 처리한다.
  • 클라이언트에 빈을 반환한 이후에는 컨테이너가 생성된 프로토타입 빈을 관리하지 않는다.
  • 프로토타입 빈을 관리할 책임은 프로토타입 빈을 받은 클라이언트에 있다.
    • @PreDestory 같은 종료 메서드가 호출되지 않는다. 클라이언트에서 직접 호출해주어야 한다.

싱글톤과 프로토타입 빈을 같이 사용할 경우의 문제점

 

  • 문제는 싱글톤 스코프인 clientBean 내부에서 사용하는 프로토타입 빈이 싱글톤 생성 시점에 주입받은 것을 그대로 사용한다는 점이다.
    • 싱글톤 생성 시점에 주입받은 프로토타입 빈이 유지가 된다.
    • clientBean 내부에서 사용하는 빈은 프로토타입 스코프로 설정하였지만 사용 할 때마다 프로토타입 빈이 새로 생성되지 않는다.

 

싱글톤과 프로토타입 빈을 같이 사용할 경우의 문제해결

  • 실제로 프로토타입 빈 사용시점에 새로운 프로토타입 빈을 생성하도록 해야한다.
  • 즉, 의존관계를 외부에서 주입받지않고 필요한 의존관계를 필요한 시점에 찾아야 한다. 이를 Dependency Lookup(DL) 의존관계 조회(탐색) 이라한다.

 

  • 1. 스프링 컨테이너를 의존성 주입받아 프로토타입 빈을 조회하는 방법
    • 스프링의 애플리케이션 컨텍스트 전체를 주입받게 되면, 스프링 컨테이너에 종속적인 코드가되고, 단위 테스트도 어려워진다.

 

  • 2. 프로바이더를 이용한 프로토타입 빈 조회 방법
    • 지정한 프로토타입 빈을 컨테이너에서 대신 찾아준다.(DL)
    • 지금 상황에서 필요한 기능만을 제공하여 적절하다.
  • ObjectFactory
    • DL 서비스를 제공하는 프로바이더
    • 기능이 단순하고 별도의 라이브러리 추가가 필요 없다, 스프링에 의존적이다.
    • 스프링이 제공하는 기능을 사용하지만, 기능이 단순하므로 단위테스트를 만들거나 mock 코드를 만들기는 훨씬 쉬워진다.
    @Autowired
    private ObjectFactory<PrototypeBean> prototypeBeanFactory;
    PrototypeBean prototypeBean = prototypeBeanFactory.getObject();

 

  • ObjectProvider
    • DL 서비스를 제공하는 프로바이더
    • ObjectFactory를 상속받아 옵션, 스트림 처리등 편의 기능이 추가되었다.
    • 별도의 라이브러리 추가가 필요 없다, 스프링에 의존적이다.
    • 스프링이 제공하는 기능을 사용하지만, 기능이 단순하므로 단위테스트를 만들거나 mock 코드를 만들기는 훨씬 쉬워진다.
    @Autowired
    private ObjectProvider<PrototypeBean> prototypeBeanProvider;
    PrototypeBean prototypeBean = prototypeBeanProvider.getObject();

 

 

  • Provider
    • JSR-330 자바 표준
    • provider 의 get() 을 호출하면 내부에서는 스프링 컨테이너를 통해 해당 빈을 찾아서 반환한다. (DL)
    • javax.inject:javax.inject:1 라이브러리 추가가 필요하다.
    • 자바 표준이므로 스프링이 아닌 다른 컨테이너에서도 사용할 수 있다. 또한 get() 메서드 하나로 기능이 매우 단순하다.
      • 단위테스트를 만들거나 mock 코드를 만들기는 훨씬 쉬워진다.
    @Autowired
    private Provider<PrototypeBean> prototypeBeanProvider;
    PrototypeBean prototypeBean = prototypeBeanProvider.get();

[참고자료]

스프링 핵심 원리 - 기본편, 김영한

반응형

+ Recent posts