[Spring] 스프링 빈을 등록하고, 의존관계 설정하기

Haeun Noh·2023년 8월 21일
0

TweeksStudy:1 (Spring)

목록 보기
12/13
post-thumbnail

0822


회원 컨트롤러가 회원서비스와 회원 리포지토리를 사용할 수 있게 의존관계를 준비하자!

0. 참고

0.1. Service

  • Controller를 통해 외부 요청을 받고,
  • Service에서 비즈니스 로직을 만들고,
  • Repository에서 데이터를 저장한다.

Service : Spring Container에 Service를 등록해줌


1. 문제 발생

private final MemberService memberService = new MemberService();
  • MemberService는 하나만 생성한 뒤 공용으로 쓰면 되는데, new를 쓰면 불필요한 인스턴스가 많이 생기게 됨.
private final MemberService memberService;

-> Spring Container에 딱 한 개만 등록하기!


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

  • 생성자에 @Autowired가 있으면 스프링이 연관된 객체를 스프링 컨테이너에 찾아서 넣어준다. 이렇게 객체 의존 관계를 외부에서 넣어주는 것을 DI (Dependency Injection) 의존성 주입이라고 한다.
  • 이전 테스트에서는 개발자가 직접 주입했고, 여기서는 @Autowired에 의해 스프링이 주입해준다.

3. 에러 발생

    private final MemberService memberService;

    @Autowired
    public MemberController(MemberService memberService) {
        this.memberService = memberService;
    }
Consider defining a been of type 'hello.hellospring.service.MemberService' in your configuration.
  • memberService가 스프링 빈으로 등록되어 있지 않다.

참고: MemberController는 스프링이 제공하는 컨트롤러여서 스프링 빈으로 자동 등록된다.
@Controller가 있으면 자동 등록됨.


4. 스프링 빈을 등록하는 2가지 방법

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

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

  • @Component 에노테이션이 있으면 스프링 빈으로 자동 등록됨

  • @Controller 컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문이다.

  • @Service, @Controller, @Repository

    • @Component 에노테이션을 포함하기 때문에 스프링 빈으로 자동 등록된다.

Q. 아무때나 @Component 가 있어도 될까요??
A. 기본적으로는 컴포넌트 스캔의 대상이 안 됩니다.

6. 회원 서비스 스프링 빈 등록

@Repository
public class MemoryMemberRepository implements MemberRepository {
  • memberServicememberRepository가 스프링 컨테이너에 스프링 빈으로 등록되었다.

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

  • 싱글톤 : 유일하게 하나만 등록한다.

7. 자바 코드로 직접 스프링 빈 등록하기 (중요!)

  • 회원 서비스와 회원 리포지토리의 @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;

@Configuration
public class SpringConfig {
    
    @Bean // 스프링 빈을 내가 등록할거야
    public MemberService memberService() {
        return new MemberService(memberRepository());
    }

    @Bean// 스프링 빈에 등록하기 위한 객체 생성
    public MemberRepository memberRepository() {
        return new MemoryMemberRepository();// 구현체
    }
}
  • @Configuration 에노테이션을 보고 스프링 빈에 등록하라는 뜻이라는 것을 알게 된다.
  • 새롭게 만들어진 MemberService 객체는 스프링 빈에 등록이 된다.

"여기서는 향후 메모리 리포지토리를 다른 리포지토리로 변경할 예정이므로, 컴포넌트 스캔 방식 대신에 자바 코드로 스프링 빈을 설정하겠다"

참고 : XML로 설정하는 방식도 있지만 최근에는 잘 사용하지 않으므로 생략한다.

참고 : DI에는 필드 주입, setter 주입, 생성자 주입 이렇게 3가지 방법이 있다. 의존관계가 실행중에 동적으로 변하는 경우는 거의 없으므로 생성자 주입을 권장한다.

@Autowired private final MemberService memberService;
  • 필드 주입
    • 별로 좋지 않음.
@Autowired
    public MemberController(MemberService memberService) {
        this.memberService = memberService;
    }
  • 생성자 주입
    • setter주입도 이와 비슷한 구조

참고 : 실무에서는 주로 정형화된 컨트롤러, 서비스, 리포지토리 같은 코드는 컴포넌트 스캔을 사용한다. 그리고 정형화되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록한다.

주의 : @Autowired를 통한 DI는 helloController, MemberService등과 같이 스프링이 관리하는 객체에서만 동작한다. 스프링 빈으로 등록하지 않고 내가 직접 생성한 객체에서는 동작하지 않는다.



profile
Tistory로 옮기게 되었습니다. @haeunnohh

0개의 댓글