@Controller
라는 애너테이션을 보고 스프링이 뜰 때 해당 컨트롤러 객체를 생성해서 관리
-> 스프링 컨테이너에서 스프링 빈이 관리된다 라고 표현
MemberController.java
package hello.hellospring.controller;
import hello.hellospring.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
@Controller
public class MemberController {
private final MemberService memberService;
@Autowired
public MemberController(MemberService memberService){
this.memberService=memberService;
}
}
@Controller
애너테이션이 있으면 MemberController 객체를 생성해서 스프링에 넣어두고 관리@Autowired
를 사용하면 스프링 컨테이너에 있는 memberService를 가져다가 연결(memberService도 스프링 컨테이너에 등록해야함)MemberService.java
package hello.hellospring.service;
import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Service
public class MemberService {
private final MemberRepository memberRepository;
@Autowired
public MemberService(MemberRepository memberRepository){
this.memberRepository=memberRepository;
}
...
@Service
애너테이션을 붙이면 MemberService를 서비스로 인식하고 스프링 컨테이너에 등록해줌@Autowired
를 사용해서 스프링 컨테이너에 있는 MemoryMemberRepository를 연결MemoryMemberRepository.java
package hello.hellospring.repository;
import hello.hellospring.domain.Member;
import org.springframework.stereotype.Repository;
import java.util.*;
@Repository
public class MemoryMemberRepository implements MemberRepository{
...
}
@Repository
애너테이션을 붙이면 스프링 컨테이너에 등록전체적으로 Controller를 통해서 외부 요청을 받고, Service에서 비즈니스 로직을 만들고, Repository를 이용해서 데이터를 저장하는것이 정형화된 패턴
정리
@Component
애너테이션이 있으면 스프링 빈으로 자동 등록
@Component
를 포함하는 애너테이션도 스프링 빈으로 자동 등록
- @Controller
- @Service
- @Repository
@Controller
컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문- Autowired는 연관관계 관련
참고: 스프링은 스프링 컨테이너에 스프링 빈을 등록할 때, 기본으로 싱글톤으로 등록한다(유일하게 하나만 등록해서 공유한다)
따라서 같은 스프링 빈이면 모두 같은 인스턴스다. 설정으로 싱글톤이 아니게 설정할 수 있지만, 특별한 경우를 제외하면 대부분 싱글톤을 사용한다.
위의 예제에서 MemberController.java
에 적은 애너테이션만 남기고 MemberService.java
, MemoryMemberRepository.java
에 적은 애너테이션은 제거
MemberController.java
package hello.hellospring.controller;
import hello.hellospring.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
@Controller
public class MemberController {
private final MemberService memberService;
@Autowired
public MemberController(MemberService memberService){
this.memberService=memberService;
}
}
SpringConfig.java
package hello.hellospring;
import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;
import hello.hellospring.service.MemberService;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class SpringConfig {
@Bean
public MemberService memberService(){
return new MemberService(memberRepository());
}
@Bean
public MemberRepository memberRepository(){
return new MemoryMemberRepository();
//MemberRepository는 인터페이스이고 MemoryMemberRepository가 구현체
}
}
@Configuration
을 읽고 @Bean
스프링 빈에 등록하라는 표시로 인식해서 객체를 호출 후 등록
- DI에는 필드 주입, setter 주입, 생성자 주입 이렇게 3가지 방법이 있다. 의존관계가 실행중에 동적으로 변하는 경우는 거의 없으므로 생성자 주입을 권장한다.
- 실무에서는 주로 정형화된 컨트롤러, 서비스, 리포지토리 같은 코드는 컴포넌트 스캔을 사용한다.
그리고 정형화 되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록한다.- @Autowired 를 통한 DI
의존성 주입(Dependency Injection)
는 helloConroller , memberService 등과 같이 스프링이 관리하는 객체에서만 동작한다. 스프링 빈으로 등록하지 않고 내가 직접 생성한 객체에서는 동작하지 않는다.(스프링 컨테이너에 올라가있는 것들만 Autowired기능이 동작 )
- 의존성 주입(Dependency Injection) : 두 객체 간의 관계(의존성)를 맺어주는 것