spring中AutoWired(9)

2016/1/16 15:22:02


1.Required

  • @Required註解適用於bean屬性的setter方法
  • 這個註解僅僅表示,受影響的bean屬性必須在配置時被填充,通過在bean定義或通過自動裝配一個明確的屬性值

2.AutoWired

  • 可以將@AutoWired註解爲”傳統”的setter方法
  • 也可以用在構造器或成員變量上
  • 默認情況下,如果因爲找不到合適的bean將會導致autowiring失敗拋出異常,刻意通過在註解上設置required屬性爲false

    @AutoWired(required=false)
    public void setXX(XX xx){
        this.xx = xx
    }
    
  • 每個類只能有一個構造器被標記爲required=true
  • AutoWired的必要屬性,建議使用@Required註解

XML:spring-autowired.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans  xmlns="http://www.springframework.org/schema/beans" 
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
        xmlns:p="http://www.springframework.org/schema/p" 
        xmlns:aop="http://www.springframework.org/schema/aop" 
        xmlns:tx="http://www.springframework.org/schema/tx" 
        xmlns:mvc="http://www.springframework.org/schema/mvc" 
        xmlns:context="http://www.springframework.org/schema/context" 
        xsi:schemaLocation="http://www.springframework.org/schema/beans
                            http://www.springframework.org/schema/beans/spring-beans-3.0.xsd 
                            http://www.springframework.org/schema/context
                            http://www.springframework.org/schema/context/spring-context-3.0.xsd 
                            http://www.springframework.org/schema/aop
                            http://www.springframework.org/schema/aop/spring-aop-3.0.xsd 
                            http://www.springframework.org/schema/tx
                            http://www.springframework.org/schema/tx/spring-tx-3.0.xsd 
                            http://www.springframework.org/schema/mvc
                            http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd"> 

        <context:annotation-config></context:annotation-config>
        <context:component-scan base-package="com.zjx"></context:component-scan>

</beans>

bean:

package com.zjx.dao.impl;

import org.springframework.stereotype.Repository;

import com.zjx.dao.InjectionDao;
@Repository
public class InjectionDaoImpl implements InjectionDao{

    @Override
    public void sava(String data) {
        System.out.println("保存數據:"+data);   
    }
}

package com.zjx.service.impl;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import com.zjx.dao.InjectionDao;
import com.zjx.service.InjectionService;
@Service
public class InjectionServiceImpl implements InjectionService {
    @Autowired
    private InjectionDao injectionDao;

    @Override
    public void save(String data) {
        System.out.println("Service接受參數:"+data);    
        data = data+":"+this.hashCode();
        injectionDao.sava(data);
    }

}

測試:

package com.zjx.interfaces.test;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.BlockJUnit4ClassRunner;

import com.zjx.service.InjectionService;

@RunWith(BlockJUnit4ClassRunner.class)
public class TestAutoWired extends UnitTestBase {
    public TestAutoWired() {
        super("classpath:spring-autowired.xml");
    }
    @Test
    public void testAutoWired(){
        InjectionService service = super.getBean("injectionServiceImpl");
        service.save("this is autowired..");
    }
}

3.@AutoWired和@Resource註解的區別

轉自:http://www.chinasb.org/archives/2011/06/2443.shtml 查看更多實例

Spring不但支持自己定義的@Autowired註解,還支持幾個由JSR-250規範定義的註解,它們分別是@Resource、@PostConstruct以及@PreDestroy。

  • @Resource的作用相當於@Autowired,只不過@Autowired按byType自動注入,而@Resource默認按 byName自動注入罷了。
  • @Resource有兩個屬性是比較重要的,分是name和type,Spring將@Resource註解的name屬性解析爲bean的名字,而type屬性則解析爲bean的類型。
  • 所以如果使用name屬性,則使用byName的自動注入策略,而使用type屬性時則使用byType自動注入策略。
  • 如果既不指定name也不指定type屬性,這時將通過反射機制使用byName自動注入策略。

@Resource裝配順序
  

  1. 如果同時指定了name和type,則從Spring上下文中找到唯一匹配的bean進行裝配,找不到則拋出異常
  2. 如果指定了name,則從上下文中查找名稱(id)匹配的bean進行裝配,找不到則拋出異常
  3. 如果指定了type,則從上下文中找到類型匹配的唯一bean進行裝配,找不到或者找到多個,都會拋出異常
  4. 如果既沒有指定name,又沒有指定type,則自動按照byName方式進行裝配;如果沒有匹配,則回退爲一個原始類型進行匹配,如果匹配則自動裝配;

@Autowired 與@Resource的區別:

  1. @Autowired與@Resource都可以用來裝配bean. 都可以寫在字段上,或寫在setter方法上
  2. @Autowired默認按類型裝配(這個註解是屬於spring的),默認情況下必須要求依賴對象必須存在,如果要允許null值,可以設置它的required屬性爲false,如:@Autowired(required=false) ,如果我們想使用名稱裝配可以結合@Qualifier註解進行使用
  3. @Resource(這個註解屬於J2EE的),默認安裝名稱進行裝配,名稱可以通過name屬性進行指定,如果沒有指定name屬性,當註解寫在字段上時,默認取字段名進行安裝名稱查找,如果註解寫在setter方法上默認取屬性名進行裝配。當找不到與名稱匹配的bean時才按照類型進行裝配。但是需要注意的是,如果name屬性一旦指定,就只會按照名稱進行裝配。
  4. 推薦使用:@Resource註解在字段上,這樣就不用寫setter方法了,並且這個註解是屬於J2EE的,減少了與spring的耦合。這樣代碼看起就比較優雅。

4.其他用途

  • 可以使用@Autowired註解那些衆所周知的解析依賴性接口,比如:BeanFactory,ApplicationContext,Environment,ResourceLoader,ApplicationEventPublisher,MessageSource
  • 可以通過添加註解給需要該類型的數組的字段或方法,以提供ApplicationContext中的所有特定類型的bean
  • 可以用於裝配key爲String的Map
  • 如果希望數組有序,可以讓bean實現org.springframwork.core.Ordered接口或使用@Order註解,但是Order只對數組有效,對map集合無效

5.注意

  • @Autowired是由Spring BeanPostProcessor處理的,所以不能在自己的BeanPostProcessor或BeanFactoryPostProcessor類型應用這些註解,這些類型必須通過XML或者Spring的@Bean註解加載

6.@Autowired裝配操作

XML:

<?xml version="1.0" encoding="UTF-8"?>
<beans  xmlns="http://www.springframework.org/schema/beans" 
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
        xmlns:p="http://www.springframework.org/schema/p" 
        xmlns:aop="http://www.springframework.org/schema/aop" 
        xmlns:tx="http://www.springframework.org/schema/tx" 
        xmlns:mvc="http://www.springframework.org/schema/mvc" 
        xmlns:context="http://www.springframework.org/schema/context" 
        xsi:schemaLocation="http://www.springframework.org/schema/beans
                            http://www.springframework.org/schema/beans/spring-beans-3.0.xsd 
                            http://www.springframework.org/schema/context
                            http://www.springframework.org/schema/context/spring-context-3.0.xsd 
                            http://www.springframework.org/schema/aop
                            http://www.springframework.org/schema/aop/spring-aop-3.0.xsd 
                            http://www.springframework.org/schema/tx
                            http://www.springframework.org/schema/tx/spring-tx-3.0.xsd 
                            http://www.springframework.org/schema/mvc
                            http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd"> 

        <context:annotation-config></context:annotation-config>
        <context:component-scan base-package="com.zjx"></context:component-scan>

</beans>

package com.zjx.multibean;

public interface BeanInterface {

}

package com.zjx.multibean;

import org.springframework.stereotype.Component;

@Component
public class BeanOneImpl implements BeanInterface{

}

package com.zjx.multibean;

import org.springframework.stereotype.Component;

@Component
public class BeanTwoImpl implements BeanInterface{

}

package com.zjx.multibean;

import java.util.List;
import java.util.Map;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

/**
 * @author acer 定義一個包裝類
 * 
 */
@Component
public class BeanInvoker {
    @Autowired
    private List<BeanInterface> list;
    @Autowired
    private Map<String, BeanInterface> map;

    public void say() {
        if (null != list && 0 != list.size()) {
            System.out.println("list...");
            for (BeanInterface bean : list) {
                System.out.println(bean.getClass().getName());
            }
        } else {
            System.out.println("List<BeanInterface> list is null!");
        }
        System.out.println();
        if (null != map && 0 != map.size()) {
            System.out.println("map...");
            for (Map.Entry<String, BeanInterface> entry : map.entrySet()) {
                System.out.println(entry.getKey() + "   "
                        + entry.getValue().getClass().getName());
            }
        } else {
            System.out.println("Map<String,BeanInterface> map is null!");
        }
    }
}

測試:
package com.zjx.interfaces.test;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.BlockJUnit4ClassRunner;

import com.zjx.multibean.BeanInvoker;

@RunWith(BlockJUnit4ClassRunner.class)
public class TestMultiBean extends UnitTestBase{
    public TestMultiBean() {
        super("classpath:spring-autowired.xml");
    }

    @Test
    public void testMultiBean(){
        BeanInvoker invoker = super.getBean("beanInvoker");
        invoker.say();
    }
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章