DINGA DINGA
article thumbnail
728x90

2023.01.22 - [Spring/Study] - [스프링 입문] 3. 회원 관리 예제 - 백엔드 개발

 

[스프링 입문] 3. 회원 관리 예제 - 백엔드 개발

비즈니스 요구사항 정리 데이터: 회원ID, 이름 기능: 회원 등록, 조회 아직 데이터 저장소가 선정되지 않음(가상의 시나리오) 일반적인 웹 애플리케이션 계층 구조 일반적인 웹 애플리케이션은

lgana1021.tistory.com

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

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

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

지금까지 만든 Member 서비스, 리포지토리, 객체 등에 화면을 붙여보자.

 

화면을 붙이려면 Member 컨트롤러와 뷰 템플릿이 필요하다.

Member 컨트롤러는 Member 서비스를 통해 회원가입과 데이터 조회 기능을 수행할 수 있어야 한다.

("Member 컨트롤러가 Member 서비스에 의존한다"라고 표현)

 

java/hello/hellospring/controller/MemberController.java

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

스프링은 @Controller가 있으면 객체(ex. MemberController)를 생성하여 스프링 컨테이너에 넣어 두고 관리한다.

("스프링 컨테이너에서 스프링 빈이 관리된다"고 표현)

 

Section2의 MVC와 템플릿 엔진 설명 때 사용됐던 그림을 보면 이해가 쉽다.

스프링 컨테이너 안에 컨트롤러가 있는 모습이다.

(컨트롤러는 스프링 빈(bean)이라 녹색 콩 모양으로 표현되어 있는 듯하다.)

 

생성자에 @Autowired가 있으면 스프링이 스프링 컨테이너에 있는 관련 객체(MemberService)를 가져와 연결시켜 준다.

이처럼 객체 의존관계를 외부에서 넣어주는 것을 DI(Dependency Injection), 의존성 주입이라 한다.

 

그런데, 저 상태에서 실행을 하면 아래 오류가 발생한다.

Consider defining a bean of type 'hello.hellospring.service.MemberService' in your configuration.

memberService가 스프링 빈으로 등록되어 있지 않기 때문에 이런 오류가 발생하는 것이다.

출처: 인프런 스프링 입문 강의 (김영한님)

helloController는 스프링이 제공하는 컨트롤러여서 스프링 빈으로 자동 등록된다.

이런 오류를 방지하기 위해서는 annotation으로 스프링 빈 등록을 해주어야 한다.

 

회원 서비스 스프링 빈 등록 (MemberService.java)

public class MemberService {

    private final MemberRepository memberRepository;

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

참고: 생성자가 하나뿐이면 @Autowired는 생략할 수 있다.

 

회원 리포지토리 스프링 빈 등록 (MemoryMemberRepository.java)

@Repository
public class MemoryMemberRepository implements MemberRepository { ... }

 

이렇게 어노테이션을 쓰면 아래 이미지처럼 memberController, memberService, memberRepository가 연결되어 스프링 컨테이너에 스프링 빈으로 등록된다.

출처: 인프런 스프링 입문 강의 (김영한님)

스프링은 스프링 컨테이너에 스프링 빈을 등록할 때 싱글톤으로 등록한다. (유일하게 하나만 등록해서 공유)

따라서 같은 스프링 빈이면 같은 인스턴스다.

싱글톤이 아니게 설정할 수 있지만, 특별한 경우를 제외하면 대부분 싱글톤을 사용한다.

 

컴포넌트 스캔 원리
- @Componet 어노테이션이 있으면 스프링 빈으로 자동 등록된다.
- @Controller가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문이다.
- @Component를 포함하는 다음 어노테이션도 스프링 빈으로 자동 등록된다.
    - @Controller
    - @Service
    - @Repository

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

Member Service와 Repository의 annotation(@Service, @Repository, @Autowired)들을 삭제하고 진행한다.

 

service/SpringConfig.java

package hello.hellospring.service;

import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;
import org.springframework.context.annotation.Bean;


@Configuration
public class SpringConfig {

    @Bean
    public MemberService memberService() {
        return new MemberService(memberRepository());
    }

    @Bean
    public MemberRepository memberRepository() {
        return new MemoryMemberRepository();
    }
}

@Bean을 이용해 memberService와 memberRepository를 스프링 빈에 등록하고,

return new MemberService(memberRepository()) 문을 통해 스프링 빈에 등록된 memberRepository를 memberService에 넣어준다.

 

우리가 지금까지 했던 것은 생성자를 통해 memberService가 MemberController에 주입이 되는 방식이다.

이것을 생성자 주입이라고 한다.

 

@Autowired 애노테이션을 이용하는 방식은 필드 주입인데, 이 방식은 중간에 변경할 일이 생겼을 때 번거롭기 때문에 잘 사용하지 않는다.

setter를 이용하는 방식은 setter 주입인데, 이 방식을 사용하려면 setter를 public으로 설정해야 한다는 단점이 있다.

 

DI에는 필드 주입, setter 주입, 생성자 주입 방식이 있다.
의존관계가 실행 중에 동적으로 변하는 경우는 거의(아예) 없으므로 생성자 주입을 권장한다.
 

 

@Autowired를 이용한 DI는 helloController, memberService 등과 같이 스프링이 관리하는 객체에서만 동작한다.

내가 직접 생성한 객체는 스프링 빈으로 등록하지 않으면 동작하지 않는다.

 

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