생성자 호출시점에 한 번만 호출되는 것이 보장된다. "불변, 필수" 의존관계에 사용된다.
생성자가 1개만 있으면 @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;
}
}
@Component
public class OrderServiceImpl implements OrderService {
private MemberRepository memberRepository;
private DiscountPolicy discountPolicy;
@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;
}
}
코드가 간결해서 사용하기 편하지만 외부에서 변경이 불가능해서 테스트 하기 힘들다는 치명적인 단점이 있다. DI 프레임워크가 없으면 아무것도 할 수 없다.
public class OrderServiceImpl implements OrderService {
@Autowired private MemberRepository memberRepository;
@Autowired private DiscountPolicy discountPolicy;
}
순수한 자바 코드로 테스트가 불가능하다. NullPointerException이 발생한다. 스프링 부트가 돌아가야 주입이 가능하기 때문이다.
@Test
void fieldInjectionTest(){
OrderServiceImpl orderService = new OrderServiceImpl();
orderService.createOrder(1L, "itemA", 10000);
}
테스트를 가능하게 하기 위해서 setter를 따로 만들어줘야 한다.
@Component
public class OrderServiceImpl implements OrderService {
@Autowired private MemberRepository memberRepository;
@Autowired private DiscountPolicy discountPolicy;
public void setMemberRepository(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
public void setDiscountPolicy(DiscountPolicy discountPolicy) {
this.discountPolicy = discountPolicy;
}
}
새로운 객체를 setter를 통해 넘겨주고 나서야 createOrder() 메서드 테스트가 가능하다.
@Test
void fieldInjectionTest(){
OrderServiceImpl orderService = new OrderServiceImpl();
orderService.setMemberRepository(new MemoryMemberRepository());
orderService.setDiscountPolicy(new FixDiscountPolicy());
orderService.createOrder(1L, "itemA", 10000);
}
사용을 권장하지 않지만 사용해도 되는 경우는 실제 애플리케이션과 관계 없는 테스트 코드에선 사용해도 된다.
한번에 여러 필드를 주입받을 수 있다. 일반적으로 잘 사용하지 않는다.
참고) 당연한 이야기지만 의존관계 자동 주입은 스픵 컨테이너가 관리하는 스프링 빈이어야 동작한다. 스프링 빈이 아닌 'Member'같은 클래스에서 '@Autowired' 코드를 적용해도 아무 기능도 동작하지 않는다.
@Component
public class OrderServiceImpl implements OrderService {
private MemberRepository memberRepository;
private DiscountPolicy discountPolicy;
@Autowired
public void init(MemberRepository memberRepository, DiscountPolicy discountPolicy) {
this.memberRepository = memberRepository;
this.discountPolicy = discountPolicy;
}
}