偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Dagger2 —— 匪夷所思,結(jié)果那么愛(ài)你

移動(dòng)開(kāi)發(fā) Android
今天我們來(lái)介紹這款依賴(lài)注入器 —— Dagger2,源自Square的Dagger,由Google開(kāi)發(fā),基于apt生成靜態(tài)編譯時(shí)的依賴(lài)注入工具,比動(dòng)態(tài)注入的方式更加高性能,但是需要更多的約定。

開(kāi)天辟地

今天我們來(lái)講講一個(gè)有一點(diǎn)點(diǎn)冷門(mén)的庫(kù)Dagger吧。我做一個(gè)不負(fù)責(zé)任的猜測(cè):做客戶端的同學(xué)可能比較少聽(tīng)到一些名詞,比如面向切面編程、控制反轉(zhuǎn)、依賴(lài)注入,相信玩過(guò)Spring的同學(xué)肯定知道這些一開(kāi)始讓人頭大后來(lái)卻很好玩的玩意兒。

今天我們來(lái)介紹這款依賴(lài)注入器 —— Dagger2,源自Square的Dagger,由Google開(kāi)發(fā),基于apt生成靜態(tài)編譯時(shí)的依賴(lài)注入工具,比動(dòng)態(tài)注入的方式更加高性能,但是需要更多的約定。

官網(wǎng):https://google.github.io/dagger/

組成

Dagger2(以下稱(chēng)為Dagger) 主要由兩個(gè)部分組成:Component和Module。分別作為注入器和注入源存在于整個(gè)依賴(lài)圖中,然后有了源和工具,那么只用在我們需要注入的地方加上@Inject注解即可,它是屬于JSR-330的一部分,我們這里就直接引入一個(gè)最簡(jiǎn)單的Demo。

