클래스의 인스턴스가 딱 1개만 생성되는 것을 보장하는 디자인 패턴이다.
따라서 객체 인스턴스를 2개 이상 생성하지 못하도록 private 생성자를 사용해서 외부에서 임의로 new 키워드를 사용하지 못하도록 막아야한다.
싱글톤 패턴을 적용하면 고객의 요청이 올 때 마다 객체를 생성하는 것이 아니라, 이미 만들어진 객체를 공유해서 효율적으로 사용할 수 있다. 하지만 싱글톤 패턴은 다음과 같은 수 많은 문제점들을 가지고 있다.
이러한 문제점을 해결하고 싱글톤 패턴을 적용시킨것이 Singleton Contaioner(Spring Container)이다.
스프링 컨테이너 덕분에 고객의 요청이 올 때 마다 객체를 생성하는 것이 아니라, 이미 만들어진 객체를
공유해서 효율적으로 재사용할 수 있다.
@Configuration
public class AppConfig {
@Bean
public MemberService memberService() {
return new MemberServiceImpl(memberRepository());
}
@Bean
public OrderService orderService() {
return new OrderServiceImpl(
memberRepository(),
discountPolicy());
}
@Bean
public MemberRepository memberRepository() {
return new MemoryMemberRepository();
}
...
}
위 코드에서 memberRepository의 경우
MemberService 에서 1번
OrderService 에서 1번
memberRepository 에서 1번
총 3번의 호출로 인해 각각 다른 memberRepository가 생성되기 때문에 싱글톤이 깨지는 것처럼 보인다.
그러나 막상 테스트를 해보면 memberRepository 인스턴스는 모두 같은 인스턴스가 공유되어 사용된다.
@Configuration을 적용하게 되면 해당 클래스를 상속받은 임의의 다른 클래스를 만들고 , 그 다른 클래스를 스프링 빈으로 등록한다.
이 임의의 다른 클래스가 싱글톤이 보장되도록 해준다.
@Bean이 붙은 매서드마다 이미 스프링 빈이 존재하면 존재하는 빈을 반환하고, 없으면 생성해서 스프링 빈으로 등록하고 반환하는 코드가 동적으로 만들어지는 덕에, 싱글톤이 보장되는 것이다.
@Bean만 사용해도 스프링 빈으로 등록은 되지만, 싱글톤을 보장하지는 않는다. 따라서 스프링 정보는 항상 @Configuration을 사용해야한다.