🔹 스프링 컨테이너 생성
🔹 ApplicationContext: 스프링 컨테이너이고 인터페이스
🔹 스프링 컨테이너는 XML 또는 어노테이션 기반의 자바 설정 클래스(AppConfig)로 만들 수 있다.
🔹 스프링 컨테이너 생성랑 때 구성정보(AppConfig)를 지정해줘야 한다.
🔸 스프링 컨테이너 applicationContext
ApplicationContext applicationContext = new AnnotationConfigApplicationContext(AppConfig.class);
🔸 AppConfig.java
@Configuration
public class AppConfig {//애플리케이션에 대한 환경 설정을 다 해주는 역할, ex) 공연 기획자
@Bean
public MemberService memberService() {
System.out.println("AppConfig.memberService");
return new MemberServiceImpl(memberRepository());//생성자주입: 생성자를 통해 객체 주입
}
@Bean
public MemoryMemberRepository memberRepository() {
System.out.println("AppConfig.MemberRepository");
return new MemoryMemberRepository();
}
@Bean
public OrderService orderService() {
System.out.println("AppConfig.orderService");
return new OrderServiceImpl(memberRepository(), discountPolicy());
}
@Bean
public DiscountPolicy discountPolicy() {
return new RateDiscountPolicy();
}
}
🔹 스프링 컨테이너는 파라미터로 넘어온 설정 클래스 정보를 사용해서 스프링 빈을 등록한다.
🔹 스프링 컨테이너는 설정 정보를 참고하여 의존관계를 주입(DI)한다.
🔹 ROLE_APPLICATION: 일반적으로 사용자가 정의한 빈
🔹 ROLE_INFRASTRUCTURE: 스프링이 내부에서 사용하는 빈
public class ApplicationContextInfoTest {
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(AppConfig.class);
@Test
@DisplayName("모든 bean 출력하기")
void findAllBean() {
String[] beanDefinitionNames = ac.getBeanDefinitionNames();
for (String beanDefinitionName : beanDefinitionNames) {
Object bean = ac.getBean(beanDefinitionName);
System.out.println("name=" + beanDefinitionName + " object=" + bean);
}
}
@Test
@DisplayName("애플리케이션 bean 출력하기")
void findApplicationBean() {
String[] beanDefinitionNames = ac.getBeanDefinitionNames();
for (String beanDefinitionName : beanDefinitionNames) {
BeanDefinition beanDefinition = ac.getBeanDefinition(beanDefinitionName);//bean에 대한 메타데이터 정보들
//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.getBean(빈이름, 타입)
🔹 ac.getBean(타입)
🔹 조회 대상 빈이 없을 때 예외 발생: "NoSuchBeanDefinitionException: No bean named 'xxxxx' available"
class ApplicationContextBasicFindTest{
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContxt(AppConfig.class);
@Test
@DisplayName("빈 이름으로 조회 안될 때")
void findBeanByNameX(){
Assertion.assertThrows(NoSuchBeanDefinitionException.class,
()->ac.getBean("xxxx",MemberService.class);
}
}
🔹 동일한 타입이 둘 이상일 때 예외 발생: "NoUniqueBeanDefinitionException"
class ApplicationContextSameBeanFindTest {
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(SameBeanConfig.class);
@Test
@DisplayName("타입으로 조회시 같은 타입이 둘 이상 있으면, 중복 오류가 발생한다")
void findBeanByTypeDuplicate() {
//DiscountPolicy bean = ac.getBean(MemberRepository.class);
assertThrows(NoUniqueBeanDefinitionException.class,
() -> ac.getBean(MemberRepository.class));
}
}
🔹 부모 타입으로 조회시 자식 타입도 함께 조회 가능
🔹 모든 자바 객체 최고 부모인 Object타입으로 조회하면 모든 스프링 빈을 조회한다.
🔹 스프링 컨테이너의 최상위 인터페이스
🔹 getBean() 제공
🔹 BeanFactory 기능을 모두 상속받아 제공
🔹 부가기능 제공: 국제화 기능, 환경변수, 애플리케이션 이벤트, 리소스 조회
🔹 BeanFactory, ApplicationContext를 스프링 컨테이너라 한다.
🔹 new AnnotationConfigApplicationContext(AppConfig.class)
🔹 AnnotationConfigApplicationContext 클래스를 사용하면서 설정 정보를 넘기는 방식
🔹 GenericXmlApplicationContext를 사용하면서 xml 설정 파일을 넘기는 방식
🔸 XmlAppContext.java
public class XmlAppContext{
@Test
void xmlAppContext(){
ApplicationContext ac = new GenericXmlApplicationContext("appConfig.xml");
MemberService memberService = ac.getBean("memberService",MemberService.class);
assertThat(memberService).isInstanceOf(MemberService.class);
}
}
🔸 appConfig.xml 설정 정보
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://
www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="memberService" class="hello.core.member.MemberServiceImpl">
<constructor-arg name="memberRepository" ref="memberRepository" />
</bean>
<bean id="memberRepository"
class="hello.core.member.MemoryMemberRepository" />
<bean id="orderService" class="hello.core.order.OrderServiceImpl">
<constructor-arg name="memberRepository" ref="memberRepository" />
<constructor-arg name="discountPolicy" ref="discountPolicy" />
</bean>
<bean id="discountPolicy" class="hello.core.discount.RateDiscountPolicy" />
</beans>
🔹 BeanDefinition: 설정 메타 정보, 역할과 구현을 개념적으로 나누었다.
🔹 @Bean, <bean> 당 각각 하나씩 메타 정보가 생성된다.
🔹 스프링 컨테이너가 이 메타정보를 기반으로 빈을 생성한다.
🔹 AnnotationConfigApplicationContext는 AnnotationBeanDefinitionReader를 사용해서 AppConfig.class를 읽고 BeanDefinition을 생성한다.