[Spring Boot][1] 4. 스프링 빈과 의존관계

sorzzzzy·2021년 8월 9일
0

Spring Boot - 입문편

목록 보기
4/10
post-thumbnail

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

✔️ main/java/hello.hellospring/controller/MemberControllerl.java


위와 같이 private final MemberService memberService = new MemberSerice(); 로 만들수도 있겠지만,
Spring이 관리를 하게 되면 다 컨테이너에 등록을 하게 되고 컨테이너로부터 받아서 쓰는 방법으로 바꿔야 한다.
그러나 위와 같은 방법으로 하면, 멤버 컨트롤러 말고 다른 여러 컨트롤러들이 멤버 서비스를 가져다 쓸 수 있는데 그럴 때 마다 여러개씩 생성하게 된다.
사실상 그럴 필요가 없다! 하나만 있으면 되기 때문에 우리는 다른 방법을 사용해보도록 하자!

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;
    }
}

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

이것을 실행하면, 아래와 같은 오류가 뜬다 ! 왜 일까 ??
Consider defining a bean of type 'hello.hellospring.service.MemberService' in your configuration.


💡 memberService가 스프링 빈으로 등록되어 있지 않기 때문

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

✔️ 스프링 빈을 등록하는 2가지 방법

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


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

💡 컴포넌트 스캔 : 설정 정보 없이 자동으로 스프링 빈을 등록해주는 어노테이션

  • @Component 애노테이션이 있으면 스프링 빈으로 자동 등록된다.
  • @Controller 컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문이다.
  • @Component 를 포함하는 다음 애노테이션도 스프링 빈으로 자동 등록된다.
    • @Controller
    • @Service
    • @Repository

회원 서비스 스프링 빈 등록하기 !

✔️ main/java/hello.hellospring/service/MemberService.java@Service @Autowired 추가

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;

import java.util.List;
import java.util.Optional;

@Service
public class MemberService {

    // private final MemberRepository memberRepository = new MemoryMemberRepository();

    private final MemberRepository memberRepository;
    
    @Autowired
    public MemberService(MemberRepository memberRepository) {
        this.memberRepository = memberRepository;
    }
    .
    .
    .

회원 리포지토리 스프링 빈 등록하기 !

✔️ main/java/hello.hellospring/repository/MemortyMemberRepository.java@Repository 추가

package hello.hellospring.repository;

import hello.hellospring.domain.Member;
import org.springframework.stereotype.Repository;

import java.util.*;

@Repository
public class MemoryMemberRepository implements  MemberRepository{
    // 옵션+엔터 -> Implement method
    private static Map<Long, Member> store = new HashMap<>();
    private static long sequence = 0L;
	.
    .
    .

그리고 나서 HelloSpringApplication 실행시켜보기 ! 결과는 ?

스프링이 컨테이너를 만들 때 문제 없이 잘 진행되었다는 의미이다.
아직 회원 컨트롤러에 관한 어떤 기능도 없기 때문에, 아직 연결만 된 상태이다!

스프링 빈 등록 이미지

memberServicememberRepository 가 스프링 컨테이너에 스프링 빈으로 등록되었다.

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



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

회원 서비스와 회원 리포지토리의 @Service, @Repository, @Autowired 애노테이션을 제거하고 진행해보자!

✔️ main/java/hello.hellospring/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 : 스프링 빈을 등록한다는 의미
        @Bean
        public MemberService memberService() {
            return new MemberService(memberRepository());
        }
        @Bean
        public MemberRepository memberRepository() {
          return new MemoryMemberRepository();
        }
}

이렇게 되면 멤버 서비스와 멤버 리포지토리를 둘 다 스프링 빈에 등록을 하고, 그러면서 스프링 빈에 등록되어있는 멤버 리포지토리를 멤버 서비스에 넣어준다.

자바 코드로 직접 스프링 빈을 등록했으니, 한 번 더 HelloSpringApplication 을 실행시켜보자!

역시나 잘 돌아간다😃

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

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


+) 🏷 DI(Dependency Injection)

  • DI에는 필드 주입, setter 주입, 생성자 주입 이렇게 3가지 방법이 있다.
  1. 필드 주입 : 생성자 없이 필드에 @Autowired 를 사용하는 방법 (별로 권장하는 방법은 아님)
  2. setter 주입 : setter@Autowired 를 사용하는 방법
  3. 생성자 주입 : 생성자를 통해 멤버 서비스가 멤버 컨트롤러에 주입이 되는 방법

  • 의존관계가 실행중에 동적으로 변하는 경우는 거의 없으므로 생성자 주입을 권장한다.

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

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


스프링 빈을 등록하는 두 가지 방법을 배웠는데 아무래도 컴포넌트 스캔을 사용하는 것이 더 간편했다!
하지만 두 가지 방법 모두 알고 있어야 한다고 하셨기 때문에 자바 코드로 직접 등록하는 방법도 꾸준히 사용해볼 것이다 😃
화이팅................🌟

profile
Backend Developer

0개의 댓글