생성자 주입
수정자(setter)주입
필드주입
일반 메서드 주입
생성자를 통해서 의존관계를 주입받는 방법
🔹특징
🔸생성자 호출 시점에 딱 1번만 호출되는 것이 보장된다.
🔸불변이거나 필수적인 의존관계에 사용한다.
🔸 생성자가 1개만 있을 경우에는 @Autowired 생략 가능하다.
🔹 예시
@Component public class OrderServiceImpl implements OrderService{ private final MemberRepository memberRepository; private final DiscountPolicy discountPolicy; @Autowired public OrderServiceImpl(MemberRespository memberRepository, DiscountPolicy discountPolicy){ this.memberRepository = memberRepository; this.discountPolicy = discountPolicy; }
필드값을 변경하는 수정자(setter)를 통해서 의존관계를 주입하는 방법
🔹특징
🔸선택, 변경 가능성이 있는 의존관계에 사용
🔸자바빈 프로퍼티 규약의 setter방식을 사용하는 방법
🔹예시
@Component 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; } }
필드에 바로 주입하는 방법
🔹특징
🔸코드가 간결하다 하지만 외부에서 변경이 불가능하기 때문에 테스트하기 힘들다.
🔸DI프레임워크가 없으면 아무것도 할 수 없다.
🔸실제 애플리케이션 구동과 관련없는 테스트코드나 @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; @Autowired public void init(MemberRepository memberRepository, DiscountPolicy discountPolicy) { this.memberRepository = memberRepository; this.discountPolicy = discountPolicy; } }
🔸자동의존관계 주입은 스프링 컨테이너가 관리하는 빈이여야지만 동작한다.
🔸 과거에는 수정자 주입과 필드 주입을 많이 사용했지만, 최근에는 생성자 주입을 권장한다.
🔸 생성자 주입을 사용하되, 가끔 옵션이 필요하다면 수정자 주입을 선택해서 사용하자.
🔹 왜냐하면
🔸대부분의 의존관계는 한번 일어나면 애플리케이션이 종료될 때까지 변경할 일이 없다.(아니, 변하면 안된다.)
🔸 수정자 주입을 사용하면 setter를 public으로 열어둬야 한다. ➡️ 누군가 데이터를 변경할 수 있다.
🔸 생성자 주입을 사용하면 주입해야할 데이터를 누락했을 때 컴파일 오류가 난다.
🔸 생성자 주입을 사용하면 final키워드를 사용할 수 있다. ➡️ 값이 설정되지 않는 오류를 애초 컴파일 시점에서 막을 수 있다.
주입할 스프링 빈이 없어도 동작해야할 경우가 있다. 그럴 경우, 아래의 방법들을 사용하면 된다.
🔸 @Autowired(required = false) : 자동 주입할 대상이 없으면 수정자 메서드 자체가 호출이 안된다.
@Autowired(required = false) public void setNoBean1(Member member) {
🔸 org.springframework.lang.@Nullable : 자동 주입할 대상이 없으면 Null이 입력된다.
@Autowired public void setNoBean2(@Nullable Member member) {
🔸 Optional<> : 자동 주입할 대상이 없으면 Optional.empty가 입력된다.
@Autowired public void setNoBean3(Optional<Member> member) {