Module

  1. @Module 
  2. public class AppModule { 
  3.     Context mApplicationContext; 
  4.  
  5.     public AppModule(Context context) { 
  6.         this.mApplicationContext = context; 
  7.     } 
  8.  
  9.     @Provides 
  10.     public Context provideContext(){ 
  11.         return mApplicationContext; 
  12.     } 
  13.  
  14.     @Provides 
  15.     public Service provideService(Context context) { 
  16.         return new Service(context, null); 
  17.     } 

這是世界的起源。

  • @Module注解表示這個(gè)類(lèi)是個(gè)Module,是一個(gè)“源”。
  • @Provides注解告訴Dagger我們想要構(gòu)造對(duì)象并提供這些依賴(lài)。

Component

  1. @Component(modules=AppModule.class) 
  2. public interface AppComponent { 
  3.     void inject(App app); 
  4.  
  5.     Context context(); 
  6.     Service service(); 

Component是一個(gè)接口,具體的實(shí)現(xiàn)由Dagger經(jīng)過(guò)apt工具為你生成(是不是有了apt就特別爽),我們給AppComponent這枚“針”指定了藥劑——AppModule,告訴它注入的時(shí)候,從AppModule里面拿到我們要的變量實(shí)例,只要給Component聲明一個(gè)無(wú)返回值,帶被注入類(lèi)型形參的方法,Dagger 就會(huì)為這個(gè)類(lèi)生成一個(gè)MemberInjector對(duì)象,用來(lái)給被注入類(lèi)注入對(duì)象。

被注入對(duì)象

  1. public class App extends Application { 
  2.  
  3.     @Inject Service mService; 
  4.      
  5.     private AppComponent mAppComponent; 
  6.  
  7.     /** 
  8.      * 應(yīng)用程序初始化 
  9.      */ 
  10.     @Override 
  11.     public void onCreate() { 
  12.         super.onCreate(); 
  13.         mAppComponent = DaggerAppComponent.builder() 
  14.                 .appModule(new AppModule(this)) 
  15.                 .build(); 
  16.  
  17.         mAppComponent.inject(this); 
  18.  
  19.     } 
  20.      
  21.     public AppComponent getAppComponent() { 
  22.         return mAppComponent; 
  23.     } 

這里的DaggerAppComponent是Dagger生成的,它的實(shí)現(xiàn)類(lèi)會(huì)在所有Component接口類(lèi)之前增加一個(gè)Dagger前綴,我們只用傳入它所需要的依賴(lài)即可,Component顯然是依賴(lài)Module的,所以需要在這里傳入AppModule,現(xiàn)在,只要用Context的地方,我們都可以拿到這個(gè)AppComponent實(shí)例,只要有這個(gè)實(shí)例,我們可以在任意地方注入被管理的類(lèi)。

作用域

我們說(shuō)依賴(lài)注入的時(shí)候,作用域(Scope)是經(jīng)常會(huì)出現(xiàn)在我們眼里的詞匯??刂谱兞可芷?,實(shí)質(zhì)就是控制它存在的作用域,服務(wù)端典型的作用域如單例(Singleton),Request,Session,等等,它們的變量分別存在于不同的生命周期。

我們默認(rèn)存在的是Singleton,也就是@Singleton注解。由它標(biāo)注的Provider生成的對(duì)象會(huì)被緩存起來(lái),用SingleCheck或者DoubleCheck進(jìn)行包裝。我們Provider指定的作用域需要和Component的作用域一致。

比如Component這樣定義:

  1. @Singleton 
  2. @Component(modules=AppModule.class) 
  3. public interface AppComponent { 
  4.     void inject(App app); 
  5.  
  6.     Service service(); 

而Module就是這個(gè)樣子

  1. @Module 
  2. public class AppModule { 
  3.     Context mApplicationContext; 
  4.  
  5.     public AppModule(Context context) { 
  6.         this.mApplicationContext = context; 
  7.     } 
  8.  
  9.     @Singleton 
  10.     @Provides 
  11.     public Context provideService(){ 
  12.         return new Service(); 
  13.     } 
  14.  

限定符

Dagger還支持使用限定符(Qualifier)來(lái)指定注入的對(duì)象,比如內(nèi)置的@Named限定符,我們?cè)谛枰囟ㄏ薅值淖兞康臅r(shí)候,可以在@Inject上,指定@Named限定符,獲取指定對(duì)象。

  1. //Module 
  2. @Providers 
  3. @Named("cache"
  4. public Service provideService(); 
  5.  
  6. // Injection 
  7. @Inject 
  8. @Named("cache"
  9. Service mService; 

這樣就給這個(gè)mService注入了名字為"cache"的實(shí)例了。

一個(gè)簡(jiǎn)單場(chǎng)景的應(yīng)用

當(dāng)我們談?wù)撘蕾?lài)注入的時(shí)候,我們?cè)谡務(wù)撌裁?

其實(shí)我們是在討論 作用域。

這是什么意思呢,我相信每一個(gè)程序員去實(shí)現(xiàn)一個(gè)單例是一件非常簡(jiǎn)單的事情,makeInstance和getInstance嘛。但是,你們想過(guò)維護(hù)“雙例”嗎?我之前碰到了一個(gè)場(chǎng)景如下:

  1. 用戶(User) 需要一張tag表,會(huì)增刪查改,并和用戶相關(guān)聯(lián)。
  2. 問(wèn)題(Question) 也需要一張tag表,也會(huì)增刪查改,和問(wèn)題關(guān)聯(lián)。
  3. 這兩張表的實(shí)體模型一模一樣。

那么我們采取的方案有兩種,雙表,或者雙庫(kù)。雙表的話,對(duì)ORM很不友好,因?yàn)镺RM是根據(jù)類(lèi)來(lái)確定表的,我們?yōu)榱舜a簡(jiǎn)潔優(yōu)雅,不可能創(chuàng)建兩個(gè)一模一樣的類(lèi),不然取名都變成一件困難的事。這里,一個(gè)優(yōu)勢(shì)是,我使用的ORM庫(kù)首先是維護(hù)一個(gè)單例,單例進(jìn)行CRUD操作,且一個(gè)單例和一個(gè)數(shù)據(jù)庫(kù)相關(guān)。于是我使用Qualifer的特性,生成了兩個(gè)實(shí)例(也就是對(duì)應(yīng)了兩個(gè)數(shù)據(jù)庫(kù)),分別注入到不同的業(yè)務(wù)模型中去,他們就可以使用同一個(gè)類(lèi),而且對(duì)tag的修改完全沒(méi)有影響。這件事要是我們自己去做的話,可能要寫(xiě)很多骯臟不堪的代碼,但是Dagger只用2個(gè)注解就把我的需求解決了。

總結(jié)

好了,本文簡(jiǎn)單的闡述了Dagger入門(mén)使用,總結(jié)起來(lái),我們只要約定好Componenet、Module,搭配Inject使用,即可實(shí)現(xiàn)一個(gè)靜態(tài)的依賴(lài)注入流程。下一次我們?cè)敿?xì)介紹Dagger生成的代碼結(jié)構(gòu)。

責(zé)任編輯:龐桂玉 來(lái)源: segmentfault
相關(guān)推薦

2018-11-20 14:48:54

2016-10-20 19:36:01

androiddagger2依賴(lài)注入

2010-09-17 13:01:44

Python

2009-03-21 15:09:32

Nehalem服務(wù)器Intel

2018-01-31 09:25:39

2012-09-04 09:55:22

代碼抓狂的代碼開(kāi)發(fā)

2020-10-10 09:08:51

數(shù)據(jù)中心

2011-06-03 12:38:05

GeekApp

2009-09-17 09:10:53

阿里馬云

2025-10-17 07:00:00

有線網(wǎng)絡(luò)網(wǎng)絡(luò)弱電

2016-12-28 13:55:16

Android框架MVP

2017-10-16 15:04:32

javaAndroidAPT技術(shù)

2021-03-09 10:12:28

編程技能開(kāi)發(fā)

2018-05-13 15:32:45

IT行業(yè)運(yùn)維Linux

2021-10-08 09:07:09

算法程序技術(shù)

2020-11-02 07:12:27

程序員上級(jí)領(lǐng)導(dǎo)管理

2023-12-25 09:19:00

AI英偉達(dá)研究

2020-01-18 19:28:33

微軟Windows 10瀏覽器

2014-07-08 14:05:48

DaggerAndroid依賴(lài)

2017-09-14 09:40:32

PythonUbuntu信號(hào)機(jī)制
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)