스프링 빈과 의존 관계

스프링 빈과 의존 관계

  1. 컴포넌트 스캔과 자동 의존 관계 설정
  2. 자바 코드로 직접 스프링 빈 등록하기

컴포넌트 스캔과 자동 의존 관계 설정

회원 컨트롤러가 회원 서비스와 회원 리포지토리를 사용할 수 있게 의존 관계 (멤버 컨트롤러가 멤버 서비스를 통해서 회원 가입, 데이터 조회 등을 할 수 있게 함)를 준비.

회원 컨트롤러에 의존 관계 추가

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가 스프링 빈으로 등록 되어 있지 않음.

Untitled

참고 : 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 {}

스프링 빈 등록 이미지

Untitled

  • memberServicememberRepository 가 스프링 컨테이너에 스프링 빈으로 등록

참고 : 스프링은 스프링 컨테이너에 스프링 빈을 등록할 때, 기본으로 싱글톤(리포지토리는 리포지토리 하나만, 서비스도 서비스 하나만)으로 등록한다(유일하게 하나만 등록해서 공유) 따라서 같은 스프링 빈이면 모두 같은 인스턴스, 설정으로 싱글톤이 아니게 설정할 수 있지만, 특별한 경우를 제외하면 대부분 싱글톤을 사용.

자바 코드로 직접 스프링 빈 등록하기

  • 회원 서비스와 회원 리포지토리의 @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 관련된 자세한 내용은 스프링 핵심 원리 강의에서 설명

You might also enjoy