본문 바로가기
Learning-log/Spring & JPA

(스프링 핵심 원리 - 기본편)스프링 빈 조회- 4-(4)동일한 타입이 둘 이상, (5)상속관계, (6) BeanFactory와 ApplicationContext

by why제곱 2023. 4. 2.

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

  • 타입으로 조회 시 같은 타입의 스프링 빈이 둘 이상이면 오류가 발생하므로 이 때는 빈 이름을 지정해줘야 함

  • 같은 타입이 둘 이상 있을 때, getBean()을 하면 위와 같은 오류가 발생

 

- 상속관계

  • 빈을 조회할 때, 상속관계로 되어있다면 ? 부모 타입으로 조회하면 자식 타입도 함께 조회됨
  • 자바 모든 객체의 최고 부모인 Object타입으로 조회하면 모든 스프링 빈을 조회하게 됨.
package hello.core.beanfind;

import hello.core.AppConfig;
import hello.core.discount.DiscountPolicy;
import hello.core.discount.FixDiscountPolicy;
import hello.core.discount.RateDiscountPolicy;
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.beans.factory.config.BeanDefinition;
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.AssertionsForClassTypes.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 findBeanByParentBeanName() {
        DiscountPolicy rateDiscountPolicy = ac.getBean("rateDiscountPolicy", DiscountPolicy.class);
        assertThat(rateDiscountPolicy).isInstanceOf(RateDiscountPolicy.class);
    }

    @Test
    @DisplayName("특정 하위타입으로 조회")
    void findBeanBySubType() {
        RateDiscountPolicy bean = ac.getBean(RateDiscountPolicy.class);
        assertThat(bean).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")
    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();
        }
    }
}
  • 실무에서는 위와 같이 sysout 으로 직접 출력하는 일 거의 없음 !! 

 

- BeanFactory와 ApplicationContext

 

  • BeanFactory : Interface 
    • 스프링 컨테이너의 최상위 인터페이스
    • 빈을 관리하고 조회하는 역할
    • getBean() 제공
  • ApplicationContext : BeanFactory를 상속 받은 Interface, 즉, BeanFactory에 부가기능을 더한 Interface라 볼 수 있음.
    • 빈을 관리하고 검색하는 기능은 이미 BeanFactory가 해주는데 차이점이 뭘까? 
    • 애플리케이션을 개발할 때 빈은 관리하고 조회하는 기능은 물론이고 수많은 부가기능이 필요
    • ApplicationContext는 BeanFactory 뿐만 아니라 여러 Interface들을 상속받고 있음
      • MessageSource : 국제화 기능, 한국에서 들어오면 한국어로, 영어권에서 들어오면 영어로 출력
      • EnviornmentCapable : 환경변수, 로컬, 개발, 운영 등을 구분해서 분리
      • ApplicationEventPublisher : 애플리케이션 이벤트, 이벤트를 발행하고 구독하는 모델을 편리하게 지원
      • ResourceLoader : 편리한 리소스 조회, 파일, 클래스패스, 외부 등에서 리소스를 편리하게 조회