[스프링 - 핵심원리] 의존관계 자동 주입

의존관계 주입 방법

의존관계 주입의 방법은 다음과 같다.

  • 생성자 주입
  • 수정자 주입 (setter 주입)
  • 필드 주입
  • 일반 메서드 주입

본 게시글에서 총 4개의 의존관계 주입 방법에 대하여 먼저 다루도록 하고, 최근 트랜드에서의 의존관계 스타일을 설명한다.



생성자 주입

  • 생성자를 통해서 의존 관계를 주입받는 방법이다.

  • 이전 Spring 핵심 원리 관련 게시글들에서 진행했던 방법들이 바로 생성자 주입이다.
  • 특징
    • 생성자 호출시점에 딱 1번만 호출되는 것이 보장된다.
    • 불변 , 필수 의존관계에 사용한다.
      • 불변 : 딱 한번만 호출되기 때문에 더이상 수정이 불가능하다.
      • 필수 : 반드시 값이 있어야 한다. ( final )
    • 생성자가 딱 1개만 있으면 @Autowired 를 생략해도 자동 주입된다!


예시 코드 - @Autowired

@Component
public class OrderServiceImpl implements OrderService {

	private final MemberRepository memberRepository;
	private final DiscountPolicy discountPolicy;

	//생성자 주입
	@Autowired
	public OrderServiceImpl(MemberRepository memberRepository, DiscountPolicy discountPolicy) {
		this.memberRepository = memberRepository;
		this.discountPolicy = discountPolicy;
	}

}


예시 코드 - @Autowired 생략

@Component
public class OrderServiceImpl implements OrderService {

	private final MemberRepository memberRepository;
	private final DiscountPolicy discountPolicy;

	//생성자 주입
	//@Autowired 가 생성자에 자동으로 적용된다.
	public OrderServiceImpl(MemberRepository memberRepository, DiscountPolicy discountPolicy) {
		this.memberRepository = memberRepository;
		this.discountPolicy = discountPolicy;
	}

}



수정자 주입(setter 주입)

  • setter라 불리는 필드의 값을 변경하는 수정자 메서드를 통해서 의존관계를 주입하는 방법이다.
  • 특징
    • 선택 , 변경 가능성이 있는 의존관계에 사용한다.
    • 자바빈 프로퍼티 규약의 수정자 메서드 방식을 사용하는 방법이다.

    자바빈 프로퍼티 규약 이란 setXxx, getXxx 라는 메서드를 통해서 필드의 값을 읽거나 수정하는 것이다.


예시 코드

@Component
public class OrderServiceImpl implements OrderService {

	private MemberRepository memberRepository;
	private DiscountPolicy discountPolicy;

	//setter로 멤버변수에 의존관계 주입
	@Autowired
	public void setMemberRepository(MemberRepository memberRepository) {
		this.memberRepository = memberRepository;
	}

	//setter로 멤버변수에 의존관계 주입
	@Autowired
	public void setDiscountPolicy(DiscountPolicy discountPolicy) {
		this.discountPolicy = discountPolicy;
	}

}



필드 주입

  • 필드에 바로 주입하는 방법
  • 특징
    • 외부에서 변경이 불가능하여 테스트하기 힘들다.
    • DI 프레임워크가 없으면 아무것도 할 수 없다.

      new 연산자로 (순수한 자바로) 객체 생성 및 테스트가 불가능하다.

  • 사용하지 않는 것을 권장한다.
    • Only ‘스프링 설정을 목적으로 하는 @Configuration 같은 곳에서만 특별한 용도로 사용


예시 코드

@Component
public class OrderServiceImpl implements OrderService {

	//필드에 바로 주입
	@Autowired
	private MemberRepository memberRepository;

	//필드에 바로 주입
	@Autowired	
	private DiscountPolicy discountPolicy;

}



일반 메서드 주입

  • 일반 메서드를 통해서 주입 받을 수 있다.

  • 특징

    • 한번에 여러 필드를 주입 받을 수 있다.
    • 일반적으로 잘 사용하지 않는다.


예시 코드

@Component
public class OrderServiceImpl implements OrderService {

	private MemberRepository memberRepository;
	private DiscountPolicy discountPolicy;

	// init(..) 은 일반 메서드이다.
	@Autowired
	public void init(MemberRepository memberRepository, DiscountPolicy discountPolicy) {
		this.memberRepository = memberRepository;
		this.discountPolicy = discountPolicy;
	}

}




옵션 처리

옵션의 종류

  • @Autowired( required=false )
    • 자동 주입할 대상이 없으면 (주입할 빈이 없으면) 수정자 메서드 자체가 호출이 안된다.
  • org.springframework.lang.@Nullable
    • 자동 주입할 대상이 없으면 null 이 입력된다.
  • Optional<>
    • 자동 주입할 대상이 없으면 Optional.empty 가 입력된다.



예시 코드

Member 빈 객체가 등록이 안되어있는 상태라고 가정한 뒤, 예시 코드를 통해 좀더 자세히 알아본다.


@Autowired( required=false )

//메서드 자체가 호출이 안된다.
@Autowired(required = false)
public void setNoBean1(Member member) {
    System.out.println("setNoBean1 = " + member);
}


org.springframework.lang.@Nullable

//null이 매개변수 member에 주입된다.
@Autowired
public void setNoBean2(@Nullable Member member) {
	System.out.println("setNoBean2 = " + member);
}


Optional.empty

//Optional.empty이 주입된다.
@Autowired(required = false)
public void setNoBean3(Optional<Member> member) {
	System.out.println("setNoBean3 = " + member);
}




