우선 의존관계를 보기위해 컨트롤러를 만들고 서비스와 레포지토리도 만들어 보자.
그리고 컨트롤러가 서비스와 레포지토리를 사용할 수 있게 의존관계도 준비해보자.
@Controller
public class MemberController {
private final MemberService memberService;
@Autowired
public MemberController(MemberService memberService) {
this.memberService = memberService;
}
}
참고: 컨트롤러에서는 @Controller가 있으면 스프링 빈으로 자동 등록된다.
생성자에 @AutoWired
가 있으면 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어준다. 이렇게 객체 의존관계를 외부에서 넣어주는 것이 DI(Dependency Injection), 의존성 주입이라 한다.
그치만 이렇게 실행하게 된다면 오류가 발생하게 될 것이다.
public class MemberService {
private final MemberRepository memberRepository;
@Autowired
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
}
왜냐하면 memberService가 스프링 빈으로 등록되어 있지 않다.
여기서 그럼 스프링 빈에는 어떻게 등록할까? 라는 의문이 들텐데
스프링 빈을 등록하는 2가지 방법❗️
@Component
애노테이션이 있으면 스프링 빈으로 자동 등록된다.@Controller
컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문이다.@Component
를 포함하는 다음 애노테이션도 스프링 빈으로 자동 등록된다.
@Controller
@Service
@Repository
@Controller
를 들어가보면 ‼️
결국 @Component
가 있어서 자동 등록된다는 이유다. (@Service
, @Repository
도 들어가 보면 다 있음)
@Controller
, @Service
, @Repository
각각 추가해주면 그림과 같이 memberController, memberService와 memberRepository가 스프링 컨테이너에 스프링 빈으로 등록된다.
참고: 스프링은 스프링 컨테이너에 스프링 빈을 등록할 때, 기본으로 싱글톤으로 등록한다. (유일하게 하나만 등록해서 공유) 따라서 같은 스프링 빈이면 모두 같은 인스턴스다. (거의 대부분 싱글톤을 사용)
@Service
, @Repository
, @Autowired
애노테이션을 제거하고 진행한다.
@Configuration
public class SpringConfig {
@Bean
public MemberService memberService() {
return new MemberService(memberRepository());
}
@Bean
public MemberRepository memberRepository() {
return new MemoryMemberRepository();
}
}
SpringConfig.java
우선은 메모리 레포지토리로 하지만 Jpa나 jdbc로 변경 가능, 컴포넌트 스캔 방식 대신에 자바 코드로 스프링 빈 설정한다.
DI에는 필드 주입, setter 주입, 생성자 주입 이렇게 3가지 방법이 있다. 의존관계가 실행중에 동적으로 변하는 경우는 거의 없으므로 생성자 주입을 권장한다.
실무에서는 주로 정형화된 컨트롤러, 서비스, 리포지토리 같은 코드는 컴포넌트 스캔을 사용한다.
그리고 정형화 되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록한다.
@Autowired
를 통한 DI는 helloController , memberService 등과 같이 스프링이 관리하는 객체에서만 동작한다. 스프링 빈으로 등록하지 않고 내가 직접 생성한 객체에서는 동작하지 않는다.
⭐⭐⭐ 인프런에서 진행한 김영한님의 강의(스프링 입문)를 토대로 정리한 포스팅입니다. 자세한 내용은 링크를 참고하시기 바랍니다. ⭐⭐⭐