뭘 어떻게 공부해야하지?
📌 빈 생명주기 콜백
📌 인터페이스 InitializingBean, DisposableBean
📌 빈 등록 초기화, 소멸 메서드 지정
📌 애노테이션 @PostConstruct, @PreDestroy
객체의 초기화와 종료작업이 필요한 경우가 있다.
사용 메서드
connect()
: 애플리케이션 시작 시점에 호출해 연결 맺기disconnect()
: 애플리케이션 종료 시점에 호출해 연결 끊기예제코드
public class NetworkClient {
private String url;
public NetworkClient() {
System.out.println("생성자 호출, url = " + url);
connect();
call("초기화 연결 메시지");
}
// 그러면 url이 이상한데?
public void setUrl(String url) {
this.url = url;
}
//서비스 시작 시 호출
public void connect() {
System.out.println("connet : " +url);
}
public void call(String message) {
System.out.println("call: "+ url + "message = " + message);
}
//서비스 종료시 호출
public void disconnect() {
System.out.println("close: " + url);
}
}
테스트 코드
public class BeanLifeCycleTest {
@Test
public void lifeCycleTest() {
ConfigurableApplicationContext ac =
new AnnotationConfigApplicationContext(LifeCycleConfig.class);
NetworkClient client = ac.getBean(NetworkClient.class);
ac.close(); ////스프링 컨테이너를 종료, ConfigurableApplicationContext 필요
}
@Configuration
static class LifeCycleConfig {
@Bean
public NetworkClient networkClient() {
NetworkClient networkClient = new NetworkClient();
networkClient.setUrl("가짜 url");
return networkClient;
}
}
}
NetworkClient()
를 통해서 객체를 생성했다. close()
를 가진 ConfigurableApplicationContext
인터페이스를 사용하였다.NetworkClient()
는 다음과 같다. System.out.println("생성자 호출, url = " + url);
connect();
call("초기화 연결 메시지");
👉 우리가 기대하는 것은, 생성자를 호출할 때 url을 출력하고, 해당 url에 연결 후 , 연결이 완료되었다는 메시지가 뜨는 것이다.
🤔 실제로는 어땠는지 확인해보자.
💻 실행화면
생성자 호출, url = null
connet : null
call: null message = 초기화 연결 메시지
👉 연결이 되지 않았다!! 당연하지!! 아직 url 정보를 제공하지 않있기 떼문이다.
📌 객체 생성후 → 의존관계 주입 생성자 주입이 아닐 경우
스프링 빈은 객체를 생성하고, 의존관계 주입이 다 끝난 후에 필요한 데이터를 사용할 수 있다. 👉 즉, 초기화 작업은 의존관계 주입이 모두 완료되고 난 다음에 호출되어야한다.
📌 초기화란 : 객체에 필요한 값이 연결되어 있거, 사용할 수 있도록 셋팅된 단계
🤔 어떻게 의존관계 주입이 완료된 시점을 알 수 있을까?
✔️ 스프링은 의존관계 주입이 완료되면 스프링 빈에게 콜백 메서드를 통해서 초기화 시점을 알려주는 다양한 기능을 제공한다.
✔️ 또한 스프링은 스프링 컨테이너가 종료되기 전에 소멸 콜백을 준다. → 안전한 종료 작업 가능
<📌 싱글톤의 경우>
1. 스프링 컨테이너 생성
2. 스프링 빈 생성
3. 의존관계 주입 (필드, 수정자, 메소드)
4. 초기화 콜백
5. --사용--
6. 소멸전 콜백
7. 스프링 종료
✔️ 객체의 생성과 초기화를 분리하자
단일 책임 원칙을 기억하자~
따라서 생성자 안에서 무거운 초기화 작업을 함께 하는 것 보다는 객체를 생성하는 부분과 초기화 하는 부분을 명확하게 나누는 것이 유지보수 관점에서 좋다. 물론 초기화 작업이 내부 값들만 약간 변경하는 정도로 단순한 경우에는 생성자에서 한번에 다 처리하는게 더 나을 수 있다.
➕ 초기화를 미룰 수 있는 장점도 존재하다.
⭐ 따라서 스프링 빈 생명주기 콜백을 지원받아야한다.
📌 인터페이스(InitializingBean, DisposableBean)
📌 설정 정보에 초기화 메서드, 종료 메서드 지정
📌 @PostConstruct, @PreDestroy 애노테이션 지원
InitializingBean
, DisposableBean
인터페이스를 통해서 객체의 초기화와 종료 작업을 지원한다.
InitializingBean
: 🔨afterPropertiesSet()
메서드로 초기화를 지원한다DisposableBean
: 🔨destroy()
메서드로 소멸을 지원한다.ac.close()
예제코드 수정
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("connet : " +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();
}
}
테스트 코드 수정
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("가짜 url");
return networkClient;
}
}
}
💻 실행화면
ac.close()
생성자 호출, url = null
-> 초기화 작업
connet : http://hello-spring.dev
call: http://hello-spring.dev message = 초기화 연결 메시지
-> 종료 작업
close: http://hello-spring.dev
➡️ 자주 사용되지 않는다.
빈 설정 정보를 사용해서 초기화 메서드, 소멸 메서드를 지정할 수 있다.
@Bean(initMethod = "init",destroyMethod = "close")
initMethod = "메소드명"
: 초기화 메서드destroyMethod = "메소드명"
: 소멸 메서드설정 정보를 사용하도록 변경
public class NetworkClient {
public void init() {
System.out.println("NetworkClient.init");
connect();
call("초기화 연결 메시지");
}
public void close(){
System.out.println("NetworkClient.close");
disconnect();
}
}
테스트 코드
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(initMethod = "init",destroyMethod = "close")
public NetworkClient networkClient() {
NetworkClient networkClient = new NetworkClient();
networkClient.setUrl("http://hello-spring.dev");
return networkClient;
}
}
}
💻 실행결과
생성자 호출, url = null
-> 생성 후에 초기화
NetworkClient.init
connet : http://hello-spring.dev
call: http://hello-spring.dev message = 초기화 연결 메시지
-> 종료 시 메소드
NetworkClient.close
close: http://hello-spring.dev
NetworkClient
에는 스프링 코드 ❌@Bean(destroyMethod="(inferred)")
: 기본값이 "(inferred)"
으로 설정되어있다.
🤔 이건 무엇일까?
외부라이브러리의 메서드명을 추론해서 종료메서드를 호출한다.
ex) 외부라이브러리의 close
,shutdown
라는 이름의 메서드를 호출한다.
➡️ 따라서 종료메서드를 스프링 빈으로 등록하면 종료메서드를 따로 적어주지 않아도 작동한다. 추론 기능을 사용하고 싶지 않은 경우 destroyMethod=""
처럼 공백으로 지정하면 된다.
@PostConstruct
, @PreDestroy
애노테이션을 사용해 등록한다.👍 스프링에서 권장하는 방법이다!!
단점 ) 외부 라이브러리에는 적용하지 못한다는 것이다
수정코드
@PostConstruct
, @PreDestroy
애노테이션을 사용해 등록한다. public class NetworkClient {
@PostConstruct
public void init() {
System.out.println("NetworkClient.init");
connect();
call("초기화 연결 메시지");
}
@PreDestroy
public void close(){
System.out.println("NetworkClient.close");
disconnect();
}
}
테스트 코드
💻 실행화면
생성자 호출, url = null
NetworkClient.init
connet : http://hello-spring.dev
call: http://hello-spring.dev message = 초기화 연결 메시지
NetworkClient.close
close: http://hello-spring.dev
@PostConstruct, @PreDestroy 애노테이션을 사용하자
코드를 고칠 수 없는 외부 라이브러리를 초기화, 종료해야 하면 @Bean의 initMethod , destroyMethod를사용하자.