본문 바로가기

공부/Spring 핵심 원리

빈 생명주기 콜백을 알아보자

📌 빈 생명주기 콜백

 

데이터베이스 커넥션 풀이나, 네트워크 소켓처럼 애플리케이션 시작 시점에 미리 연결 하고, 종료 시점에 연결을 모두 종료하려면 객체의 초기화 종료 작업이 필요하다.

 

더보기
package com.example.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);
    }

    public void call(String message){
        System.out.println("call: "+url+ "message = " + message);
    }

    public void disconnect(){
        System.out.println("close : "+url);
    }
}
더보기
package com.example.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();
    }

    @Configuration
    static class LifeCycleConfig{

        @Bean
        public NetworkClient networkClient(){
            NetworkClient networkClient = new NetworkClient();
            networkClient.setUrl("http://hello-spring.dev");
            return networkClient;
        }
    }
}

 

 

생성자로 객체를 생성하면서 connect가 호출 된다. 아직 의존 관계 주입이 안된 상태라 메시지를 호출 하지 못한다.

초기화 작업은 빈 생성, 의존관계 주입이 끝난 다음에 실행되야한다.

 

👧 어떻게? 스프링에서 의존 관계 주입이 완료된 시점을 알면 된다.

    = 스프링 빈이 콜백 메서드를 통해 초기화 시점을 알려주는 다양한 기능을 제공함

 

** 초기화 콜백 : 빈이 생성되고, 빈의 의존관계 주입이 완료 된 후 호출된다.

** 빈이 소멸되기 직전에 호출

 

📌 스프링 빈의 이벤트 라이프 사이클

1. 스프링 컨테이너 생성

2. 스프링빈 생성

3. 의존 관계 주입

4. 초기화 콜백

5. 사용

6. 소멸 전 콜백 

7. 스프링 종료

 

 

👧 객체의 생성과 초기화를 분리하자

 : 생성자는 메모리를 할당해서 객체를 생성하는 책임을 갖고, 초기화는 생성된 값을 외부 커넥션을 연결하는 등의 동작을 수행하는게  유지 보수 관점에서 좋다.

 

 

📌 초기화 시점을 알려주는 다양한 방법 

 

👧 @PostConstruct, @PreDestory를 사용합시다.

 

인터페이스 InitializingBean, DisposableBean

더보기

 

package com.example.core.lifecycle;

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);

    }

    public void setUrl(String url){
        this.url=url;
    }

    public void connect(){
        System.out.println("connect = " + url);
    }

    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 {

        connect();
        call("초기화 연결 메시지");
    }

    @Override
    public void destroy() throws Exception {
        disconnect();
    }
}

- InitializingBean, DisposableBean을 상속 받으면 빈의 초기화 시점과, 소멸 시점을 알 수 있다.

- InitializingBean - afterPropertiesSet : 빈 생성 후 의존 주입 후 실행된다.

- DisposableBean - destory() : 소멸

- 단점 : 인터페이스가 스프링 전용 인터페이스다. 초기화, 소멸 메서드의 이름을 변경할 수 없고, 내가 코드를 고칠 수 없는 외부 라이브러리에는 적용할 수 없다.(거의 사용하지 않음)

 

빈 등록 초기화, 소멸 메서드

더보기

 

package com.example.core.lifecycle;



public class NetworkClient {

    private String url;

    public NetworkClient(){
        System.out.println("생성자 호출,url = " + url);

    }

    public void setUrl(String url){
        this.url=url;
    }

    public void connect(){
        System.out.println("connect = " + url);
    }

    public void call(String message){
        System.out.println("call: "+url+ "message = " + message);
    }

    public void disconnect(){
        System.out.println("close : "+url);
    }

    public void init(){
        System.out.println("NetworkClient.init");
        connect();
        call("초기화 연결 메시지");
    }

    public void close(){
        System.out.println("NetworkClient.close");
        disconnect();
    }

}

 

- 스프링 빈이나 스프링 코드에 의존적이지 않다.

- 메서드 이름을 자유롭게 줄 수 있다.

- 설정 정보를 사용 해 코드를 고칠 수 없는 외부 라이브러리에도 초기화, 종료 메서드를 적용할 수 있다.

- @Bean의 destoryMethod 속성에는 기본값으로 INFER_METHOD가 되있다.

- 이 기능은 close,shutdown 이라는 이름의 메서드를 호출해 종료 메서드를 추론해서 호출해 준다.

- 따라서 종료를 따로 해주지 않아도 종료된다.

 

 

 애노테이션 @PostConstruct, @PreDestory(권장 방법)

- 최신 스프링에서 권장하는 방법

- 매우 편리하다. javax.anootaion.PostConstruct는 스프링이 종속적이 아니라, 다른 컨테이너에서도 동작한다.

- 하나의 단점은 외부 라이브러리에는 적용 못한다. 외부 라이브러리는 @Bean의 기능을 사용하면 된다.

 

 

300x250