📹 참고 : 인프런 [ 스프링 입문 - 김영한 ]
지금까지 리포지토리, 서비스를 만들어 보았다. 로직을 화면에 붙이기 위해서는 컨트롤러와 뷰 템플릿이 필요한데, 이번에는 컨트롤러에 대해 알아보자.
멤버 컨트롤러에서 멤버 서비스를 이용하여 회원가입하고, 데이터를 조회할 수 있도록 해야한다. 이것을 서로 의존 관계가 있다고 표현.
@Controller 어노테이션을 붙이면, 스프링이 동작할 때 컨테이너에서 컨트롤러 객체를 생성해서 스프링에 넣어두고 관리한다. => 스프링 컨테이너에서 스프링 빈이 관리된다.
java/hello/hellospring/controller/MemberController.java
@Controller
public class MemberController {
private final MemberService memberService;
@Autowired
public MemberController(MemberService memberService) {
this.memberService = memberService;
}
}
MemberService는 여러개의 인스턴스를 생성할 필요가 없다.
하나만 생성해서 공통으로 사용하면 됨 -> 생성자 주입 사용
생성자를 만들어 준 후에 @Autowired annotation을 붙여준다.
@Autowired 가 있으면 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어준다. 이렇게 객체 의존관계를 외부에서 넣어주는 것을 DI (Dependency Injection), 의존성 주입이라 한다.
하지만 위와 같은 오류를 발견할 수 있다.
이 오류는 현재 멤버서비스는 스프링 빈 객체가 아닌 순수한 자바 클래스이기 때문.
즉, memberService가 스프링 빈으로 등록되지 않았다.
@Component 애노테이션이 있으면 스프링 빈으로 자동 등록된다.@Controller 컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문이다.@Component 를 포함하는 다음 애노테이션도 스프링 빈으로 자동 등록된다.@Controller@Service@Repository
위와 같은 연결을 위해서 멤버 서비스 빈 등록과 멤버 리포지토리 빈 등록을 해줘야 한다.
@Service
public class MemberService {
private final MemberRepository memberRepository;
@Autowired
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
}
...
@Repository
public class MemoryMemberRepository implements MemberRepository {...}
참고: 생성자에
@Autowired를 사용하면 객체 생성 시점에 스프링 컨테이너에서 해당 스프링 빈을 찾아 서 주입한다. 생성자가 1개만 있으면@Autowired는 생략할 수 있다.
참고: 스프링은 스프링 컨테이너에 스프링 빈을 등록할 때, 기본으로 싱글톤으로 등록한다(유일하게 하나만 등록해서 공유한다) 따라서 같은 스프링 빈이면 모두 같은 인스턴스다. 설정으로 싱글톤이 아니게 설정할 수 있지 만, 특별한 경우를 제외하면 대부분 싱글톤을 사용한다.
🌟 회원 서비스와 회원 리포지토리의 @Service, @Repository, @Autowired 애노테이션을 제거하고 진행 한다.
java/hello/hellospring/SpringConfig.java
@Configuration
public class SpringConfig {
@Bean //스프링 빈 등록
public MemberService memberService(){
return new MemberService(memberRepository()); //생성자 주입
}
@Bean
public MemberRepository memberRepository(){
return new MemoryMemberRepository();
}
}
패키지 수준에 Config 클래스를 생성 해 준 후 @Configuration 어노테이션을 붙여준다. 그 후 @Bean 어노테이션을 붙여 스프링 빈으로 등록해준다.
-> 스프링이 뜰 때 Configuration을 읽고 스프링이 인식하여 스프링 빈에 등록해준다.
참고
- DI에는 필드 주입, setter 주입, 생성자 주입 이렇게 3가지 방법이 있다. 의존관계가 실행중에 동적으 로 변하는 경우는 거의 없으므로 생성자 주입을 권장한다.
- 실무에서는 주로 정형화된 컨트롤러, 서비스, 리포지토리 같은 코드는 컴포넌트 스캔을 사용한다. 그 리고 정형화 되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록한다.
- XML로 설정하는 방식도 있지만 최근에는 잘 사용하지 않으므로 생략한다.
주의
@Autowired를 통한 DI는helloController,memberService등과 같이 스프링이 관리하 는 객체에서만 동작한다. 스프링 빈으로 등록하지 않고 내가 직접 생성한 객체에서는 동작하지 않는다.