hoon's bLog

Spring gradle project 스프링 핵심 원리 기본편 | BeanFactory, ApplicationContext 이해 및 차이 본문

IT/Spring

Spring gradle project 스프링 핵심 원리 기본편 | BeanFactory, ApplicationContext 이해 및 차이

개발한기발자 2024. 4. 18. 10:52
반응형


본 포스팅은 인프런에 있는 인터넷 강좌인,

김영한 강사님의 스프링 핵심 원리 기본편을 공부하며,

개인적으로 공부하고, 정리하는 용도로 포스팅을 해보겠다.

 

Spring gradle project 환경설정 및 회원 가입 서비스 예제 만들기
Spring gradle project 주문/할인 도메인 설계

Spring gradle project 객체 지향 원리 적용

Spring gradle project AppConfig 리팩토링 OCP 위반 해결 및 중복 제거

Spring gradle project 좋은 객체 지향 설계 5가지 원칙 적용 및 스프링 전환

Spring gradle project 스프링 컨테이너와 스프링 빈

 

4. 스프링 컨테이너와 스프링 빈

저번 포스팅에 이어 이번 포스팅에서는,

스프링 Bean 조회를 통해 중복 및 상속 관계에 대해 알아보겠다.

스프링 빈 조회 - 동일한 타입이 둘 이상

  • 타입으로 조회 시 같은 타입의 스프링 빈이 둘 이상이면 오류가 발생하는데, 이때 빈 이름을 별도로 지정해야 한다.
  • ac.getBeansOfType()을 사용하면 해당 타입의 모든 빈을 조회할 수 있다
import hello.core.member.MemberRepository;
import hello.core.member.MemoryMemberRepository;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.NoUniqueBeanDefinitionException;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import java.util.Map;

import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.jupiter.api.Assertions.assertThrows;

public class ApplicationContextSameBeanFindTest {
    AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(SameBeanConfig.class);

    @Test
    @DisplayName("타입으로 조회시 같은 타입이 둘 이상 있으면, 중복 오류가 발생한다")
    void findBeanByTypeDuplicate(){
    //  MemberRepository bean = ac.getBean(MemberRepository.class);
        assertThrows(NoUniqueBeanDefinitionException.class, () -> ac.getBean(MemberRepository.class));
    }

    @Test
    @DisplayName("타입으로 조회시 같은 타입이 둘 이상 있으면, 빈 이름을 지정하면 된다")
    void findBeanByName(){
        MemberRepository memberRepository = ac.getBean("memberRepository1", MemberRepository.class);
        assertThat(memberRepository).isInstanceOf(MemberRepository.class);
    }

    @Test
    @DisplayName("특정 타입을 모두 조회하기")
    void findAllBeanByType(){
        Map<String, MemberRepository> beansOfType = ac.getBeansOfType(MemberRepository.class);
        for (String key : beansOfType.keySet()) {
            System.out.println("key = " + key + " value = " + beansOfType.get(key));
        }
        
        System.out.println("beansOfType = " + beansOfType);
        assertThat(beansOfType.size()).isEqualTo(2);
    }

    @Configuration
    static class SameBeanConfig {
        @Bean
        public MemberRepository memberRepository1(){
            return new MemoryMemberRepository();
        }

        @Bean
        public MemberRepository memberRepository2(){
            return new MemoryMemberRepository();
        }
    }
}
  • findBeanByTypeDuplicate() : 같은 타입(MemberRepository)의 빈이 두 개 이상 있는 경우에 ac.getBean(MemberRepository.class) 호출로 인해 NoUniqueBeanDefinitionException 예외가 발생하는지 테스트한다.
  • findBeanByName() : 타입이 중복되어 있는 상황에서 빈의 이름을 명시적으로 지정하여 빈을 정상적으로 조회할 수 있는지 테스트 memberRepository1라는 이름으로 MemberRepository 타입의 빈을 정상적으로 가져오는지 확인하고, 가져온 객체가 MemberRepository 타입의 인스턴스인지 검증한다.
  • findAllBeanByType() : MemberRepository 타입의 모든 빈을 조회하고, getBeansOfType(MemberRepository.class)를 사용하여 해당 타입의 모든 빈을 Map 형태로 가져오며, 이 Map의 크기가 2인지 확인하여 두 개의 빈이 정상적으로 등록되었는지 테스트한다.
    또한, 이 Map의 키(빈의 이름)와 값(빈의 인스턴스)을 출력하여 확인할 수 있다.

스프링 빈 조회 - 상속 관계

  • 부모 타입으로 조회하면, 자식 타입도 함께 조회한다.
  • 자바 객체의 최고 부모인 Object 타입으로 조회하면, 모든 스프링 빈을 조회한다
import hello.core.discount.DiscountPolicy;
import hello.core.discount.FixDiscountPolicy;
import hello.core.discount.RateDiscountPolicy;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.NoUniqueBeanDefinitionException;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import java.util.Map;

import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.jupiter.api.Assertions.assertThrows;

public class ApplicationContextExtendsFindTest {
    AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(TestConfig.class);

