내돈내고 내가 공부한것을 올리며, 중요한 단원은 저 자신도 곱씹어 볼겸 상세히 기록하고 얕은부분들은 가겹게 포스팅 하겠습니다.
1. 싱글톤 컨테이너
스프링 컨테이너는 싱글톤 패턴을 적용하지 않아도 객체 인스턴스를 싱글톤으로 관리한다. 이렇게 싱글톤 객체를 생성하고 관리하는 기능을
싱글톤 레지스트리 라고 한다.
이러한 기능 덕에 싱글톤을 위한 지저분한 코드가 들어갈 필요도 없으며, DIP, OCP, 테스트 등 의 제약사항이 없어진다 할 수 있다!
간단한 테스트 하나 해보자~
@Test
@DisplayName("스프링 컨테이너와 싱글톤")
void springContainer(){
ApplicationContext ac = new AnnotationConfigApplicationContext(AppConfig.class);
// 1. 조회: 호출할 때 마다 객체를 생성
MemberService memberService1 = ac.getBean("memberService", MemberService.class);
// 2. 조회: 호출할 때 마다 객체를 생성
MemberService memberService2 = ac.getBean("memberService", MemberService.class);
// 참조값이 같은 것을 확인
System.out.println("memberService1 = " + memberService1);
System.out.println("memberService2 = " + memberService2);
assertThat(memberService1).isSameAs(memberService2);
}
실행시 memberService1, 2 모두 동일한 객체임을 알 수 있다!
2. 싱글톤 방식의 주의점
싱글톤 패턴이든, 스프링 같은 싱글톤 컨테이너를 사용하든, 객체 인스턴스를 하나만 생성해서 공유하는 싱글톤 방식은 여러 클라이언트가 하나의 같은 객체 인스턴스를 공유하기 때문에 싱글톤 객체는 상태를 유지(stateful)하게 설계하면 안된다.
무상태(stateless)를 유지해야 한다!
● 특정 클라이언트에 의존적인 필드가 있으면 안된다.
● 특정 클라이언트가 값을 변경할 수 있는 필드가 있으면 안된다!
● 가급적 읽기만 가능해야 한다.
● 필드 대신에 자바에서 공유되지 않는, 지역변수, 파라미터, ThreadLocal 등을 사용해야 한다.
멀티 스레드 상에서 크리티컬섹션 부분을 여러 스레드들이 동시에 접근해 발생하는 문제를 없애햐 한다는 의미인 것 같다.
간단한 예를 살펴보자.
class StatefulServiceTest {
@Test
void statefulServiceSingleton() {
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(TestConfig.class);
StatefulService statefulService1 = ac.getBean(StatefulService.class);
StatefulService statefulService2 = ac.getBean(StatefulService.class);
// ThreadA: A사용자 10000원 주문
statefulService1.order("userA", 10000);
// ThreadB: B사용자 20000원 주문
statefulService2.order("userB", 20000);
int price1 = statefulService1.getPrice();
System.out.println("price1 = " + price1);
Assertions.assertThat(statefulService1.getPrice()).isEqualTo(20000);
}
static class TestConfig{
@Bean
public StatefulService statefulService(){
return new StatefulService();
}
}
}
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;
}
}
결과는 어떻게 나올까? 다음과 같다.
논리상 10000원 이여야 할 것 같지만 실상은 2만원 이다.
왜냐? StatefulService의 price 변수가 모든 인스턴스에서 공유되는 변수가 되기 때문이다.
따라서 지역변수로 값을 설정후 반환하는 방식등을 이용하여 stateless하게 설계해야 한다.
3. @Configuration과 싱글톤
우리의 AppConfig를 잠시 확인해 보자.
@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();
}
...
}
위의 코드를 보면 처음 등록될때 memberService() -> memberRepository() 1번, orderService() -> memberRepository() 1번
합쳐서 총 2번 memberRepository()이 호출되는 것 같다.
이러면 객체가 2개 생기는 거 아닌가? 싱글톤이 깨지는 거 아닌가?
다음 테스트를 확인해보자.
public class ConfigurationSingletonTest {
@Test
void configuration() {
ApplicationContext ac = new AnnotationConfigApplicationContext(AppConfig.class);
MemberServiceImpl memberService = ac.getBean("memberService", MemberServiceImpl.class);
OrderServiceImpl orderService = ac.getBean("orderService", OrderServiceImpl.class);
MemberRepository memberRepository = ac.getBean("memberRepository", MemberRepository.class);
System.out.println("memberService -> memberRepository = " + memberService.getMemberRepository());
System.out.println("orderService -> memberRepository = " + orderService.getMemberRepository());
System.out.println("memberRepository = " + memberRepository);
assertThat(memberService.getMemberRepository()).isSameAs(memberRepository);
assertThat(orderService.getMemberRepository()).isSameAs(memberRepository);
}
}
실행결과는 다음과 같다.
3개 다 같은 객체를 호출하고 있다. 어떻게 된거지? 다음 단락에서 확인해 보자!
4. @Configuration과 바이트코드 조작의 마법
스프링 컨테이너는 싱글톤 레지스트리다. 따라서 스프링 빈이 싱글톤이 되도록 보장해주어야 한다.
그런데 스프링이 자바 코드까지 어떻게 하기는 어렵다. 저 자바 코드를 보면 분명 3번 호출되어야 하는 것이 맞다.
그래서 스프링은 클래스의 바이트코드를 조작하는 라이브러리를 사용한다. 모든 비밀은 @Configuration 을 적용한 AppConfig 에 있다.
@Test
void configurationDeep() {
ApplicationContext ac = new AnnotationConfigApplicationContext(AppConfig.class);
AppConfig bean = ac.getBean(AppConfig.class);
System.out.println("bean = " + bean.getClass());
}
설정파일로 넘긴 AppConfig.class 안에 있는 것들이 빈으로 등록된다. 또한 AppConfig자체도 빈으로 등록이 된다.
위의 테스트 결과를 출력하면 어떻게 나올가? 예상되는 결과는 class hello.core.AppConfig 이다.
위의 테스트 결과를 보면 예상했던 결과와는 달리 뒤에 클래스 명에 xxxCGLIB가 붙으면서 상당히 복잡해진 것을 볼 수 있다.
만든 클래스가 아니라 스프링이 CGLIB라는 바이트코드 조작 라이브러리를 사용해서 AppConfig 클래스를 상속받은 임의의 다른 클래스를 만들고, 그 다른 클래스를 스프링 빈으로 등록한 것이다!
그림으로 확인해 보자!
원래의 AppConfig를 상속한 다른 AppConfig가 빈으로 등록되며, 싱글톤이 되도록 보장해 준다!
AppConfig@CGLIB 예상 코드는 다음과 같다.
@Bean
public MemberRepository memberRepository() {
if (memoryMemberRepository가 이미 스프링 컨테이너에 등록되어 있으면?) {
return 스프링 컨테이너에서 찾아서 반환;
} else { //스프링 컨테이너에 없으면 기존 로직을 호출해서 MemoryMemberRepository를 생성하고 스프링 컨테이너에 등록
return 반환
}
}
@Bean이 붙은 메서드마다 이미 스프링 빈이 존재하면 존재하는 빈을 반환하고, 스프링 빈이 없으면 생성해서 스프링 빈으로 등록하고 반환하는 코드가 동적으로 만들어진다. 덕분에 싱글톤이 보장되는 것이다.
마지막으로 궁금한점이 있다.
만약 @Configuration을 적용하지 않는다면 어떻게 될까? 딱 @Bean만 적용하면 말이다!
=> 이렇게 되면 스프링 빈으로 등록은 되지만, 싱글톤을 보장하지 않는다.
직전단락 위에서 3개의 객체가 모두 같은 인스턴스였다면, 이번에는 3개 모두 다른 인스턴스를 호출하고 있다. 3번 모두다 new 를 직접 호출하기 때문이다.
크게 고민할 것이 없다. 스프링 설정 정보는 항상 @Configuration 을 사용하자.
'BackEnd > Spring' 카테고리의 다른 글
[Spring] 의존관계 자동 주입 (0) | 2022.02.05 |
---|---|
[Spring] 컴포넌트 스캔 (0) | 2022.02.04 |
[Spring] 스프링 컨테이너와 스프링 빈 (0) | 2022.02.01 |
[Spring] 스프링 핵심 원리 이해2 - 객체 지향 원리 적용 (0) | 2022.02.01 |
[Spring] 스프링 핵심 원리 이해1 - 예제 만들기 (0) | 2022.01.31 |
댓글