Spring

[스프링 핵심 원리] 07. 빈 생명주기 콜백

binning 2025. 1. 22. 17:06

1. 빈 생명주기 콜백

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

- 스프링은 의존관계 주입이 완료되면 스프링 빈에게 콜백 메서드를 통해서 초기화 시점을 알려주는 다양한 기능을 제공한다. 또한 스프링은 스프링 컨테이너가 종료되기 직전에 소멸 콜백을 준다.

 

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

: 스프링 컨테이너 생성 > 스프링 빈 생성 > 의존관계 주입 > 초기화 콜백 > 사용 > 소멸전 콜백 > 스프링 종료

 

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("disconnect: " + url);
    }
}

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

- 생성자는 필수 정보(파라미터)를 받고, 메모리를 할당해서 객체를 생성하는 책임을 가진다. 반면에 초기화는 이렇 게 생성된 값들을 활용해서 외부 커넥션을 연결하는등 무거운 동작을 수행한다.

- 따라서 생성자 안에서 무거운 초기화 작업을 함께 하는 것 보다는 객체를 생성하는 부분과 초기화 하는 부분을 명 확하게 나누는 것이 유지보수 관점에서 좋다.

- 물론 초기화 작업이 내부 값들만 약간 변경하는 정도로 단순한 경우 에는 생성자에서 한번에 다 처리하는게 더 나을 수 있다.

 

2. 인터페이스 InitializingBean, DisposableBean

public class NetworkClient implements InitializingBean, DisposableBean {

    @Override
    public void afterPropertiesSet() throws Exception {
        connect();
        call("초기화 연결 메시지");
    }
    
    @Override
    public void destroy() throws Exception {
        disconnect();
    }
}

- 이 인터페이스는 스프링 전용 인터페이스다. 해당 코드가 스프링 전용 인터페이스에 의존한다.

- 초기화, 소멸 메서드의 이름을 변경할 수 없다.

- 스프링 초창기에 나온 방법으로, 거의 사용하지 않는다.

 

3. 빈 등록 초기화, 소멸 메서드 지정

public class NetworkClient {

    public void init() {
        connect();
        call("초기화 연결 메시지");
    }
    
    public void close() {
        disconnect();
    }
}
@Configuration
static class LifeCycleConfig {

    @Bean(initMethod = "init", destroyMethod = "close")
    public NetworkClient networkClient() {
        NetworkClient networkClient = new NetworkClient();
        networkClient.setUrl("http://127.0.0.1:8080");
        return networkClient;
    }
}

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

- 스프링 빈이 스프링 코드에 의존하지 않는다.

- @Bean의 destroyMethod는 기본값이 (inferred)로 close, shutdown의 이름을 가진 메서드를 자동으로 호출한다.

 

4. 애노테이션 @PostConstruct, @PreDestroy

public class NetworkClient {

    @PostConstruct
    public void init() {
        connect();
        call("초기화 연결 메시지");
    }
    
    @PreDestroy
    public void close() {
        disconnect();
    }
}

- 최신 스프링에서 가장 권장하는 방법이다.

- 애노테이션 하나만 붙이면 되므로 매우 편리하다.

- 유일한 단점은 외부 라이브러리에는 적용하지 못한다는 것이다.

> 애노테이션을 사용하면서, 코드를 고칠 수 없는 외부 라이브러리를 초기화, 종료해야 하면 @Bean의 initMethod, destroyMethod를 사용하자.