모르는게 많은 개발자

[Spring] @Autowired @Qualifier 개념/예제 본문

스프링

[Spring] @Autowired @Qualifier 개념/예제

Awdsd 2020. 4. 4. 23:18
반응형

저번 포스팅에서  XML을 이용해 bean에 DI를 주입하는 방법을 해봤다.

이번에는 Annotation @Autowired와 @Qualifier을 이용해 DI를 주입하는 것을 써보려한다.

 

1. @Autowired

@Autowired는 기존에 XML에 <property>, <constructor-arg>를 통해  DI를 주입해오던 방식을 자동으로 해주는 Annotation이다.

    <bean id="exam" class="spring.di.entity.NewlecExam">
        <constructor-arg name="kor" value="30"/>
        <constructor-arg name="eng" value="20"/>
        <constructor-arg name="com" value="10"/>
        <constructor-arg name="math" value="0"/> 
    </bean>
    
    //기존에는 <property> 태그를 이용해 exam 객체를 주입했다.
    <bean id="console" class="spring.di.ui.InlineExamConsole" >
        <property name="exam" ref="exam" />
    </bean>

위의 방식에서 <property>속성 DI역할을 @Autowired를 활용하여 DI를 주입 할 수 있다.

public class InlineExamConsole implements ExamConsole {

    private Exam exam;		
	
    public InlineExamConsole() {
		
    }
    public InlineExamConsole(Exam exam) {		
        this.exam = exam;
    }

    @Override
    public void print() {
        System.out.printf("total is %d, avg is %f\n", exam.total(), exam.avg());
    }
	
    //@Autowired를 이용해 Exam객체를 자동으로 주입해줄 수 있다.
    @Autowired
    @Override
    public void setExam(Exam exam) {
        // TODO Auto-generated method stub
        this.exam = exam;
    }
}

하지만 @Autowired를 사용하기 위해서는 XML파일에 <context:annotation-config />을 추가해야 한다.

    <context:annotation-config />        //@Autowired를 탐색하기 위해 추가
    <bean id="exam" class="spring.di.entity.NewlecExam">
        <constructor-arg name="kor" value="30"/>
        <constructor-arg name="eng" value="20"/>
        <constructor-arg name="com" value="10"/>
        <constructor-arg name="math" value="0"/> 
    </bean>
    
    //기존에는 <property> 태그를 이용해 exam 객체를 주입했다.
 

 


그런데 만약 Exam bean이 여러개라면?

    <context:annotation-config />        //@Autowired를 탐색하기 위해 추가
    <bean id="exam" class="spring.di.entity.NewlecExam">
        <constructor-arg name="kor" value="30"/>
        <constructor-arg name="eng" value="20"/>
        <constructor-arg name="com" value="10"/>
        <constructor-arg name="math" value="0"/> 
    </bean>
    
    <bean id="exam1" class="spring.di.entity.NewlecExam">
        <constructor-arg name="kor" value="30"/>
        <constructor-arg name="eng" value="20"/>
        <constructor-arg name="com" value="10"/>
        <constructor-arg name="math" value="0"/> 
    </bean>
    
    //기존에는 <property> 태그를 이용해 exam 객체를 주입했다.

이런 경우에는 setter의 인자 이름을 비교하여 인자와 같은 id인 bean을 주입하게 된다.

하지만 이것을 인자가 아닌 직접 지정해 줄 수 있는 Annotation이 @Qualifier이다.

 

2. @Qualifier

방금 말한 것처럼 같은 클래스의 bean이 여러개일 경우 @Autowried DI가 애매하다. 이 문제는 @Qualifier Annotation에 id를 입력함으로써 주입 될 bean을 지정함으로써 해결할 수 있다. (XML은 위와 동일)

    @Autowired
    @Qualifier("exam")		//XML bean에서 id가 exam인 bean을 주입하라는 뜻
    @Override
    public void setExam(Exam exam) {
        // TODO Auto-generated method stub
        this.exam = exam;
    }

또, 위의 코드는 생성자가 생성되고 setExam을 호출하여 exam빈을 넣어주는 순서로 실행된다. 만약 생성자가 실행되는 과정에서 DI주입을 하고 싶으면 필드위에 @Autowired를 써주면 된다.

public class InlineExamConsole implements ExamConsole {
    @Autowired
    @Qualifier("exam")
    private Exam exam;		
	
    public InlineExamConsole() {
		
    }
    public InlineExamConsole(Exam exam) {		
        this.exam = exam;
    }

    @Override
    public void print() {
        System.out.printf("total is %d, avg is %f\n", exam.total(), exam.avg());
    }
	
    @Override
    public void setExam(Exam exam) {
        // TODO Auto-generated method stub
        this.exam = exam;
    }
}

3. required

만약, exam이라는 bean이 존재하지 않을 때 @Qualifier("exam")을 하면 어떻게 될까? 당연히 에러가 날 것이다.

그럼 bean이 존재하지 않으면 그냥 null값으로 되는 방법은 없을까? required 옵션을 사용하면 된다.

	@Autowired(required = false)
	@Qualifier("exam")
	@Override
	public void setExam(Exam exam) {
		// TODO Auto-generated method stub
		this.exam = exam;
	}

@Autowired에 위의 코드처럼 required = false를 넣게 되면 @Qualifier의 bean이 존재하지 않으면 null값이 들어가게끔 처리할 수 있다.

 

 

 

참고

https://www.youtube.com/watch?v=Od-WtriilwY&list=PLq8wAnVUcTFUHYMzoV2RoFoY2HDTKru3T&index=13

반응형
Comments