1. 빈 생명주기 콜백 시작
스프링 빈은 간단하게 다음과 같은 라이프사이클을 가진다.
- 객체 생성 -> 의존관계 주입
- 생성자 주입은 예외다.
- 객체 생성에 있어서, 의존관계가 주입된 후 생성되는 것이 자바의 룰이기 때문.
- 수정자 주입인 경우, 객체 생성 단계에서는 아직 의존관계가 주입되어있지 않다.
- 생성자 주입은 예외다.
- 의존관계 주입이 모두 끝난 다음에 초기화 작업을 진행해야 한다.
- 의존관계 주입이 완료되면, 스프링은 콜백메소드를 통해서 알려준다.
- 또한, 스프링은 빈이 소멸하기 전에 소멸 콜백을 준다.
스프링 빈의 이벤트 라이프 사이클
- 스프링 컨테이너 생성
- 스프링 빈 생성
- 의존관계 주입
- 초기화 콜백
- 객체를 사용
- 소멸전 콜백
- 스프링 종료
생성과 초기화를 분리하는 것이 좋다.
- 생성은 메모리에 올리고, 필수적인 정보 (파라미터 값 세팅) 만 주는것 까지
- 실제 객체의 초기화 작업은 별도의 초기화 메소드로 분리하는 것이 좋다.
- ex) 외부와 커넥션을 맺는 작업
생성과 초기화를 분리해 놓으면 Lazy Loading을 사용할 수 있다.
스프링은 3가지 방법으로 LifeCycle 콜백을 지원한다.
- 인터페이스 (InitializingBean, DisposableBean)
- 설정 정보에 초기화 메서드, 종료 메서드 지정
- @PostConstruct, @PreDestroy 어노테이션 지원
2. 인터페이스 InitializingBean, DisposableBean
InitializingBean
- afterProprertiesSet
DisposableBean
- destroy
아래와 같이 작성하여, 수정자 주입인 경우에도 lifecycle callback을 이용해서
- connect
- call
- disconnect 를 사용할 수 있다.
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();
}
}
단점
- 이 인터페이스는 스프링 전용 인터페이스다.
- 코드가 스프링 전용 인터페이스에 의존한다.
- 코드를 고칠 수 없는 외부 라이브러리에 적용할 수 없다.
- 스프링 초창기에 나온 방법이며, 더 나은 방법이 있기에 잘 사용하지 않는다.
3. 빈 등록 초기화, 소멸 메소드
빈을 등록하는 부분에서 어떤 메소드가 초기화며, 어떤 메소드가 소멸 메소드인지 지정할 수 있다.
- @Bean(initMethod = "초기화 메소드", destroyMethod = "소멸 메소드")
@Configuration
static class LifeCycleConfig {
@Bean(initMethod = "init", destroyMethod = "close")
public NetworkClient networkClient() {
NetworkClient networkClient = new NetworkClient();
networkClient.setUrl("http://hello-spring.dev");
return networkClient;
}
}
public class NetworkClient {
...
public void init() throws Exception {
System.out.println("networkClient.init");
connect();
call("초기화 연결 메시지");
}
public void close() throws Exception {
System.out.println("networkClient.close");
disconnect();
}
}
장점
- 스프링 빈이 스프링 코드에 의존하지 않는다. (설정 정보에서 적용)
- 설정 정보에서 적용하기 때문에 외부라이브러리에도 초기화, 종료 메서드를 적용할 수 있다.
- 메서드 이름을 자유롭게 할 수 있다.
종료 메소드 추론
- destroyMethod 의 default 값은 "(inferred)"로 등록되어있다.
- @Bean(destroyMethod="(inferred)"
- 대부분의 외부 라이브러리들은 close / shutdown 으로 종료 메서드를 사용한다.
- inferred는 close / shutdown이 있으면 자동으로 호출해준다.
- 따라서 스프링빈을 등록하면 종료 메소드를 따로 적어주지 않아도 잘 동작한다.
- 사용하기 싫다면
- @Bean(destroyMethod="") 로 지정
4. @PostConstruct, @PreDestroy
@PostConstruct : 초기화 메서드 위에 붙인다.
@PreDestroy : 소멸 메서드 위에 붙인다.
public class NetworkClient {
...
@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();
}
}
특징
- 현재 스프링에서 가장 권장되는 방법
- javax.annotation.PostConstruct
- 스프링에 종속적인 것이 아닌, 자바 표준이다.
- 스프링이 아닌 다른 컨테이너에서도 동작한다.
- 유일한 단점은 외부 라이브러리에는 적용하지 못한다.
- @Bean(init, destrory) 를 사용하자.
'강의 정리 > 스프링 핵심 원리 - 기본편' 카테고리의 다른 글
7. 빈 스코프 (0) | 2020.12.24 |
---|---|
4. 컴포넌트 스캔 (0) | 2020.12.17 |
3. 싱글톤 컨테이너 (0) | 2020.12.15 |
2. 스프링 컨테이너와 스프링 빈 (0) | 2020.12.13 |
1. 객체지향 설계와 스프링 (0) | 2020.12.12 |