반응형
이게 중요하다, 빈을 조회할때 상속관계로 되어있으면
스프링빈을 조회할때 기본 대원칙은 부모 타입으로 조회하면, 자식 타입도 함께 조회되는거다. 자식 타입은 그냥 다 끌려나온다
그래서 모든 자바 객체의 최고 부모인 Object 타입으로 조회하면, 모든 스프링 빈을 조회한다.
- 1번 타입 조회하면 아래 모두 나옴
- 2번 조회하면 2번 아래만 나옴
예제 코드
package hello.core.beanfind;
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.AssertionsForClassTypes.assertThat;
import static org.junit.jupiter.api.Assertions.assertThrows;
public class ApplicationContextExtendsFindTest {
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(TestConfig.class);
@Test
@DisplayName("부모 타입으로 조회 시, 자식이 둘 이상 없으면 중복 오류가 발생한다")
void findBeanByParentTypeDuplicate(){
// DiscountPolicy bean = ac.getBean(DiscountPolicy.class);
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 bean = ac.getBean(RateDiscountPolicy.class);
assertThat(bean).isInstanceOf(RateDiscountPolicy.class);
}
@Test
@DisplayName("부모 타입으로 모두 조회하기")
void findAllBeanByparentType(){
// getBeansOfType으로 Bean모두 가져와서
Map<String, DiscountPolicy> beansOfType = ac.getBeansOfType(DiscountPolicy.class);
// Bean 갯수가 2개면 통과하게
assertThat(beansOfType.size()).isEqualTo(2);
// 실제 테스트 케이스 짤 땐 출력문 남기면안됌
for (String key : beansOfType.keySet()){
System.out.println("key = "+key+ " value = "+ beansOfType.get(key));
}
}
@Test
@DisplayName("부모 타입으로 모두 조회하기 - Object")
void findAllBeanByObjectType(){
// Object로 조회하면 스프링 내부 Bean까지 전부 튀어나옴
// 자바 객체는 모든게 Object 타입이기 때문에 스프링 객체에 등록된 모든 애들이 튀어나옴
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();
}
}
}
실제 개발할땐 ApplicationContext에서 Bean을 조회할 일이 없다.
그런데 굳이 써 본 이유는 기본 기능이기도 하고, 가끔 순수한 자바 애플리케이션에서 스프링 컨테이너를 생성할 때가 있다.
그럴때 ApplicationContext를 쓰는거다.
부모 타입으로 조회할때 자식이 어디까지 조회되는지 알고 있어야 자동 의존관계 주입에서 문제없이 잘 해결할수 있기때문에 살펴봤다.
반응형
'Spring > 스프링 핵심 원리 - 기본편' 카테고리의 다른 글
28. 다양한 설정 형식 지원 - 자바 코드, XML (0) | 2023.04.17 |
---|---|
27. BeanFactory와 ApplicationContext (0) | 2023.04.17 |
25. 스프링 빈 조회 - 동일한 타입이 둘 이상 (0) | 2023.04.02 |
24. 스프링 빈 조회 - 기본 (0) | 2023.04.02 |
23. 컨테이너에 등록된 모든 빈 조회 (0) | 2023.04.02 |