====== Customizing the nature of a bean ======
===== 개요 =====
컨테이너의 빈 라이프사이클 관리와 상호 작용하기 위해 Spring InitializingBean 및 DisposableBean 인터페이스를 구현할 수 있는데, 컨테이너는 전자의 경우 afterPropertiesSet()을 호출하고 후자의 경우 destroy()를 호출하여 빈이 초기화 및 소멸될 때 특정 작업을 수행하도록 한다.
===== 설명 =====
==== Lifecycle callbacks ====
Spring Framework는 Container 내부의 bean의 행동을 변화시길 수 있는 다양한 callback interface를 제공한다.
=== 객체화 callbacks(Initialization callbacks) ===
''org.springframework.beans.factory.InitializingBean'' interface를 구현하면 bean에 필요한 모든 property를 설정한 후, 초기화 작업을 수행한다. ''InitializingBean'' interface는 다음 메소드를 명시하고 있다.
void afterPropertiesSet() throws Exception;
일반적으로, ''InitializingBean'' interface의 사용을 권장하지 않는다. 왜냐하면 code가 불필요하게 Spring과 결합되기(couple) 때문이다. 대안으로, bean 정의는 초기화 메소드를
지정할 수 있다. XML 기반 설정의 경우, '''init-method''' attribute를 사용한다.
public class ExampleBean {
public void init() {
// do some initialization work
}
}
위 예제는 아래 예제와 같다.
public class AnotherExampleBean implements InitializingBean {
@Override
public void afterPropertiesSet() {
// do some initialization work
}
}
=== 파괴 callbacks(Destruction callbacks) ===
''org.springframework.beans.factory.DisposableBean'' interface를 구현하면, Container가 파괴될 때 bean이 callback를 받을 수 있다. ''DisposableBean'' interface는 다음 메소드를 명시하고 있다.
void destroy() throws Exception;
일반적으로, ''DisposableBean'' interface의 사용을 권장하지 않는다. 왜냐하면 code가 불필요하게 Spring과 결합되기(couple) 때문이다. 대안으로, bean 정의는 초기화 메소드를
지정할 수 있다. XML 기반 설정의 경우, '''destroy-method''' attribute를 사용한다.
public class ExampleBean {
public void cleanup() {
// do some destruction work (like releasing pooled connections)
}
}
위 예제는 아래 예제와 같다.
public class AnotherExampleBean implements DisposableBean {
@Override
public void destroy() {
// do some destruction work (like releasing pooled connections)
}
}
=== 기본 객체화 및 파괴 메소드(Default initialization & destroy methods) ===
Spring Container는 모든 bean에 대해서 같은 이름의 초기화 및 파괴 메소드를 지정할 수 있다.
public class DefaultBlogService implements BlogService {
private BlogDao blogDao;
public void setBlogDao(BlogDao blogDao) {
this.blogDao = blogDao;
}
// this is (unsurprisingly) the initialization callback method
public void init() {
if (this.blogDao == null) {
throw new IllegalStateException("The [blogDao] property must be set.");
}
}
}
''%%%%'' element의 '''default-init-method''' attribute를 이용하여 기본 객체화 callback 메소드를 지정할 수 있다. 파괴 callback 메소드의 경우 '''default-destroy-method''' attribute를 이용하여 지정할 수 있다.
''%%%%'' element에 '''init-method''', '''destroy-method''' attribute가 정의되어 있는 경우, 기본값은 무시된다.
=== 생명주기 메커니즘 병합 ===
Spring Framework에서는 3가지 방식의 생명주기 메커니즘이 존재한다: ''InitialzingBean''과 ''DisposableBean'' interface; 맞춤 ''init()''과 ''destroy()'' 메소드; 그리고 [[eGovframework:RTE:FDL:IoC Container:Annotation-based configuration:#@PostConstruct and @PreDestroy annotations]]
만약 서로 다른 생명주기 메커니즘을 같이 사용할 경우, 개발자는 적용되는 순서를 알고 있어야 한다. 객체화 메소드의 순서는 다음과 같다.
- @PostConstruct annotation이 있는 메소드
- ''InitializingBean'' callback interface에 정의된 ''afterPropertiesSet()''
- 맞춤 ''init()'' 메소드
파괴 메소드의 호출 순서는 다음과 같다.
- @PreDestroy annotation이 있는 메소드
- ''DisposableBean'' callback interface에 정의된 ''destroy()''
- 맞춤 ''destroy()'' 메소드
=== 시작 및 종료 callbacks(Startup and Shutdown Callbacks) ===
Lifecycle 인터페이스는 자체 수명 주기 요구 사항(예: 일부 백그라운드 프로세스 시작 및 중지)이 있는 모든 객체에 대해 필수 메서드를 정의할 수 있다.
public interface Lifecycle {
void start();
void stop();
boolean isRunning();
}
모든 Spring 관리 객체는 Lifecycle 인터페이스를 구현할 수 있다. 그런 다음 애플리케이션 컨텍스트 자체가 시작 및 중지 신호를 수신하면(예: 런타임에 중지/재시작 시나리오의 경우) 해당 컨텍스트 내에 정의된 모든 Lifecycle 구현으로 해당 호출을 캐스케이드한다.
public interface LifecycleProcessor extends Lifecycle {
void onRefresh();
void onClose();
}
===== 참고자료 =====
* [[https://docs.spring.io/spring-framework/docs/5.3.27/reference/html/core.html#beans-factory-nature|Spring Framework - Reference Document / 1.6 Customizing the nature of a bean]]