의존관계 자동 주입(1)

다양한 의존관계 주입 방법

방법

  1. 생성자 주입
  2. 수정자 주입(setter 주입)
  3. 필드 주입
  4. 일반 메서드 주입

1. 생성자 주입

  • 특징
    1. 생성자 호출 시점에 딱 한 번만 호출되는 것이 보장
    2. “불변, 필수” 의존관계에서 사용
    3. 한 번 생성되면 불변
  • 예시
      public OrderServiceImpl(MemberRepository memberRepository, DiscountPolicy discountPolicy) {
              this.memberRepository = memberRepository;
              this.discountPolicy = discountPolicy;
          }
    
  • 생성자가 딱 1개 있으면 @Autowired가 생략되도 자동 주입 된다.

2.수정자 주입

  • setter를 사용하여 의존관계 주입
  • 특징
    1. 선택, 변경 가능성이 있는 의존 관계에서 사용
    2. 자바빈 프로퍼티 규약의 수정자 메서드 방식을 사용하는 방법
      • 자바빈 프로퍼티 규약: 과거의 필드 값을 직접 변경하지 않고 setter, getter라는 메서드를 통해 값을 읽거나 수정하는 규칙
  • 예시

      @Autowired
      public void setMemberRepository(MemberRepository memberRepository) {
          System.out.println("memberRepository = " + memberRepository);
          this.memberRepository = memberRepository;
      }
    
      @Autowired
      public void setDiscountPolicy(DiscountPolicy discountPolicy) {
          System.out.println("discountPolicy = " + discountPolicy);
          this.discountPolicy = discountPolicy;
      }
    


3. 필드 주입

  • 필드에 바로 주입하는 방식
  • 특징
    1. 코드가 간결해지지만 외부에서 변경이 불가능해 테스트 하기 힘들다
    2. DI 프레임워크가 없다면 사용 불가능
    3. 사용하지 않는 것이 좋다
  • 예시

      @Autowired private MemberRepository memberRepository;
      @Autowired private DiscountPolicy discountPolicy;
    


4. 일반 메서드 주입

  • 일반 메서드를 통해 주입하는 방식
  • 특징
    1. 한번에 여러 필드를 주입 받을 수 있다.
    2. 일반적으로 잘 사용하지 않는다.
    3. 아무 메서드에나 @Autowired를 통해 의존관계 주입 가능

옵션 처리

  • 주입할 스프링 빈이 없어도 동작해야 할 경우가 있다.
  • 만약 @Autowired만 사용하면 자동 주입 대상이 없을 경우 오류가 발생한다.
  • 자동 주입 대상을 옵션으로 처리하는 방법
    1. @Autowired(required=false): 자동 주입할 대상이 없으면 수정자 메서드 자체가 호출되지 않음
    2. org.springframework.lang.@Nullable: 자동 주입할 대상이 없으면 null이 입력된다.
    3. Optional<>: 자동 주입할 대상이 없으면 Optional.empty가 입력된다.

생성자 주입

  • 최근 스프링을 포함한 DI 프레임워크 대부분이 생성자 주입을 권장한다. 불변
    1. 대부분 의존관계 주입은 한번 주입되면 종료시점까지 의존관계를 변경할 일이 없다.
    2. 수정자 주입 사용 시 setter메서드가 필요하다
    3. 실수로 의존관계를 변경할 수 있으므로 수정자 주입은 좋은 방식이 아니다.

누락

  • 생성자 주입 시 주입 데이터가 누락된경우 컴파일 오류가 발생
  • 하지만 수정자 주입은 데이터가 누락되더라도 오류가 발생하지 않음

final 키워드

  • 생성자에서 값 설정이 되지 않은 경우 오류를 컴파일 시점에 막아준다.

롬복과 최신 트랜드

  • 생성자를 만들고 주입 받은 값을 대입하는 코드도 만들어야 하는게 귀찮다
  • getter, setter를 자동으로 만들어준다. -RequiredArgsConstructor를 사용할 경우 final 필드를 모아 생성자를 자동으로 만들어준다.

@Autowired 필드명

조회할 빈이 2개 이상인 경우

  • 스프링 빈을 수동 등록하여 문제 해결이 가능하지만 의존 관계 자동 주입에서 해결할 수 있는 방법이 존재한다.
  • 해결 방법

1. @Autowired 필드 명 매칭

  • @Autowired는 타입 매칭을 시도하고 이 때 여러 빈이 있으면 필드 이름, 파라미터 이름으로 빈 이름을 추가 매칭한다.
  • 타입 매칭 결과가 2개 이상일 경우 필드 명, 파라미터 명으로 빈 이름을 매칭할 수 있다.

2. @Qualifier

  • 주입 시 @Qualifier(“~~”)를 입력하는 방식
  • 하지만 @Qualifier는 @Qualifier를 찾는 용도로만 쓰는 것이 좋다.

3. @Primary

  • 우선 순위를 정하는 방식
  • @Autowired 사용 시 @Primary가 우선권을 갖는다.

우선 순위

  • Primary는 기본값처럼 동작하는 것이고 Quilifier을 상세하게 동작
  • 스프링은 자동보다는 수동, 넓은 범위의 선택권보다는 좁은 범위의 선택권이 우선 순위가 높다.
  • 따라서 Qulifier의 우선 순위가 높다.
[출처]

homebdy
homebdy 개발에 이제 막 발 담근 사람. 개발에 이제 막 발 담근 사람. 개발에 이제 막 발 담근 사람. 개발에 이제 막 발 담근 사람.
comments powered by Disqus