@Configuration & Singleton
@Configuration
public class AppConfig {
@Bean
public MemberService memberService() {
return new MemberServiceImpl(memberRepository());
}
@Bean
public MemberRepository memberRepository() {
return new MemoryMemberRepository();
}
- memberService 빈을 만드는 코드를 보면 memberRepository() 호출
- 메서드를 호출하면 new MemoryMemberRepository() 를 호출
- orderService 빈을 만드는 코드도 동일하게 memberRepository() 호출
- 메서드를 호출하면 new MemoryMemberRepository() 를 호출
- 결과적으로 각각 다른 2개의 MemoryMemberRepository 가 생성
인스턴스 확인
- 확인해보면 memberRepository 인스턴스는 모두 같은 인스턴스 공유
- AppConfig 의 자바 코드를 보면 다른 인스턴스가 생성될거라 예상
@Configuration 과 바이트코드 조작의 마법
- 스프링 컨테이너는 싱글톤 레지스트리
- 스프링 빈이 싱글톤이 되도록 보장해야한다.
- 스프링이 자바 코드까지 조작하기는 어렵다.
- → 스프링은 클래스의 바이트 코드를 조작하는 라이브러리 사용
클래스 정보 확인
- AppConfig 스프링 빈 조회
- 예상된 순수한 클래스
- class spring.core.AppConfig
- 예상과 다르게 클래스 명에 CGLIB 가 붙는다.
- class spring.core.AppConfigEnhancerBySpringCGLIBdb479d70
- 스프링이 CGLIB 라는 바이트 코드 조작 라이브러리를 사용해서 해당 클래스를 상속 받은 임의의 다른 클래스를 만들고 스프링 빈으로 등록
AppConfig@CGLIB 예상 코드
@Bean
public MemberRepository memberRepository() {
if (memoryMemberRepository가 이미 스프링 컨테이너에 등록되어 있으면?) {
return 스프링 컨테이너에서 찾아서 반환;
} else {
기존 로직을 호출해서 MemoryMemberRepository를 생성하고 스프링 컨테이너에 등록
return 반환
}
}
@Bean
이 붙은 메서드마다 이미 스프링 빈이 존재하면 존재하는 빈을 반환하고, 스프링 빈이 없으면 생성해서 스프링 빈으로 등록하고 반환하는 코드가 동적으로 만들어진다.
- 덕분에 싱글톤이 보장된다.
@Configuration 정리
@Bean
만 사용해도 스프링 빈으로 등록은 되지만, 싱글톤을 보장하지 않는다.
- 스프링 설정 정보는 항상
@Configuration
사용