의존관계 주입 스타일의 트랜드

권장되는 의존관계 주입 방식

  • 과거에는 수정자 주입, 필드 주입을 많이 사용했다.
  • 하지만, 최근에는 스프링을 포함한 DI 프레임워크 대부분이 생성자 주입을 권장 한다.
  • 그 이유는 아래와 같다.


생성자 주입을 권장하는 이유

  • 불변
    • 대부분의 의존관계 주입은 애플리케이션 종료 전까지 변하면 안된다. (불변해야 한다.)

    • 수정자 주입 사용시, setXxx 메서드를 public 으로 열어두어야 한다.
    • 따라서, 누군가 임의로 변경할 가능성이 있다.
  • 누락
    • 프레임워크 없이 순수한 자바 코드로만 단위 테스트 하는 경우를 대비해보자.

    • 아래와 같이 수정자 의존관계인 경우

        public class OrderServiceImpl implements OrderService {
        	private MemberRepository memberRepository;
        	private DiscountPolicy discountPolicy;
      
        	@Autowired
        	public void setMemberRepository(MemberRepository memberRepository) {
        		this.memberRepository = memberRepository;
        	}
      
        	@Autowired
        	public void setDiscountPolicy(DiscountPolicy discountPolicy) {
        		this.discountPolicy = discountPolicy;
        	}
              	
        	//...
        }
      
    • 지금은 프레임워크 없이 순수한 자바 코드로만 단위 테스트를 수행 하고 있다. 아래 테스트 코드를 실행해보자.

        @Test
        void createOrder() {
        	OrderServiceImpl orderService = new OrderServiceImpl();
        	orderService.createOrder(1L, "itemA", 10000);
        }
      
    • 실행은 되지만, NullPointException 이 발생한다. 왜냐하면, memberRepositorydiscountPolicy 모두 의존관계 주입이 누락 되었기 때문이다. (스프링이 없으므로)
    • 하지만, 생성자 주입을 사용한다면 데이터를 누락했을 때 컴파일 오류가 발생한다.
      • 따라서, 문제를 캐치하기가 용이하다.
    • OrderServiceImpl 를 다음과 같이 수정해보자.

        public class OrderServiceImpl implements OrderService {
        	private final MemberRepository memberRepository; //final 추가
        	private final DiscountPolicy discountPolicy; //final 추가
      
        	//생성자 주입
        	@Autowired
        	public OrderServiceImpl(MemberRepository memberRepository, DiscountPolicy discountPolicy) {
        		this.memberRepository = memberRepository;
        		this.discountPolicy = discountPolicy;
        	}
              	
        	//...
        }
      
    • 위 코드와 같이 수정할 경우 final 덕분에 생성자에 의존관계 주입이 되지 않는다면 (순수한 자바코드로 테스트 한다면) 컴파일 오류가 발생한다.



롬복과 최신 트랜드

롬복 라이브러리를 추가하는 방법은 따로 검색하여 적용해보자. 본 게시글에서는 추가방법에 대해 다루지 않는다.

@RequiredArgsConstructor 애너테이션

  • 롬복 라이브러리에서 제공하는 @RequiredArgsConstructor 애너테이션을 추가하면 final이 붙은 필드를 모아서 생성자를 자동으로 만들어준다.

  • 만약 생성자가 단 한개라면, @Autowired 가 자동으로 붙는다는 점을 기억하자!
    • 따라서, @RequiredArgsConstructor 애너테이션이 만들어준 생성자를 제외하고 다른 생성자가 없다면 @Autowired 가 자동으로 붙는다.
  • 따라서 @RequiredArgsConstructor 를 적용하면 다음과 같다.
    • 기존 코드

        @Component
        public class OrderServiceImpl implements OrderService {
      
        	private final MemberRepository memberRepository;
        	private final DiscountPolicy discountPolicy;
      
        	//@Autowired 가 생략된 생성자
        	public OrderServiceImpl(MemberRepository memberRepository, DiscountPolicy discountPolicy) {
        		this.memberRepository = memberRepository;
        		this.discountPolicy = discountPolicy;
        	}
      
        }
      
    • @RequiredArgsConstructor 적용 코드

        @Component
        @RequiredArgsConstructor //새로 추가된 롬복 애너테이션
        public class OrderServiceImpl implements OrderService {
      
        	private final MemberRepository memberRepository;
        	private final DiscountPolicy discountPolicy;
      
        	// @Autowired 는 생성자가 한개라면 자동으로 붙는다.
        	// @RequiredArgsConstructor 가 아래 생성자를 자동으로 생성해준다.
        	/*
        	public OrderServiceImpl(MemberRepository memberRepository, DiscountPolicy discountPolicy) {
        		this.memberRepository = memberRepository;
        		this.discountPolicy = discountPolicy;
        	}
        	*/
      
        }
      
    • 최종 코드

        @Component
        @RequiredArgsConstructor
        public class OrderServiceImpl implements OrderService {
      
        	private final MemberRepository memberRepository;
        	private final DiscountPolicy discountPolicy;
      
        }
      
      • 훨씬 간결해졌다.
    • 이와 같이 최근에는 롬복 라이브러리를 활용하여 의존관계를 설정하는 것을 선호한다.




  • 본 게시글은 김영한님의 강의를 토대로 정리한 글입니다.
  • 더 자세한 내용을 알고 싶으신 분들이 계신다면, 해당 강의를 수강하시는 것을 추천드립니다.