스프링 빈과 의존 관계
스프링 빈과 의존 관계
컴포넌트 스캔과 자동 의존 관계 설정
회원 컨트롤러가 회원 서비스와 회원 리포지토리를 사용할 수 있게 의존 관계 (멤버 컨트롤러가 멤버 서비스를 통해서 회원 가입, 데이터 조회 등을 할 수 있게 함)를 준비.
회원 컨트롤러에 의존 관계 추가
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 {
// 기능은 없지만, 스프링 컨테이너라는 스프링 통에 MemberController 객체를 생성해서 스프링에 넣어두고 관리 (스프링 빈이 관리된다.)
} */
@Controller
public class MemberController {
// 멤버서비스를 가져다 쓰기 위해서 new로 가져옴
// private final MemberService memberService = new MemberService();
// 하지만, 스프링이 관리하게 되면 모두 스프링 컨테이너에 등록하고 스프링 컨테이너를 통해서 사용하게 해야 한다.
// 해결 방안
private final MemberService memberService;
@Autowired // memberService를 스프링이 스프링 컨테이너에 있는 memberService를 가져다 연결 해준다.(Dependency Injection)
public MemberController(MemberService memberService) {
this.memberService = memberService;
}
// 오류발생 : 'hello.hellospring.service.MemberService' that could not be found.
// 해결 : MemberSerivce가 순수한 자바 코드 여서 생긴 문제 => MemberService에 @Service 달아 주기, MemoryMemberRepository에 @Repository 달아 주기
}
- 생성자에
@Autowired
가 있으면 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어줌, 이렇게 객체 의존관계를 외부에서 넣어주는 것을 DI(Dependency Injection), 의존성 주입이라 함 - 이전 테스트에서는 개발자가 직접 주입했고, 여기선
@Autowired
에 의해 스프링이 주입 해줌
오류발생
Consider defining a bean of type 'hello.hellospring.service.MemberService' in
your configuration
memberService가 스프링 빈으로 등록 되어 있지 않음.
참고 : helloController는 스프링이 제공하는 컨트롤러여서 스프링 빈으로 자동 등록 된다.
@Controller
가 있으면 자동 등록
스프링 빈을 등록하는 2가지 방법
- 컴포넌트 스캔과 자동 의존 관계 설정
- 자바 코드로 직접 스프링 빈 등록
컴포넌트 스캔 원리
@Component
애노테이션이 있으면 스프링 빈으로 자동 등록@Controller
컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문@Component
를 포함하는 다음 애노테이션도 스프링 빈으로 자동 등록@Controller
@Service
@Repository
회원 서비스 스프링 빈 등록
@Service
public class MemberService {
// 같은 인스턴스에서 데이터를 사용하게 만들기 위해
// private final MemberRepository memberRepository = new MemoryMemberRepository();
private final MemberRepository memberRepository;
@Autowired
// 외부에서 넣어주게 바꿔줌 (alt + Insert)
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
}
참고 : 생성자에
@Autowired
를 사용하면 객체 생성 시점에 스프링 컨테이너에서 해당 스프링 빈을 찾아서 주입, 생성자가 1개만 있으면@Autowired
생략 가능
회원 리포지토리 스프링 빈 등록
@Repository
public class MemoryMemberRepository implements MemberRepository {}
스프링 빈 등록 이미지
memberService
와memberRepository
가 스프링 컨테이너에 스프링 빈으로 등록
참고 : 스프링은 스프링 컨테이너에 스프링 빈을 등록할 때, 기본으로 싱글톤(리포지토리는 리포지토리 하나만, 서비스도 서비스 하나만)으로 등록한다(유일하게 하나만 등록해서 공유) 따라서 같은 스프링 빈이면 모두 같은 인스턴스, 설정으로 싱글톤이 아니게 설정할 수 있지만, 특별한 경우를 제외하면 대부분 싱글톤을 사용.
자바 코드로 직접 스프링 빈 등록하기
- 회원 서비스와 회원 리포지토리의
@Service
,@Repository
,@Autowired
애노테이션을 제거하고 진행한다.
package hello.hellospring.service;
import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
// JAVA코드로 직접 등록하는 방법
@Configuration
public class SpringConfig {
@Bean // Spring Bean에 등록할꺼다
public MemberService memberService() {
return new MemberService(memberRepository());
}
@Bean // 얘도 Spring Bean에 등록
public MemberRepository memberRepository() {
return new MemoryMemberRepository();
}
}
여기서는 향후 메모리 리포지토리를 다른 리포지토리로 변경할 예정이므로, 컴포넌트 스캔 방식 대신에 자바 코드로 스프링 빈을 설정한다.
참고 : XML로 설정하는 방식도 있지만 최근에는 잘 사용하지 않음
참고 : DI에는 필드 주입, setter주입, 생성자 주입 이렇게 3가지 방법이 있음, 의존관계가 실행중에 동적으로 변하는 경우는 거의 없으므로 생성자 주입을 권장
// 1. 필드 주입 : 선호 되지 않음, 바꿀수 있는 방법이 없기 때문에
// @Autowired private MemberService memberService;
// 2. 생성자 주입 (요즘 권장)
// Auto wired : memberService를 스프링 컨테이너에 있는 memberService를 가져다 연결(DI)
@Autowired
public MemberController(MemberService memberService) {
this.memberService = memberService;
}
// 3. setter 주입 : 누군가 memberController를 호출했을때 public으로 열려있어야함
// (private MemberService memberService : final 지워야함)
// @Autowired
// public void setMemberService(MemberService memberService) {
// this.memberService = memberService;
// }
참고 : 실무에서는 주로 정형화된 컨트롤러, 서비스, 리포지토리 같은 코드는 컴포넌트 스캔을 사용하나 정형화 되지 않거나, 상황(MemoryMemberRepository를 다른 Repository로 교체)에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록한다. 이를 통해 구 현체를 교체 하는데 기존 운영중인 코드를 손대지 않고 교체하기 위함
주의 :
@Autowired
를 통한 DI는helloController
,memberService
등 과 같이 스프링이 관리하는 객체에서만 동작, 스프링 빈으로 등록 하지 않고 내가 직접 생성한 객체에서는 동작하지 않음
// 스프링 컨테이너에 등록되어있는 것들만 Autowired 사용가능
// 내가 직접 생성한 것들에는 동작하지 않음
// @Autowired
// public static void main(String[] args) {
// MemberService memberService = new MemberService();
// }
스프링 컨테이너, DI 관련된 자세한 내용은 스프링 핵심 원리 강의에서 설명