溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

200行Java代碼如何實現(xiàn)依賴注入框架詳解

發(fā)布時間:2020-10-02 20:39:24 來源:腳本之家 閱讀:157 作者:老錢 欄目:編程語言

依賴注入介紹

先回顧下依賴注入的概念:

我們常提起的依賴注入(Dependency Injection)和控制反轉(zhuǎn)(Inversion of Control)是同一個概念。具體含義是:當(dāng)某個角色(可能是一個Java實例,調(diào)用者)需要另一個角色(另一個Java實例,被調(diào)用者)的協(xié)助時,在 傳統(tǒng)的程序設(shè)計過程中,通常由調(diào)用者來創(chuàng)建被調(diào)用者的實例。但在Spring里,創(chuàng)建被調(diào)用者的工作不再由調(diào)用者來完成,因此稱為控制反轉(zhuǎn);創(chuàng)建被調(diào)用者 實例的工作通常由Spring容器來完成,然后注入調(diào)用者,因此也稱為依賴注入。

其實簡單的說,依賴注入起到的作用就是講對象之間的依賴關(guān)系從原先的代碼中解耦出來,通過配置文件或注解等方式加上Spring框架的處理讓我們對依賴關(guān)系靈活集中的進行管理。

依賴注入框架

依賴注入框架并不神秘,其實它是非常簡單的東西。不要去看spring的依賴注入源碼,因為你只要一去看就意味著你再也寫不敢下手自己擼了,它的功能因為過于強大,所以設(shè)計也過于復(fù)雜,普通程序員一眼看去只能望洋興嘆。

我也并沒有去細致閱讀spring源碼。即便如此也只用了半天的時間便自己擼了一個基本滿足標(biāo)準(zhǔn)依賴注入規(guī)范「JSR-330」的小框架iockids。這個小框架只有一個主類Injector,大約200行代碼,它具備以下功能。

  1. 單例/非單例注入
  2. 構(gòu)造器注入
  3. 字段注入
  4. 循環(huán)依賴注入
  5. Qualifier注入

我們看一個稍微復(fù)雜一點的使用示例

import javax.inject.Inject;
import javax.inject.Named;
import javax.inject.Singleton;
import iockids.Injector;
@Singleton
class Root {
 @Inject
 @Named("a")
 Node a;
 @Inject
 @Named("b")
 Node b;
 @Override
 public String toString() {
  return String.format("root(%s, %s)", a.name(), b.name());
 }
}

interface Node {
 String name();
}

@Singleton
@Named("a")
class NodeA implements Node {
 @Inject
 Leaf leaf;
 @Inject
 @Named("b")
 Node b;
 @Override
 public String name() {
  if (b == null)
   return String.format("nodeA(%s)", leaf);
  else
   return String.format("nodeAWithB(%s)", leaf);
 }
}

@Singleton
@Named("b")
class NodeB implements Node {
 Leaf leaf;
 @Inject
 @Named("a")
 Node a;
 @Inject
 public NodeB(Leaf leaf) {
  this.leaf = leaf;
 }

 @Override
 public String name() {
  if (a == null)
   return String.format("nodeB(%s)", leaf);
  else
   return String.format("nodeBWithA(%s)", leaf);
 }
}

class Leaf {
 @Inject
 Root root;
 int index;
 static int sequence;
 public Leaf() {
  index = sequence++;
 }

 public String toString() {
  if (root == null)
   return "leaf" + index;
  else
   return "leafwithroot" + index;
 }

}

public class Demo {
 public static void main(String[] args) {
  var injector = new Injector();
  injector.registerQualifiedClass(Node.class, NodeA.class);
  injector.registerQualifiedClass(Node.class, NodeB.class);
  var root = injector.getInstance(Root.class);
  System.out.println(root);
 }
}

上面這份代碼用到了iockids提供的所有功能。

  1. Root/NodeA/NodeB類是單例類
  2. Leaf類是非單例類
  3. 它們都使用了字段注入
  4. NodeB使用了構(gòu)造器注入
  5. NodeA和NodeB還使用了Qualifier名稱注入
  6. Leaf類中有Root類型的字段,這便是循環(huán)依賴
  7. NodeA中有NodeB字段,NodeB中有NodeA字段,這也是循環(huán)依賴

為了便于理解上述代碼,我畫了依賴圖

200行Java代碼如何實現(xiàn)依賴注入框架詳解

上面的代碼輸出如下

root(nodeAWithB(leafwithroot0), nodeBWithA(leafwithroot1))

從這個輸出中,我們也可以大致想象出依賴結(jié)構(gòu)。

iockids提供了豐富的注入錯誤異常報告,防止用戶注入配置出錯。

比如我們將上面的NodeA和NodeB的名稱都配置成一樣的a,就會曝出下面的錯誤堆棧

iockids.InjectException: duplicated qualifier javax.inject.Named with the same class iockids.demo.Node
 at iockids.Injector.registerQualifiedClass(Injector.java:87)
 at iockids.Injector.registerQualifiedClass(Injector.java:70)
 at iockids.demo.Demo.main(Demo.java:106)

如果我們將NodeB的構(gòu)造器隨意加一個參數(shù)

 @Inject
 public NodeB(Leaf leaf, int k) {
  this.leaf = leaf;
 }

運行時就會拋出下面的錯誤

iockids.InjectException: no accessible constructor for injection class int
 at iockids.Injector.createNew(Injector.java:120)
 at iockids.Injector.createNew(Injector.java:94)
 at iockids.Injector.createFromParameter(Injector.java:167)
 at iockids.Injector.createFromConstructor(Injector.java:145)
 at iockids.Injector.createNew(Injector.java:123)
 at iockids.Injector.createFromQualified(Injector.java:216)
 at iockids.Injector.createFromField(Injector.java:173)
 at iockids.Injector.injectMembers(Injector.java:233)
 at iockids.Injector.createNew(Injector.java:136)
 at iockids.Injector.createFromQualified(Injector.java:216)
 at iockids.Injector.createFromField(Injector.java:173)
 at iockids.Injector.injectMembers(Injector.java:233)
 at iockids.Injector.createNew(Injector.java:136)
 at iockids.Injector.createNew(Injector.java:94)
 at iockids.Injector.getInstance(Injector.java:245)
 at iockids.demo.Demo.main(Demo.java:107)

項目開源地址:https://github.com/pyloque/iockids (本地下載)

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對億速云的支持。

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI