728x90
반응형
본 포스팅은 학습용으로, 김영한 님의 강의를 바탕으로 복습용도로 작성되었습니다.
선행 게시물
[Spring] 스프링 컨테이너 생성
본 포스팅은 학습용으로, 김영한 님의 강의를 바탕으로 복습용도로 작성되었습니다. 스프링 컨테이너 생성 스프링 컨테이너 생성 과정 ApplicationContext applicationContext = new AnnotationConfigApplicationConte
muscleking3426.tistory.com
컨테이너에 등록된 모든 빈 조회
- Test 를 활용
- ApplicationContextInfoTest
import hello.core.AppConfig;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class ApplicationContextInfoTest {
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(AppConfig.class);
@Test
@DisplayName("모든 빈 출력하기")
void findAllBean() {
String[] beanDefinitionNames = ac.getBeanDefinitionNames();
for(String beanDefinitionName : beanDefinitionNames){
Object bean = ac.getBean(beanDefinitionName);
System.out.println("name = " + beanDefinitionName + "Object = " + bean);
}
}
@Test
@DisplayName("애플리케이션 빈만 출력하기")
void findApplicationBean() {
String[] beanDefinitionNames = ac.getBeanDefinitionNames();
for(String beanDefinitionName : beanDefinitionNames){
BeanDefinition beanDefinition = ac.getBeanDefinition(beanDefinitionName);
//Role ROLE_APPLICATION: 직접 등록한 애플리케이션 빈
//Role ROLE_INFRASTRUCTURE: 스프링이 내부에서 사용하는 빈
if(beanDefinition.getRole() == BeanDefinition.ROLE_APPLICATION){
//내가 앱을 개발하기 위해서 등록한 빈 객체만 출력하기
Object bean = ac.getBean(beanDefinitionName);
System.out.println("name = " + beanDefinitionName + " Object = " + bean);
}
}
}
}
- 모든 빈 출력
- 스프링에 등록된 모든 빈 정보 출력
- ac.getBeanDefinitionNames( )
- 스프링에 등록된 모든 빈 이름 조회
- ac.getBean( )
- 빈 이름으로 빈 객체(인스턴스) 조회
- 애플리케이션 빈 출력
- 스프링 내부에서 사용하는 빈 제외
- 개발자(사용자)가 직접 등록한 빈만 출력한다.
- 스프링이 내부에서 사용하는 빈은 getRole( )로 구분
- ROLE_APPLICATION : 일반적으로 사용자가 정의한 빈
- ROLE_INFRASTRUCTURE : 스프링 내부에서 사용하는 빈
기본적인 스프링 빈 조회
- 스프링 컨테이너에서 스프링 빈을 찾는 가장 기본적인 방법
- ac.getBean(빈 이름, 타입)
- ac.getBean(타입)
- 조회 대상 스프링 빈이 없으면 아래와 같이 예외가 발생한다.
- NoSuchBeanDefinitionException
- 스프링 빈 조회 테스트
- ApplicationContextBasicFindTest
import hello.core.AppConfig;
import hello.core.member.MemberService;
import hello.core.member.MemberServiceImpl;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import static org.assertj.core.api.Assertions.*;
import static org.junit.jupiter.api.Assertions.*;
public class ApplicationContextBasicFindTest {
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(AppConfig.class);
@Test
@DisplayName("빈 이름으로 조회")
void findBeanByName() {
MemberService memberService = ac.getBean("memberService", MemberService.class);
assertThat(memberService).isInstanceOf(MemberServiceImpl.class);
}
@Test
@DisplayName("이름 없이 타입으로 만 조회")
void findBeanByType() {
MemberService memberService = ac.getBean(MemberService.class);
assertThat(memberService).isInstanceOf(MemberServiceImpl.class);
}
@Test
@DisplayName("구체 타입으로 조회")
void findBeanByName2() {
MemberService memberService = ac.getBean("memberService", MemberServiceImpl.class);
assertThat(memberService).isInstanceOf(MemberServiceImpl.class);
}
@Test
@DisplayName("빈 이름으로 조회 없음")
void findBeanByNameX(){
//ac.getBean("xxxx", MemberService.class);
//MemberService xxxx = ac.getBean("xxxx", MemberService.class);
//NoSuchBeanDefinitionException: No bean named 'xxxx' available
assertThrows(NoSuchBeanDefinitionException.class,
() -> ac.getBean("xxxx", MemberService.class) );
// -> 오른쪽에 있는 로직을 실행하면 , 왼쪽에있는 NoSuchBeanDefinitionException 이 발생해야한다.
}
}
- 구체타입으로 조회하면, 변경 시 유연성이 떨어진다.
- 객체지향 5원칙 중 DIP 위배
객체지향 5원칙
[Spring] 객체지향 프로그래밍과 5원칙
본 포스팅은 학습용으로, 김영한님의 강의를 바탕으로 복습용도로 작성되었습니다. ✏️ 객체 지향 특징 추상화 캡슐화 상속 다형성 🧰 다형성을 실세계에 비유하면? 역할과 구현으로 세상을
muscleking3426.tistory.com
동일한 타입이 두개 이상인 경우
- 타입으로 조회 시 같은 타입의 스프링 빈이 둘 이상이면 오류 발생
- 빈 이름을 지정
- ac.getBeansOfType( )
- 해당 타입의 모든 빈을 조회
- ApplicationContextSameBeanFindTest
import hello.core.member.MemberRepository;
import hello.core.member.MemoryMemberRepository;
import org.junit.jupiter.api.Assertions;
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.*;
public class ApplicationContextSameBeanFineTest {
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(SameBeanConfig.class);
@Test
@DisplayName("타입으로 조회시 같은 타입이 둘 이상 있으면, 중복 오류가 발생한다")
void findBeanTypeDuplication() {
//MemberRepository bean = ac.getBean(MemberRepository.class); // -> NoUniqueBeanDefinitionException memberRepository1,memberRepository2
Assertions.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 + " values = " + 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();
}
}
}
스프링 빈 상속관계
- 부모 타입으로 조회하면, 자식 타입도 함께 조회한다.
- 자바의 최상위 객체인 Object 타입으로 조회하면, 모든 스프링 빈을 조회한다.
- ApplicationContextExtendsFindTest
import hello.core.discount.DiscountPolicy;
import hello.core.discount.FixDiscountPolicy;
import hello.core.discount.RateDiscountPolicy;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.NoUniqueBeanDefinitionException;
import org.springframework.context.ApplicationContext;
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.*;
public class ApplicationContextExtendsFindTest {
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(TestConfig.class);
@Test
@DisplayName("부모 타입으로 조회시 자식이 둘 이상 있으면, 중복 오류가 발생한다.")
void findBeanByParentTypeDuplicate() {
Assertions.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", 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();
}
}
}
728x90
반응형
'Spring' 카테고리의 다른 글
[Spring] 싱글톤과 @Configuration (0) | 2023.02.08 |
---|---|
[Spring] 싱글톤 컨테이너 (0) | 2023.02.08 |
[Spring] 스프링 컨테이너 생성 (0) | 2023.02.08 |
[Spring] Spring의 핵심 및 목적 (0) | 2023.02.07 |
[Spring] 객체지향 프로그래밍과 5원칙 (0) | 2023.02.07 |