Skip to content

조회 빈이 2개이상 - 해결 #35

@mingsound21

Description

@mingsound21

조회 빈이 2개이상 - 해결

1. @Autowired 필드 명 매칭

@Autowired 는 타입 매칭을 시도하고, 이때 여러 빈이 있으면 필드 이름, 파라미터 이름으로 빈 이름을 추가
매칭한다.

아래는 필드 명을 빈 이름으로 변경한 것이다.

@Autowired
  private DiscountPolicy rateDiscountPolicy

@Autowired 매칭 정리

  1. 타입 매칭

  2. 타입 매칭의 결과가 2개 이상일 때 필드 명, 파라미터 명으로 빈 이름 매칭

2. @qualifier @qualifier끼리 매칭 빈 이름 매칭

@qualifier 는 추가 구분자를 붙여주는 방법이다. 주입시 추가적인 방법을 제공하는 것이지 빈 이름을 변경하는 것은 아니다.

빈 등록시 @qualifier를 붙여 준다.

@Component
  @Qualifier("mainDiscountPolicy")
  public class RateDiscountPolicy implements DiscountPolicy {}

주입시에 @qualifier를 붙여주고 등록한 이름을 적어준다.

생성자 자동 주입시

@Autowired
  public OrderServiceImpl(MemberRepository memberRepository,
                          @Qualifier("mainDiscountPolicy") DiscountPolicy
  discountPolicy) {
      this.memberRepository = memberRepository;
      this.discountPolicy = discountPolicy;
}

수정자 자동 주입시

@Autowired
  public DiscountPolicy setDiscountPolicy(@Qualifier("mainDiscountPolicy")
  DiscountPolicy discountPolicy) {
      this.discountPolicy = discountPolicy;
  }

@qualifier 정리1. @qualifier끼리 매칭
2. 빈 이름 매칭
3. NoSuchBeanDefinitionException 예외 발생

3. @primary 사용

@primary 는 우선순위를 정하는 방법이다. @Autowired 시에 여러 빈이 매칭되면 @primary 가 우선권을 가진다.

사용법은 간단하다, 같은 타입중에서 우선적으로 주입하게 하고 싶은 것에 @primary 어노테이션을 붙이면된다.

아래는 RateDiscountPolicy가 우선권을 가지도록 설정한 것이다.

@Component
  @Primary
  public class RateDiscountPolicy implements DiscountPolicy {}

  @Component
  public class FixDiscountPolicy implements DiscountPolicy {}

여기까지 보면 @primary@qualifier 중에 어떤 것을 사용하면 좋을지 고민이 될 것이다.
@qualifier 의 단점은 주입 받을 때 다음과 같이 모든 코드에 @qualifier 를 붙여주어야 한다는 점이다.

@primary, @qualifier 활용

코드에서 자주 사용하는(90%) 메인 데이터베이스의 커넥션을 획득하는 스프링 빈이 있고, 코드에서 특별한
기능으로 가끔 사용하는(10%) 서브 데이터베이스의 커넥션을 획득하는 스프링 빈이 있다고 생각해보자. 메인
데이터베이스의 커넥션을 획득하는 스프링 빈은 @primary 를 적용해서 조회하는 곳에서 @qualifier
지정 없이 편리하게 조회하고, 서브 데이터베이스 커넥션 빈을 획득할 때는 @qualifier 를 지정해서
명시적으로 획득 하는 방식으로 사용하면 코드를 깔끔하게 유지할 수 있다. 물론 이때 메인 데이터베이스의
스프링 빈을 등록할 때 @qualifier 를 지정해주는 것은 상관없다.

@qualifier@primary사이의 우선순위

@primary 는 기본값 처럼 동작하는 것이고, @qualifier 는 매우 상세하게 동작한다. 이런 경우 어떤 것이
우선권을 가져갈까? 스프링은 자동보다는 수동이, 넒은 범위의 선택권 보다는 좁은 범위의 선택권이 우선
순위가 높다. 따라서 여기서도 @qualifier 가 우선권이 높다.

Metadata

Metadata

Assignees

No one assigned

    Labels

    No labels
    No labels

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions