Spring通過注解@Autowired/@Resource獲取bean實例時為什么可以直接獲取接口而不是注入的類


------------恢復內容開始------------

 

       這個問題困擾了我好久,一直疑問這個接口的bean是怎么注入進去的?因為只看到使用@Service注入了實現類serviceImpl,使用時怎么卻獲取的接口,而且還能調用到實現類的方法,難道這個接口是在什么時候自動注入了進去,且和實現類關聯上了?

接口

public interface TestService {

    public String test();
}

實現類impl

復制代碼
public class TestServiceImpl implements TestService{

    @Override
    public String test() {
        return "TestServiceImpl";
    }
}
復制代碼

Controller的調用:

復制代碼
@RestController
public class TestCtl {

    @Autowired
    private TestService testService;
    
    @RequestMapping("/test")
    public String test() {
        return testService.test();
    }
}
復制代碼

請求結果:

  

       后來才知道,並沒有注入接口的bean,只注入了實現類serviceImpl的bean,接口只是用來接收的,這里就要說到@Autowired/@Resource的注入原理了:@Autowired是Spring的注解,Autowired默認先按byType,如果發現找到多個bean,則,又按照byName方式比對,如果還有多個,則報出異常;@Resource 是JDK1.6支持的注解,默認按照名稱(Byname)進行裝配, 如果沒有指定name屬性,當注解寫在字段上時,默認取字段名,按照名稱查找,如果注解寫在setter方法上默認取屬性名進行裝配。當找不到與名稱匹配的bean時才按照類型進行裝配。但是需要注意的是,如果name屬性一旦指定,就只會按照名稱進行裝配。      

       再來說Controller獲取實例的過程:使用@Autowired,程序在spring的容器中查找類型是TestService的bean,剛好找到有且只有一個此類型的bean,即testServiceImpl,所以就把testServiceImpl自動裝配到了controller的實例testService中,testService其實就是TestServiceImpl實現類;

如果使用的是@Resource,則是先在容器中查找名字為testService的bean,但並沒有找到,因為容器中的bean名字是TestServiceImpl(如果@Service沒指定bean的value屬性,則注入bean的名字就是類名,如果指定了則是指定的名字),然后再通過類型查找TestService類型的bean,找到唯一的了個TestService類型bean(即TestServiceImpl),所以就自動裝配實例成功了。

 

注:

byName 通過參數名 自動裝配,如果一個bean的name 和另外一個bean的 property 相同,就自動裝配。

byType 通過參數的數據類型自動自動裝配,如果一個bean的數據類型和另外一個bean的property屬性的數據類型兼容,就自動裝配

效率上來說@Autowired/@Resource差不多,不過推薦使用@Resource一點,因為當接口有多個實現時@Resource直接就能通過name屬性來指定實現類,而@Autowired還要結合@Qualifier注解來使用,且@Resource是jdk的注釋,可與Spring解耦。

 

如果一個接口有多個實現類時,通過注解獲取實例時怎么知道應該獲取的是哪一個實現類serviceImpl呢?

再增加了一個實現類TestServiceImpl2

復制代碼
@Service
public class TestServiceImpl2 implements TestService{

    @Override
    public String test() {
        return "TestServiceImpl2";
    }
}
復制代碼

多個實現類的話可通過以下2種方式來指定具體要使用哪一個實現:

1、 通過指定bean的名字來明確到底要實例哪一個類

@Autowired 需要結合@Qualifier來使用,如下:

    @Autowired
    @Qualifier("testServiceImpl")
    private TestService testService;

@Resource可直接通過指定name屬性的值即可,不過也可以使用@Qualifier(有點多此一舉了...)

    @Resource(name = "testServiceImpl")
    private TestService testService;    

@Resource如果不顯示的指定name值,就會自動把實例變量的名稱作為name的值的,所以也可以直接這樣寫:

   @Resource
    private TestService testServiceImpl;

2、 通過在實現類上添加@Primary注解來指定默認加載類

復制代碼
@Service
@Primary
public class TestServiceImpl2 implements TestService{

    @Override
    public String test() {
        return "TestServiceImpl2";
    }
}
復制代碼

這樣如果在使用@Autowired/@Resource獲取實例時如果不指定bean的名字,就會默認獲取TestServiceImpl2的bean,如果指定了bean的名字則以指定的為准。

 

       為什么非要調用接口來多此一舉,而不直接調用實現類serviceImpl的bean來得簡單明了呢?

1、 直接獲取實現類serviceImpl的bean也是可以的;

2、 至於加一層接口的原因:一是AOP程序設置思想指導,給別人調用的接口,調用者只想知道方法和功能,而對於這個方法內部邏輯怎么實現的並不關心;二是可以降低各個模塊間的關聯,實現松耦合、程序分層、高擴展性,使程序更加靈活,他除了在規范上有卓越貢獻外,最精髓的是在多態上的運用;繼承只能單一繼承,接口卻可以多實現

3、 當業務邏輯簡單,變更較少,項目自用時,省略掉接口直接使用實現類更簡單明了;反之則推薦使用接口;

------------恢復內容結束------------


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM