데이터베이스 커넥션 풀이나, 네트워크 소켓처럼 애플리케이션 시작 시점에 필요한 연결을 미리 해두고, 애플리케이션 종료 시점에 연결을 모두 종료하는 작업을 진행하려면, 객체의 초기화와 종료 작업이 필요하다.
간단하게 외부 네트워크에 미리 연결하는 객체를 하나 생성한다고 가정해보자. 실제로 네트워크에 연결하는 것은 아니 고, 단순히 문자만 출력하도록 했다. 이 NetworkClient
는 애플리케이션 시작 시점에 connect()
를 호출해서 연 결을 맺어두어야 하고, 애플리케이션이 종료되면 disConnect()
를 호출해서 연결을 끊어야 한다.
@Test
public void lifeCycleTest(){
AnnotationConfigApplicationContext 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;
}
}
위 코드는 객체를 생성하고 세팅하는 과정이다.
스프링 빈의 이벤트 라이프사이클
스프링 컨테이너 생성 -> 스프링 빈 생성 -> 의존관계 주입 -> 초기화 콜백 -> 사용 -> 소멸전 콜백 -> 스프링 종료
(빈 생성과 의존관계 주입은 생성자를 사용하면 한번에 이뤄질 수 있다.)
위가 라이프 사이클인데 일단 아래 코드를 보자
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);
}
보면 생성자 호출을 하면서 커넥트와 call이라는 무거운 초기화 작업을 하게 된다.
아직 url 세팅을 하지 않았기 때문에 전부 NULL일 것이다.
참고: 객체의 생성과 초기화를 분리하자.
생성자는 필수 정보(파라미터)를 받고, 메모리를 할당해서 객체를 생성하는 책임을 가진다. 반면에 초기화는 이렇게 생성된 값들을 활용해서 외부 커넥션을 연결하는등 무거운 동작을 수행한다.
따라서 생성자 안에서 무거운 초기화 작업을 함께 하는 것 보다는 객체를 생성하는 부분과 초기화 하는 부분을 명 확하게 나누는 것이 유지보수 관점에서 좋다. 물론 초기화 작업이 내부 값들만 약간 변경하는 정도로 단순한 경우 에는 생성자에서 한번에 다 처리하는게 더 나을 수 있다.
자 그럼 이를 해결할 수 있는 방법들을 알아보자
스프링은 크게 3가지 방법으로 빈 생명주기 콜백을 지원한다.
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 {
System.out.println("NetworkClient.afterPropertiesSet");
connect();
call("초기화 연결 메시지");
}
@Override
public void destroy() throws Exception {
System.out.println("NetworkClient.destroy");
disconnect();
}
}
위처럼 생성과 초기화를 분리할 수 있다.
afterPropertiessSet을 이용해서 의존관계 주입이 끝나면 호출이 될 수 있도록 한다.
그럼 위와 같은 실행 결과를 얻을 수 있다.
객체의 생성과 초기화가 명확히 분리된 것을 알 수 있다.
하지만 인터페이스 방식은 너무 오래된 방식이라서 잘 사용되지 않는다.
초기화, 소멸 인터페이스 단점
@Bean(initMethod = "init", destroyMethod = "close")
public NetworkClient networkClient(){
NetworkClient networkClient = new NetworkClient();
networkClient.setUrl("http://hello-spring.dev");
return networkClient;
}
public void init() {
System.out.println("NetworkClient.afterPropertiesSet");
connect();
call("초기화 연결 메시지");
}
public void close() {
System.out.println("NetworkClient.destroy");
disconnect();
}
위처럼 빈 등록방식을 이용할 수도 있다.
설정 정보 사용 특징
종료 메서드 추론
@Bean의 destroyMethod
속성에는 아주 특별한 기능이 있다.close
, shutdown
이라는 이름의 종료 메서드를 사용한다.destroyMethod
는 기본값이 (inferred)
(추론)으로 등록되어 있다.close
, shutdown
라는 이름의 메서드를 자동으로 호출해준다. 이름 그대로 종료 메서드를 추 론해서 호출해준다.destroyMethod=""
처럼 빈 공백을 지정하면 된다. @PostConstruct
public void init() {
System.out.println("NetworkClient.afterPropertiesSet");
connect();
call("초기화 연결 메시지");
}
@PreDestroy
public void close() {
System.out.println("NetworkClient.destroy");
disconnect();
}
위와 같이 설정하면 끝이다.
@PostConstruct, @PreDestroy 애노테이션 특징
javax.annotation.PostConstruct
이다. 스프링에 종속적인 기술이 아니라 JSR-250 라는 자바 표준이다. 따라서 스프링이 아닌 다른 컨테이너에서도 동작한다.정리
@PostConstruct, @PreDestroy 애노테이션을 사용하자
@Bean
의 initMethod
, destroyMethod
를 사용하자.