java構造器復用方式
構造器復用
創(chuàng)建一個Employee類,屬性有(名字,性別,年齡,職位,薪水),提供3個構造方法,可以初始化(⑴)(名字,性別,年齡,職位,薪水),(2)(名字,性別,年齡)(3)(職位,薪水).要求充分復用構造器Hello.java
public class Hello { public static void main(String[] args) { } } class EMployee{ String name; char gender; int age; String job; double sal; // 職位 薪水 public EMployee(String job,double sal) { this.job = job; this.sal = sal; } // 姓名 性別 年齡 public EMployee(String name,char gender,int age) { this.name = name; this.gender = gender; this.age = age; } // 名字 性別 年齡 職位 薪水 public EMployee(String job,double sal,String name,char gender,int age) { this(name,gender,age); this.job = job; this.sal = sal; } }
推薦使用構造器注入
? Spring框架對Java開發(fā)的重要性不言而喻,平時使用最多的就是其中的IOC,我們通過將組件交由Spring的IOC容器管理,將對象的依賴關系由Spring控制,避免硬編碼所造成的過度程序耦合。
前幾天的時候,朋友問我為什么要使用構造器的注入方式,后面抽時間了解了一下,下面就是筆者要討論的就是其注入方式。
常見的三種注入方式
field注入
@Controller public class FooController { @Autowired //@Inject private FooService fooService; //簡單的使用例子,下同 public List<Foo> listFoo() { return fooService.list(); } }
這種注入方式應該是筆者目前為止開發(fā)中見到的最常見的注入方式。原因很簡單:
- 注入方式非常簡單:加入要注入的字段,附上
@Autowired
,即可完成。 - 使得整體代碼簡潔明了,看起來美觀大方。
構造器注入
@Controller public class FooController { private final FooService fooService; @Autowired public FooController(FooService fooService) { this.fooService = fooService; } //使用方式上同,略 }
setter注入
@Controller public class FooController { private FooService fooService; //使用方式上同,略 @Autowired public void setFooService(FooService fooService) { this.fooService = fooService; } }
構造器注入的好處
- 依賴不可變:其實說的就是final關鍵字,這里不再多解釋了。不明白的園友可以回去看看Java語法。
- 依賴不為空(省去了我們對其檢查):當要實例化FooController的時候,由于自己實現了有參數的構造函數,所以不會調用默認構造函數,那么就需要Spring容器傳入所需要的參數,所以就兩種情況:1、有該類型的參數->傳入,OK 。2:無該類型的參數->報錯。所以保證不會為空,Spring總不至于傳一個null進去吧 ??
- 完全初始化的狀態(tài):這個可以跟上面的依賴不為空結合起來,向構造器傳參之前,要確保注入的內容不為空,那么肯定要調用依賴組件的構造方法完成實例化。而在Java類加載實例化的過程中,構造方法是最后一步(之前如果有父類先初始化父類,然后自己的成員變量,最后才是構造方法,這里不詳細展開。)。所以返回來的都是初始化之后的狀態(tài)。
//承接上面field注入的代碼,假如客戶端代碼使用下面的調用(或者再Junit測試中使用) //這里只是模擬一下,正常來說我們只會暴露接口給客戶端,不會暴露實現。 FooController fooController = new FooController(); fooController.listFoo(); // -> NullPointerException
如果使用field注入,缺點顯而易見,對于IOC容器以外的環(huán)境,除了使用反射來提供它需要的依賴之外,無法復用該實現類。而且將一直是個潛在的隱患,因為你不調用將一直無法發(fā)現NPE的存在。
還值得一提另外一點是:使用field注入可能會導致循環(huán)依賴,即A里面注入B,B里面又注入A:
public class A { @Autowired private B b; } public class B { @Autowired private A a; }
如果使用構造器注入,在spring項目啟動的時候,就會拋出:
BeanCurrentlyInCreationException:Requested bean is currently in creation: Is there an unresolvable circular reference?
從而提醒你避免循環(huán)依賴,如果是field注入的話,啟動的時候不會報錯,在使用那個bean的時候才會報錯。
總結
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Java中的動態(tài)數組和棧Vector Stack使用區(qū)別介紹
這篇文章主要為大家介紹了Java中的動態(tài)數組和棧Vector Stack使用介紹,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-10-10Spring Boot使用Druid進行維度的統計和監(jiān)控
這篇文章主要介紹了Spring Boot使用Druid進行維度的統計和監(jiān)控,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-04-04在IntelliJ IDEA中創(chuàng)建和運行java/scala/spark程序的方法
這篇文章主要介紹了在IntelliJ IDEA中創(chuàng)建和運行java/scala/spark程序的教程,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-05-05SpringBoot整合MyBatis逆向工程及 MyBatis通用Mapper實例詳解
這篇文章主要介紹了SpringBoot整合MyBatis逆向工程及 MyBatis通用Mapper實例詳解 ,需要的朋友可以參考下2017-09-09詳解Spring boot Admin 使用eureka監(jiān)控服務
本篇文章主要介紹了詳解Spring boot Admin 使用eureka監(jiān)控服務,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-12-12