이 글은 강의 : 김영한님의 - "스프링 핵심원리 - 기본편"을 듣고 정리한 내용입니다. 😁😁
이번에는 스프링 빈 조회에서 상속 관계에 대해서 배워볼 것이다.
부모 타입도 조회하면 자식 타입도 함께 조회된다!!! 🤗🤗
모든 자바 객체의 최상위 클래스인 Object 타입으로 조회하면 모든 스프링 빈을 조회한다.
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.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);
// rateDiscountPolicy가 RateDiscountPolicy의 인스턴스인지 검증
assertThat(rateDiscountPolicy).isInstanceOf(RateDiscountPolicy.class);
}
// 안좋은 방법
@Test
@DisplayName("특정 하위 타입으로 조회")
void findBeanBySubType() {
RateDiscountPolicy bean = ac.getBean(RateDiscountPolicy.class);
// bean이 RateDiscountPolicy의 인스턴스인지 검증
assertThat(bean).isInstanceOf(RateDiscountPolicy.class);
}
@Test
@DisplayName("부모 타입으로 모두 조회")
void findAllBeanByParentType() {
// DiscountPolicy 타입인 모든 빈을 Map으로 반환
Map<String, DiscountPolicy> beansOfType = ac.getBeansOfType(DiscountPolicy.class);
// DiscountPolicy 타입인 빈의 개수가 맞는지 검증
assertThat(beansOfType.size()).isEqualTo(2);
// DiscountPolicy 타입인 빈의 정보 출력(빈 이름, 빈 객체)
for (String key : beansOfType.keySet()) {
System.out.println("key = " + key + " value = " + beansOfType.get(key));
}
}
// 스프링 컨테이너에 등록된 모든 빈 조회(내부 빈 & 사용자 등록 빈)
@Test
@DisplayName("부모 타입으로 모두 조회하기 - Object")
void findAllBeanByObjectType() {
// Object 타입인 모든 빈을 Map으로 반환
Map<String, Object> beansOfType = ac.getBeansOfType(Object.class);
// Object 타입인 빈의 정보 출력(빈 이름, 빈 객체)
for (String key : beansOfType.keySet()) {
System.out.println("key = " + key + " value = " + beansOfType.get(key));
}
}
@Configuration
static class TestConfig {
// 부모 타입이 DiscountPolicy인 2개의 빈
@Bean
public DiscountPolicy rateDiscountPolicy() {
return new RateDiscountPolicy();
}
@Bean
public DiscountPolicy fixDiscountPolicy() {
return new FixDiscountPolicy();
}
}
}
이 테스트 클래스 내에서만 사용하기 위해 만든 Config 클래스이기 때문에 static으로 선언.
@Configuration
static class TestConfig {
// 부모 타입이 DiscountPolicy인 2개의 빈
@Bean
public DiscountPolicy rateDiscountPolicy() {
return new RateDiscountPolicy();
}
@Bean
public DiscountPolicy fixDiscountPolicy() {
return new FixDiscountPolicy();
}
}
getBean(부모타입)으로 조회시, 자식이 둘 이상이면 중복 오류
@Test
@DisplayName("부모 타입으로 조회시, 자식이 둘 이상 있으면, 중복 오류가 발생한다")
void findBeanByParentTypeDuplicate() {
// 오른쪽 로직을 수행했을 때 왼쪽 예외가 터져야 테스트 성공
assertThrows(NoUniqueBeanDefinitionException.class,
() -> ac.getBean(DiscountPolicy.class));
}
자식이 2개 이상이어도 getBean(빈 이름, 부모 타입)으로 조회하면 성공
@Test
@DisplayName("부모 타입으로 조회시, 자식이 둘 이상 있으면, 빈 이름을 지정하면 된다")
void findBeanByParentTypeBeanName() {
DiscountPolicy rateDiscountPolicy = ac.getBean("rateDiscountPolicy", DiscountPolicy.class);
// rateDiscountPolicy가 RateDiscountPolicy의 인스턴스인지 검증
assertThat(rateDiscountPolicy).isInstanceOf(RateDiscountPolicy.class);
}
특정 하위 타입으로 조회하는 것은 좋지 않은 방식!! - 이렇게 하지 말자.
// 안좋은 방법
@Test
@DisplayName("특정 하위 타입으로 조회")
void findBeanBySubType() {
RateDiscountPolicy bean = ac.getBean(RateDiscountPolicy.class);
// bean이 RateDiscountPolicy의 인스턴스인지 검증
assertThat(bean).isInstanceOf(RateDiscountPolicy.class);
}
getBeansOfType(부모타입)으로 부모 타입 + 하위 타입 빈 조회
@Test
@DisplayName("부모 타입으로 모두 조회")
void findAllBeanByParentType() {
// DiscountPolicy 타입인 모든 빈을 Map으로 반환
Map<String, DiscountPolicy> beansOfType = ac.getBeansOfType(DiscountPolicy.class);
// DiscountPolicy 타입인 빈의 개수가 맞는지 검증
assertThat(beansOfType.size()).isEqualTo(2);
// DiscountPolicy 타입인 빈의 정보 출력(빈 이름, 빈 객체)
for (String key : beansOfType.keySet()) {
System.out.println("key = " + key + " value = " + beansOfType.get(key));
}
}
getBeansOfType(Object)로 모든 빈 조회한다.
// 스프링 컨테이너에 등록된 모든 빈 조회(내부 빈 & 사용자 등록 빈)
@Test
@DisplayName("부모 타입으로 모두 조회하기 - Object")
void findAllBeanByObjectType() {
// Object 타입인 모든 빈을 Map으로 반환
Map<String, Object> beansOfType = ac.getBeansOfType(Object.class);
// Object 타입인 빈의 정보 출력(빈 이름, 빈 객체)
for (String key : beansOfType.keySet()) {
System.out.println("key = " + key + " value = " + beansOfType.get(key));
}
}
정리하자면,
1. 스프링 빈 이름 조회
getBeanDefinitionNames() : ApplicationContext(스프링 컨테이너)에 등록된 모든 스프링 빈 이름을 조회한다.
2. 스프링 빈 객체 조회
getBean(타입) : ApplicationContext(스프링 컨테이너)에 등록된 해당 타입의 스프링 빈 객체 조회
getBean(빈 이름, 타입) : 스프링 컨테이너에 등록된 빈 이름, 타입의 스프링 빈 객체 조회
getBeansOfType(타입) : 스프링 컨테이너에 등록된 해당 타입의 모든 스프링 빈 객체 조회
이번 챕터를 통해 스프링 빈 조회 기능에 대해 공부한 이유에 대해서 생각해보자.
실제로 개발할 때는 ApplicationContext에서 스프링 빈을 조회할 일이 거의 없다.
하지만 스프링 빈 조회는 기본 기능!!, 가끔 순수 자바 애플리케이션에서 스프링 컨테이너를 생성해서 써야하는 경우, 스프링 빈 조회를 사용한다.
부모 타입으로 조회할 때 어디까지 조회되는지에 대한 이해가 있어야 자동 의존 관계 주입에 대해 문제없이 잘 해결할 수 있다.