🥔 스프링 빈과 의존관계
스프링 빈을 등록하는 2가지 방법
- 컴포넌트 스캔과 자동 의존관계 설정
- 자바 코드로 직접 스프링 빈 등록하기
컴포넌트 스캔과 자동 의존관계 설정
회원 컨트롤러가 회원서비스와 회원 레파지토리를 사용할 수 있도록 의존 관계를 준비하자.
src/main/java/hello.hellospring/controller 하위에 MemberController파일을 만든다.
MemberController
package hello.hellospring.controller;
import hello.hellospring.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
@Controller // MemberController 객체를 생성해서 스프링에 넣어두고 관리
public class MemberController {
private final MemberService memberService;
@Autowired // memberService를 스프링 컨테이너에 있는 것으로 연결시켜줌
public MemberController(MemberService memberService) {
this.memberService = memberService;
}
}
- 스프링이 처음에 올라갈 때 스프링 컨테이너가 생성되는데 @Controller 어노테이션이 있으면 MemberController 객체를 생성해서 컨테이너에 넣어두고 스프링이 관리하게 된다.
- 생성자에 @Autowired가 있으면 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어준다. 이렇게 객체 의존관계를 외부에서 넣어주는 것을 DI(Dependency Injection), 의존성 주입이라고 한다.
- 현재 MemberService는 순수한 자바 클래스이고 스프링이 MemberService를 알 수 없다. 즉 스프링 컨테이너에 등록되어 있는 상태가 아닌 것이다.
- MemberService에 @Service 어노테이션을 붙여주어 스프링 컨테이너에 등록해줘야 한다.
회원 서비스 스프링 빈 등록
회원 레파지토리 스프링 빈 등록
- memberController는 memberService가 필요하다. 그러기 위해서 memberController에서 @Autowired를 어노테이션을 통해 스프링 컨테이너가 관리하고 있는 memberService를 가져와 사용할 수 있도록 하는 것이다.
- memberService는 memberRepository가 필요하다. 그러기 위해 memberService에서 @Autowired를 어노테이션을 통해 스프링 컨테이너가 관리하고 있는 memberRepository를 가져와 사용할 수 있도록 하는 것이다.
- 이것을 DI(Dependency Injection), 의존성 주입이라고 한다.
컴포넌트 스캔 원리
- @Component 어노테이션이 있으면 스프링 빈으로 자동 등록된다.
- 컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문이다. @Controller 어노테이션을 타고 들어가보면 내부에 @Component 가 있다.
- @Component를 포함하는 다음 어노테이션도 스프링 빈으로 자동 등록된다.
- @Controller
- @Service
- @Repository
스프링은 스프링 컨테이너에 스프링 빈을 등록할 때, 기본으로 싱글톤으로 등록한다.
(유일하게 하나만 등록해서 공유한다.) 따라서 같은 스프링 빈이면 모두 같은 인스턴스다.
설정으로 싱글톤이 아니게 설정할 수 있지만, 특별한 경우를 제외하면 대부분 싱글톤을 사용한다.
자바 코드로 직접 스프링 빈 등록하기
이전에 작성했던 회원 서비스와 회원 레파지토리의 @Service, @Repository, @Autowired 어노테이션을 제거하고, SpringConfig라는 이름의 파일을 하나 만들자.
SpringConfig
@Bean 어노테이션을 통해 스프링 빈을 등록할 수 있다.
이렇게 SpringConfig를 통해서도 스프링 빈을 등록할 수 있고 실행시키면 문제없이 서버가 올라간다.
- DI에는 필드 주입, setter 주입, 생성자 주입 3가지가 있다. 의존관계가 실행중에 동적으로 변하는 경우는 거의 없기 때문에 생성자 주입을 권장한다.
- 실무에서는 주로 정형화된 컨트롤러, 서비스, 레파지토리와 같은 코드는 컴포넌트 스캔을 사용한다. 그리고 정형화 되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록한다.
- @Autowired를 통한 DI는 helloController, memberService등과 같이 스프링이 관리하는 객체에서만 동작한다. 스프링 빈으로 등록하지 않고 내가 직접 생성한 객체에서는 동작하지 않는다.