본문 바로가기

Spring Core

[Spring Core] 빈 생명주기 콜백 , 빈 스코프

Spring Bean 생명주기

Spring Bean의 생명주기는 Spring 프레임워크에서 빈이 생성되고 소멸될 때까지의 전체 과정이다.

이 과정은 여러 단계로 이루어져 있으며, 각 단계는 Spring 컨테이너에 의해 관리된다.

Spring Bean 생명주기 단계

  1. Bean 정의(Definition): 개발자가 XML 설정 파일, 애노테이션 또는 Java 설정 클래스를 통해 Bean을 정의한다.
  2. Bean 생성(Creation): Spring 컨테이너가 정의된 Bean을 인스턴스화합니다. 기본적으로는 Java의 new 키워드를 사용하여 객체를 생성한다.
  3. 의존성 주입(Dependency Injection): 생성된 Bean에 필요한 의존성을 주입한다. 이는 생성자 주입, 세터 주입 등을 통해 이루어진다.
  4. 초기화(Initialization):
    • BeanNameAwareBeanFactoryAware: Bean이 자신의 이름과 BeanFactory를 알 수 있도록 한다.
    • BeanPostProcessor의 postProcessBeforeInitialization 메서드: Bean의 초기화 전 호출.
    • InitializingBean의 afterPropertiesSet 메서드 또는 사용자 정의 초기화 메서드: Bean의 초기화 로직이 실행.
    • BeanPostProcessor의 postProcessAfterInitialization 메서드: Bean의 초기화 후 호출.
  5. 사용(Use): 애플리케이션에서 Bean이 사용된다. 이는 서비스 로직에서 Bean을 호출하여 실제 기능을 수행하는 단계이다.
  6. 소멸(Destruction):
    • DisposableBean의 destroy 메서드 또는 사용자 정의 소멸 메서드: Bean이 소멸되기 전에 정리 작업을 수행한다.

@PostConstruct, @Predestroy

1. 클래스가 InitializingBean과 DisposalBean을 상속받고 각각 afterPropertiesSet과 destroy 메서드를 구현하는 방식이다.

이때 초기화 콜백시 afterPropertiesSet이, 소멸전 콜백색 destroy가 호출된다.
2. @Bean 애노테이션에 내장된 설정방식으로 각각 initMethod 필드와 destroyMethod필드의 생성과 소멸시 적용될 메서드명을 정해주면 된다.
→ @Bean(initMethod="start", destroyMethod="end")
3. @PostConstruct, @Predestroy 가 일반적으로 가장많이 사용하는 방식이다.

package hello.core.lifecycle;
import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;

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("close + " + url);
    }
    
    @PostConstruct
    public void init() {
        System.out.println("NetworkClient.init");
        connect();
        call("초기화 연결 메시지");
    }
    
    @PreDestroy
    public void close() {
        System.out.println("NetworkClient.close");
        disConnect();
    }
}

 

@PostConstruct` , `@PreDestroy` 이 두 애노테이션을 사용하면 가장 편리하게 초기화와 종료를 실행할 수 있다.

 

@PostConstruct, @PreDestroy  특징

  • 최신 스프링에서 가장 권장하는 방법이다.
  • 애노테이션 하나만 붙이면 되므로 매우 편리하다.
  • 패키지를 잘 보면 `javax.annotation.PostConstruct` 이다. 스프링에 종속적인 기술이 아니라 JSR-250라는 자바 표준이다. 따라서 스프링이 아닌 다른 컨테이너에서도 동작한다.
  • 컴포넌트 스캔과 잘 어울린다.
  • 유일한 단점은 외부 라이브러리에는 적용하지 못한다는 것이다. 외부 라이브러리를 초기화, 종료 해야 하면 @Bean의 기능을 사용하자

빈 스코프

스프링 빈이 스프링 컨테이너의 시작과 함께 생성되어서 스프링 컨테이너가 종료될 때 까지 유지된다.
이것은 스프링 빈이 기본적으로 싱글톤 스코프로 생성되기 때문이다.

스코프는 번역 그대로 빈이 존재할 수 있는 범위를 뜻한다.

 

스프링 지원 스코프

싱글톤

  -  기본 스코프, 스프링 컨테이너의 시작과 종료까지 유지되는 가장 넓은 범위의 스코프이다.
프로토타입 

  - 스프링 컨테이너는 프로토타입 빈의 생성과 의존관계 주입까지만 관여하고 더는 관리하지 않는 매우
짧은 범위의 스코프이다.
웹 관련 스코프
 - request : 웹 요청이 들어오고 나갈때 까지 유지되는 스코프이다.
 - session : 웹 세션이 생성되고 종료될 때 까지 유지되는 스코프이다.
 - application : 웹의 서블릿 컨텍스트와 같은 범위로 유지되는 스코프이다

컴포넌트 스캔 자동 등록

@Scope("prototype")
@Component
public class HelloBean {}

수동 등록

@Scope("prototype")
@Bean
PrototypeBean HelloBean() {
    return new HelloBean();
}

 

싱글톤 스코프의 빈을 조회하면 스프링 컨테이너는 같은 인스턴스의 스프링 빈을 반환한다.

 

 

 

프로토타입 스코프를 스프링 컨테이너에 조회하면 스프링 컨테이너는 새로운 인스턴스를 생성해서 반환한다.

 

 

1. 프로토타입 스코프의 빈을 스프링 컨테이너에 요청한다.
2. 스프링 컨테이너는 이 시점에 프로토타입 빈을 생성하고, 필요한 의존관계를 주입한다.

 

 

3. 스프링 컨테이너는 생성한 프로토타입 빈을 클라이언트에 반환한다.
4. 이후에 스프링 컨테이너에 같은 요청이 오면 항상 새로운 프로토타입 빈을 생성해서 반환한다.

 

[참고]

스프링 핵심 원리 - 기본편 | 김영한 - 인프런 (inflearn.com)

 

스프링 핵심 원리 - 기본편 | 김영한 - 인프런

김영한 | 스프링 입문자가 예제를 만들어가면서 스프링의 핵심 원리를 이해하고, 스프링 기본기를 확실히 다질 수 있습니다., 스프링 핵심 원리를 이해하고, 성장하는 백엔드 개발자가 되어보

www.inflearn.com

 

'Spring Core' 카테고리의 다른 글

[Spring Core] 스프링AOP 용어 정리  (0) 2024.06.23
[Spring Core] 스프링 용어 정리  (0) 2024.06.23
[Spring Core] Component Scan  (0) 2024.05.25