스프링 빈 생명주기(Spring Bean LifeCycle)
스프링을 통해 이러한 초기화 작업과 종료 작업이 어떻게 진행되는지 예제로 확인
package hello.core.lifecycle;
public class NetworkClient {
private String url;
public NetworkClient() {
System.out.println("생성자 호출, url = " + url);
connect();
call("초기화 연결 메시지");
}
public void setUrl(String url){
this.url = url;
}
// 서비스 시작시 호출
public void connect(){
System.out.println("connect : " + url);
}
// 연결이 된 상태에서 call을 호퉁해서 연결한 서버에 메시지를 던질 수 있다고 가정
public void call(String message){
System.out.println("call: " + url + " message: " + message);
}
// 서비스 종료시 호출
public void disconnect(){
System.out.println("close : " + url);
}
}
package hello.core.lifecycle;
import org.junit.jupiter.api.Test;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
public class BeanLifeCycleTest {
@Test
public void lifeCycleTest() {
ConfigurableApplicationContext ac = new AnnotationConfigApplicationContext(LifeCycleConfig.class);
NetworkClient client = ac.getBean(NetworkClient.class);
ac.close(); //스프링 컨테이너를 졸료, ConfigurableApplicationContext 필요
}
@Configuration
static class LifeCycleConfig {
@Bean
public NetworkClient networkClient(){
NetworkClient networkClient = new NetworkClient();
networkClient.setUrl("http://hello-spring.dev");
return networkClient;
}
}
}
스프링 빈 라이프사이클
객체 생성 -> 의존관계 주입
("생성자 주입" 예외, 생성자 주입은 객체를 생성할 대 스프링 빈이 파라미터로 같이 들어와야 함)
스프링 빈의 이벤트 라이프사이클 (참고. 싱글톤에 대한 예시)
: 1)스프링 컨테이너 생성
2)스프링 빈 생성
3)의존관계 주입(setter 주입, 필드 주입)
4)초기화 콜백
5)사용
6)소멸 전 콜백
7)스프링 종료
참고
생성자는 필수 정보를 받아 객체를 생성하고 메모리를 할당함
초기화는 생성된 값을 활용해 외부 커넥션 등 무거운 작업을 수행함
따라서 생성자와 초기화 작업을 분리하는 게 유지보수에 좋음
단, 초기화가 간단한 경우엔 생성자에서 처리하는 게 나을 수 있음
싱글톤 빈들은 스프링 컨테이너가 종료될 때 싱글톤 빈들도 함께 종료 -> 스프링 컨테이너가 종료되기 직전에 소멸전 콜백이 발생
: 인터페이스로 초기화와 소멸전 콜백을 받는 방법
package hello.core.lifecycle;
import hello.core.discount.DiscountPolicy;
import hello.core.member.Member;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;
public class NetworkClient implements InitializingBean, DisposableBean {
private String url;
public NetworkClient() {
System.out.println("생성자 호출, url = " + url);
// connect();
// call("초기화 연결 메시지");
}
public void setUrl(String url){
this.url = url;
}
// 서비스 시작시 호출
public void connect(){
System.out.println("connect : " + url);
}
// 연결이 된 상태에서 call을 호퉁해서 연결한 서버에 메시지를 던질 수 있다고 가정
public void call(String message){
System.out.println("call: " + url + " message: " + message);
}
// 서비스 종료시 호출
public void disconnect(){
System.out.println("close : " + url);
}
@Override
public void afterPropertiesSet() throws Exception {
System.out.println("NetworkClient.afterPropertiesSet");
connect();
call("초기화 연결 메시지");
}
@Override
public void destroy() throws Exception {
System.out.println("NetworkClient.destroy");
disconnect();
}
}
InitializingBean은 afterPropertiesSet() 메서드로 초기화를 지원
DisposableBean은 detroy() 메서드로 소멸을 지원
BeanLifeCycleTest 클래스의 lifeCycleTest 테스트
초기화, 소멸 인터페이스 단점
설정 정보에 @Bean(initMethod = "init", destroyMethod = "close") 처럼 초기화, 소멸 메서드를 지정
package hello.core.lifecycle;
import hello.core.discount.DiscountPolicy;
import hello.core.member.Member;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;
public class NetworkClient {
private String url;
public NetworkClient() {
System.out.println("생성자 호출, url = " + url);
// connect();
// call("초기화 연결 메시지");
}
public void setUrl(String url){
this.url = url;
}
// 서비스 시작시 호출
public void connect(){
System.out.println("connect : " + url);
}
// 연결이 된 상태에서 call을 호퉁해서 연결한 서버에 메시지를 던질 수 있다고 가정
public void call(String message){
System.out.println("call: " + url + " message: " + message);
}
// 서비스 종료시 호출
public void disconnect(){
System.out.println("close : " + url);
}
public void init() throws Exception{
System.out.println("NetworkClient init");
connect();
call("초기화 연결 메시지");
}
public void close() throws Exception{
System.out.println("NetworkClient close");
disconnect();
}
}
package hello.core.lifecycle;
import hello.core.discount.DiscountPolicy;
import hello.core.member.Member;
import jakarta.annotation.PostConstruct;
import jakarta.annotation.PreDestroy;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;
public class NetworkClient {
private String url;
public NetworkClient() {
System.out.println("생성자 호출, url = " + url);
// connect();
// call("초기화 연결 메시지");
}
public void setUrl(String url){
this.url = url;
}
// 서비스 시작시 호출
public void connect(){
System.out.println("connect : " + url);
}
// 연결이 된 상태에서 call을 호퉁해서 연결한 서버에 메시지를 던질 수 있다고 가정
public void call(String message){
System.out.println("call: " + url + " message: " + message);
}
// 서비스 종료시 호출
public void disconnect(){
System.out.println("close : " + url);
}
@PostConstruct
public void init() throws Exception{
System.out.println("NetworkClient init");
connect();
call("초기화 연결 메시지");
}
@PreDestroy
public void close() throws Exception{
System.out.println("NetworkClient close");
disconnect();
}
}
BeanLifeCycleTest 클래스에서 이전에 작성했던 초기화(initMethod), 소멸메서드(destroyMethod) 속성을 제거한 후 실행 테스트
@PostConstruct와 @PreDestroy 애노테이션은 최신 스프링에서 권장하는 초기화 및 종료 방법임.
요약하자면, @PostConstruct와 @PreDestroy는 스프링에서 초기화와 종료를 간편하게 처리할 수 있는 표준 애노테이션이지만, 외부 라이브러리에는 적용할 수 없다는 점이 단점임