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

Bummy·2023년 4월 28일
0
post-thumbnail

어노테이션을 활용하여 스프링 빈을 등록할 수도 있지만 자바 코드로 직접 스프링 빈을 등록할 수도 있다.

회원 서비스와 회원 리포지토리의 @Service, @Repository, @Autowired 어노테이션을 제거하고 진행한다.

hello > hellospring > SpringConfig 파일을 하나 생성해준다 .

SpringConfig.java

package hello.hellospring;

import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;
import hello.hellospring.sevice.MemberService;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Conditional;
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을 보고 @Bean을 스프링 빈에 등록해준다.

참고

  • XML으로 설정하는 방식도 있지만 최근에는 잘 사용하지 않으므로 생략함
  • DI에는 필드 주입, setter 주입, 생성자 주입 이렇게 3가지 방식이 있다.
  • 의존관계가 실행 중에 동적으로 변하는 경우는 거의 없으므로 생성자 주입 방식을 권장한다.
  • 실무에서는 주로 정형화된 컨트롤러, 서비스. 리포지토리 같은 코드는 컴포넌트 스캔을 사용한다.
  • 또한 정형화되자 않거나, 상황에 따라 구현 클래스를 변경해야 하면 설정을 통해 스프링 빈으로 등록한다.

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


자바 코드로 직접 Spring Bean을 등록하는 이유

클래스가 정형화 되지 않거나, 상황에 따라 구현 클래스를 변경해야 하면 자바 코드로 직접 Spring Bea을 등록한다.

이 예제에서는 아직 DB가 선정되지 않았다는 가상의 시나리오가 있다. 하지만 일단 개발을 진행해야 하기 때문에 interface로 MemberRepository를 작성하고 구현 클래스로 MemoryMemberRepository를 작성한 상황이다. 나중에 DB가 선정되면 MemoryMemberRepository를 그 DB를 사용한 구현 클래스로 바꿔치기 할 것 이다.

  • 이렇게 Spring Bean을 자바 코드로 직접 등록했다면 변경하는 작업을 기존의 동작하는 코드에 전혀 손을 대지 않고 SpringConfig.java만 수정하는 것으로 바꿔치기가 가능하다.

DI의 3가지 방식

  1. Field Injection
  2. Setter Injection
  3. Constructor Injection

1. Field Injection

@Controller
public class MemberController
{
    @Autowired private MemberService memberService;
}
  • Spring이 올라갈 때 Spring Bean이 연결되는 작업 외에 다른 작업을 할 수 없어 좋지 못한 방식이다.

2. Setter Injection

@Controller
public class MemberController
{
    private MemberService memberService;

    @Autowired
    public void setMemberService(MemberService memberService)
    {
        this.memberService = memberService;
    }
}
  • 객체 생성이 된 후에 setter가 호출되어 DI가 일어난다.
  • 의존 관계가 실행 중에 동적으로 변하는 경우는 거의 없기 때문에 setter가 맨 처음에 호출된 이후로 다시 호출될 일이 없다.
  • 그런데도 setter가 public으로 열려 있기에 개발자에 의해 잘못 호출될 수도 있어 좋은 방식은 아니다.

3. Constructor Injection

@Controller
public class MemberController
{
    private MemberService memberService;

    @Autowired
    public MemberController(MemberService memberService)
    {
        this.memberService = memberService;
    }
}
  • 최근 가장 권장되는 방식이며, Setter Injection에서의 문제를 해결하면, Field Injection의 단점이 보완된다.

0개의 댓글