    @Test
    @DisplayName("부모 타입으로 조회시, 자식이 둘 이상 있으면, 중복 오류가 발생한다.")
    void findBeanByParentTypeDuplicate(){
        assertThrows(NoUniqueBeanDefinitionException.class, () -> ac.getBean(DiscountPolicy.class));
    }

    @Test
    @DisplayName("부모 타입으로 조회시, 자식이 둘 이상 있으면, 빈 이름을 지정하면 된다.")
    void findBeanByParentTypeBeanName(){
        DiscountPolicy rateDiscountPolicy = ac.getBean("rateDiscountPolicy", DiscountPolicy.class);
        assertThat(rateDiscountPolicy).isInstanceOf(RateDiscountPolicy.class);
    }

    @Test
    @DisplayName("특정 하위 타입으로 조회 -> 비추하는 방법...")
    void findBeanBySubType(){
        RateDiscountPolicy rateDiscountPolicy = ac.getBean(RateDiscountPolicy.class);
        assertThat(rateDiscountPolicy).isInstanceOf(RateDiscountPolicy.class);
    }

    @Test
    @DisplayName("부모 타입으로 모두 조회하기")
    void findAllBeanByParentType(){
        Map<String, DiscountPolicy> beansOfType = ac.getBeansOfType(DiscountPolicy.class);
        assertThat(beansOfType.size()).isEqualTo(2);
        for (String key : beansOfType.keySet()) {
            System.out.println("key = " + key + " value = " + beansOfType.get(key));
        }
    }

    @Test
    @DisplayName("부모 타입으로 모두 조회하기 - Object <- 모든 객체는 object 타입이기 때문에 스프링 빈에 등록된거 다나옴...")
    void findAllBeanByObjectType(){
        Map<String, Object> beansOfType = ac.getBeansOfType(Object.class);
        for (String key : beansOfType.keySet()) {
            System.out.println("key = " + key + " value = " + beansOfType.get(key));
        }
    }

    @Configuration
    static class TestConfig {
        @Bean
        public DiscountPolicy rateDiscountPolicy(){ // <- 부모타입으로 선언하는걸 지향
            return new RateDiscountPolicy();
        }

        @Bean
        public DiscountPolicy fixDiscountPolicy(){
            return new FixDiscountPolicy();
        }

    }
}
  • findBeanByParentTypeDuplicate() : 부모 타입 DiscountPolicy로 빈을 조회할 때 같은 타입의 빈이 두 개 이상 있으면 NoUniqueBeanDefinitionException이 발생하는지 테스트
  • findBeanByParentTypeBeanName() : 부모 타입으로 조회할 때 빈의 이름을 명시하여 특정 빈(rateDiscountPolicy)을 정상적으로 가져오는지 확인하고, 가져온 빈이 RateDiscountPolicy 클래스의 인스턴스인지 확인한다.
  • findBeanBySubType() : 특정 하위 타입 RateDiscountPolicy로 직접 조회한다.
    (이 방법은 특정 구현에 의존하기 때문에 일반적으로 권장되지 않는다.)
  • findAllBeanByParentType() : DiscountPolicy 타입의 모든 빈을 조회해서, 조회된 빈의 수가 정확히 2개인지 확인하고, 각 빈의 이름과 인스턴스를 출력하여 확인한다.
  • findAllBeanByObjectType() : 모든 스프링 빈은 기본적으로 Object의 하위 타입이므로, 애플리케이션 컨텍스트에 등록된 모든 객체를 반환한다.

BeanFactory와 ApplicationContext

BeanFactory, ApplicationContext 모두 Spring Container인데,

과연 각자의 기능은 무엇이고 차이는 무엇일까?

BeanFactory

  • 스프링 컨테이너의 최상위 인터페이스
  • 스프링 빈을 관리하고 조회하는 역할을 담당
  • getBean()을 제공(지금까지 우리가 사용했던 대부분의 기능은 BeanFactory가 제공하는 기능이다.)

ApplicationContext

  • BeanFactory 기능을 모두 상속받아서 제공한다.
  • 빈을 관리하고 검색하는 기능을 BeanFactory가 제공해 주는데, 애플리케이션을 개발할 때는 ApplicationContext가 빈을 관리는 물론이고, 수많은 부가기능을 제공한다.
  • 때문에 BeanFactory보다는 ApplicationContext 사용하는 것을 지향한다!

ApplicatonContext가 제공하는 부가기능

  • MessageSource : 메시지소스를 활용한 국제화 기능, 예를 들어서 한국에서 들어오면 한국어로, 영어권에서 들어오면 영어로 출력
  • EnvironmentCapable : 환경변수(로컬, 개발, 운영 등)를 구분해서 처리
  • ApplicationEventPublisher : 이벤트를 발행하고 구독하는 모델을 편리하게 지원
  • ResourceLoader : 편리한 리소스 조회 : 파일, 클래스패스, 외부 등에서 리소스를 편리하게 조회

스프링 빈 설정 메타 정보도 같이 해보려 했는데,

정리하다 보니 생각보다 양이 많다.

다음 포스팅에서는 스프링 빈 설정 및 메타 정보에 대해서 정리하도록 하겠다!

 

끝.

 

728x90
반응형