juuuding

[Section 4] 스프링 빈과 의존관계 본문

Spring/스프링 입문

[Section 4] 스프링 빈과 의존관계

jiuuu 2023. 3. 25. 01:45

* 스프링 빈(Spring Bean)이란? 

 - Spring IOC 컨테이너가 관리하는 자바 객체를 빈(Bean)이라고 부른다. IOC(Inversion Of Control)의 정의와 특징은 다음과 같다.

 

 [IOC의 정의와 특징]

  1) 처음 배우는 자바 프로그램에서는 각 객체들이 프로그램의 흐름을 정하고 각 객체를 직접 생성하고 조작하는 작업(객체를 직접 생성하여 메소드 호출)을 했다. 즉, 모든 작업을 사용자가 제어하는 구조였다. 

   2) 하지만 IOC가 적용된 경우, 객체의 생성을 특별한 관리 위임 주체에 맡긴다. 이 경우 사용자는 객체를 직접 생성하지 않고, 객체의 생명주기를 컨트롤하는 주체는 다른 주체가 된다. 즉, 사용자의 제어권을 다른 주체에게 넘기는 것을 IOC(제어의 역전)이라고 한다.

 

 

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

 

우선 회원 컨트롤러가 회원 서비스와 회원 리포지토리를 사용할 수 있게 의존관계를 준비한다.

 

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

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: Spring container가 관리함

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

- private final MemberService memberservice = new MemberService()도 가능하지만, 위의 코드처럼 스프링 container에 하나 등록하고 쓰는 것이 좋다. 

 

※오류 발생

memberService가 스프링 빈으로 등록되어 있지 않다

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

 

 

스프링 빈을 등록하는 방법은 2가지가 있다.

 첫 번째, 컴포넌트 스캔과 자동 의존관계 설정.

 두 번째, 자바 코드로 직접 스프링 빈 등록하기.

 

 

[컴포넌트 스캔 원리]

- @Component 애노테이션이 있으면 스프링 빈으로 자동 등록된다.

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

- @Component를 포함하는 다음 애노테이션도 스프링 빈으로 자동 등록된다.

   @Controller, @Service, @Repository

 

 

[회원 서비스 스프링 빈 등록]

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

- @Autowired: 생성자에 이것을 사용하면 객체 생성 시점에 스프링 컨테이너에서 해당 스프링 빈을 찾아서 주입한다. 만약 생성자가 1개라면 생략 가능하다.

 

 

[회원 리포지토리 스프링 빈 등록]

@Repository
public class MemoryMemberRepository implements MemberRepository {}

 

스프링 빈 등록 이미지

- memberService와 memberRepository가 스프링 컨테이너에 스프링 빈으로 등록되었다.

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

 

 

 

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

 

package hello.hellospring;

import hello.hellospring.domain.Member;
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
    public MemberService memberService(){
        return new MemberService(memberRepository());
    }

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

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

 

※ 참고사항

- XML로 설정하는 방법도 있지만, 최근에는 잘 사용하지 않는다

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

- 실무에서는 주로 정형화된 컨트롤러, 서비스, 리포지토리 같은 코드는 컴포넌트 스캔을 사용한다. 그리고 정형화 되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록한다. 여기서는 MemoryMemberRepository를 나중에 바꿔야 하기 때문에, 바꿀 때 SpringConfig를 제외한 다른 것은 손대지 않고 바꾸기 위해서 직접 스프링 빈을 등록했다.

'Spring > 스프링 입문' 카테고리의 다른 글

[Section 3] 회원 관리  (0) 2023.03.25
[Section 2] 스프링 웹 개발 기초  (0) 2023.03.24
[Section 1] 라이브러리 & View 환경 설정  (0) 2023.03.22