<Spring> 스프링 핵심원리 이해 8 - 스프링 컨테이너와 빈 생명주기 콜백

스프링 컨테이너의 생명주기

  • 생성->빈 설정->사용->소멸순으로 구성된다.
  • 아래와 같은 방식으로 스프링 컨테이너에 대한 라이프사이클 메서드를 사용할 수 있다.
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(); 
	ac.register();//빈 추가로 등록
	ac.referesh();//등록하기 위한 갱신
	ac.close();//소멸 메서드, 이때 빈도 같이 소멸

스프링 빈의 생명주기

  • 스프링 빈 생성->의존관계 주입->초기화->소멸의 단계를 가진다.
  • 해당 라이플사이클에서 초기화시 사용되는 콜백과 소멸 단계에서 사용되는 콜백이 존재하는데 아래 예제와 코드를 통해 알아 볼 것이다.

빈 생명주기 콜백 도입 배경

  • 먼저 콜백에 대해 설명하면, 주로 콜백함수를 부를 때 사용되는 용어이며 콜백함수를 등록하면 특정 이벤트가 발생했을때 해당 메서드가 호출된다.즉 조건에 따라 실행될 수도 실행되지 않을 수도 있는 개념이라 보면된다.
  • 데이터베이스 커넥션 풀이나, 네트워크 소켓처럼 애플리케이션 시작 시점에서 필요한 연결을 미리하고 종료시점에 모든 연결을 종료하는 작업을 진행하려면 객체의 초기화와 종료 작업이 필요하다.
  • 아래 코드 예제는 간단하게 외부 네트워크에 객체 하나를 생성한다.
  • test에 lifecycle 패키지내에 다음 클래스들을 생성한다.

<NetworkClient 클래스>

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 msg) {
        System.out.println("call : " + url + "message = " + msg);
    }

    //서비스 종료시 호출
    public void disconnect() {
        System.out.println("close = " + url);
    }
}

<BeanLifeCycleTest 클래스>

public class BeanLifeCycleTest {
    @Test
    public void lifeCycleTest(){
        ConfigurableApplicationContext ac = new AnnotationConfigApplicationContext(LifeCycleConfig.class);
        NetworkClient networkClient = ac.getBean(NetworkClient.class);
        ac.close();
    }

    @Configuration
    static class LifeCycleConfig{
        @Bean
        public NetworkClient networkClient() {
            NetworkClient networkClient = new NetworkClient();
            networkClient.setUrl("https://hong.com");
            return networkClient;
        }
    }
}
  • 위의 코드의 결과는 다음과 같다.
  • 코드로만 보면 setUrl 메서드를 사용해서 객체 초기화가 된것 같지만 아래의 결과를 보면 그렇지 않다.
  • 해당 결과는 생성자 시점에는 수정자를 통해 url 정보가 들어오지 않았기에 null값이 출력되기 때문에 나타난다.
  • 객체를 생성한 다음, 외부에서 수정자 주입을 통해 setUrl()이 호출되어야 한다.
  • 이제부터 위의 언급된 빈 라이프 사이클과 콜백을 이용해서 우리가 기대하는 출력값을 출력해보자.

빈 생명주기 콜백의 종류

인터페이스(InitializingBean, DisposableBean)사용

  • 아래와 같이 해당 인터페이스들을 implements한다.
  • 기대하던바와 같이 Url이 설정한 값이 출력되는것을 확인 할 수 있다.
  • 해당 인터페이스는 스프링 전용 인터페이스라서 외부 라이브러리에서는 사용이 불가하다.
  • 또한 초기화,소멸 메서드의 이름을 변경 할 수 없다.
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 msg) {
        System.out.println("call : " + url + "message = " + msg);
    }
    //서비스 종료시 호출
    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();
    }
}

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

  • 아래와 같이 설정 정보에서 스프링 빈을 등록할때, 초기화 소멸 콜백 함수를 지정하는 방법이다.
  • 사용할 콜백 함수의 이름을 자유롭게 지정할수 있으며, 스프링 빈이 스프링 코드에 의존하지 않는다.또한 스프링 설정 정보를 사용하는 방법이기에 외부 라이브러리에서도 사용 가능하다.

<설정정보에 초기화 소멸 콜백 함수 지정>

	@Configuration
    static class LifeCycleConfig{
        @Bean(initMethod = "init", destroyMethod="close")
        public NetworkClient networkClient() {
            NetworkClient networkClient = new NetworkClient();
            networkClient.setUrl("https://hong.com");
            return networkClient;
        }
    }

<설정정보를 사용하도록 변경>

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 msg) {
        System.out.println("call : " + url + "message = " + msg);
    }
    //서비스 종료시 호출
    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();
    }
}

@PostConstruct, @PreDestroy 애노테이션 사용하기

  • 최신 스프링에서 가장 권장하는 방법이다.
  • 애노테이션 하나만 붙이면 모든 설정이 끝나기에 굉장히 편리하다.

<설정정보 변경>

	@Configuration
    static class LifeCycleConfig{
        @Bean
        public NetworkClient networkClient() {
            NetworkClient networkClient = new NetworkClient();
            networkClient.setUrl("https://hong.com");
            return networkClient;
        }
    }

<NetworkClient 클래스 변경>

public class NetworkClient{
    private String 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();
    }
}

소스코드 : https://github.com/brido4125/core-spring

해당 글은 김영한님의 <스프링 핵심> 강좌를 기반으로 작성되었습니다.