Notice
Recent Posts
Recent Comments
Link
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | |||||
3 | 4 | 5 | 6 | 7 | 8 | 9 |
10 | 11 | 12 | 13 | 14 | 15 | 16 |
17 | 18 | 19 | 20 | 21 | 22 | 23 |
24 | 25 | 26 | 27 | 28 | 29 | 30 |
Tags
- 문자열
- 리눅스마스터 1급 정리
- Kotlin
- 카카오
- JavaScript
- 리눅스마스터1급
- 개발자 회고록
- Linux
- 월간코드챌린지
- Java
- 연습문제
- 코딩테스트
- 리눅스
- 백준 javascript
- 자바스크립트 코딩의 기술
- GoingBus
- 반복문
- 리눅스마스터 3과목
- 고잉버스
- map
- 스프링 빈
- 프로그래머스
- Memoir
- toCharArray
- 스프링 컨테이너
- 백준 java
- java 백준 1차원 배열
- 명령어
- 코테
- 자바
Archives
- Today
- Total
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
반응형