#Dagger2圖文完全教程
BEGIN
----------
本文屬代碼GG原創,非經本人同意,禁止轉載。
需要交流,聯系微信:code_gg_boy
更多精彩,時時關注微信公眾號code_gg_home

沒有更多開場白,直接說下我對它的理解。
Dagger2 是一個Android依賴注入框架。而android開發當前非常流行的非MVP模式莫屬了,Dagger2的目標便是將MVP中的V P 進一步解耦,達到模塊化最大的解耦,使得代碼更容易維護。
舉個栗子:有個A對象 B對象 和C對象,如果C對象創建需要A和B,那么我們是不是需要構造里面傳入參數A和參數B,然后在使用的地方如下寫個代碼:
```java
C c=new C(new A(),new B());
```
如果我們使用了Dagger2時候,我們就不需要管這些了,只需要關聯住能提供創建A 和 B的地方 ,然后在需要C的地方寫下:
```java
@Inject
C c;
```
然后在這個類的初始化地方進行注入即可。
我們初步來看,會發現Dagger2優勢不大,沒什么吸引人的,那么請你靜下心來,看完再得出結論。
----------
閑話休敘,我們來直接上代碼:(常規寫法)
#1 編寫一個類:
```java
public class Test3 {
public Test3() {
}
}
```
#2 使用的地方
```java
public class MainActivity extends AppCompatActivity {
Test3 test3;
@Override
protected void onCreate(Bundle savedInstanceState) {
//.....
test3 = new Test3();
}
}
```
----------
- 如果我們改為使用Dagger2的方式的話,則可以寫成如下方式:
#1 創建一個類
使用了注解方式,使得Dagger2能找到它。
```java
public class Test3 {
//這里可以看到加入了注解方式
@Inject
public Test3() {
}
}
```
#2 新增一個對象:
```java
@Singleton
//用這個標注標識是一個連接器
@Component()
public interface MainActivityComponent {
//這個連接器要注入的對象。這個inject標注的意思是,我后面的參數對象里面有標注為@Inject的屬性,這個標注的屬性是需要這個連接器注入進來的。
void inject(MainActivity activity);
}
```
#3 調用的地方改為:
```java
public class MainActivity extends AppCompatActivity {
//加入注解,標注這個test3是需要注入的
@Inject
Test3 test3;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//使用組件進行構造,注入
DaggerMainActivityComponent.builder().build().inject(this);
}
```
----------
這是最簡單的一種使用了。首先我們看到,第一印象是我去,這個更復雜了啊。我只能說確實,因為這個是它對的最基礎的使用,看起來很笨拙,但是當它在大型項目里面,在依賴更多的情況下,則會發生質的飛躍,會發現它非常好用,并且將你需要傳遞的參數都隱藏掉,來實現解耦。
----------
我先說下Dagger2的注釋思路:關鍵的點是@Component,這個是個連接器,用來連接提供方和使用方的,所以它是橋梁。它使用在組件里面標記使用的Module(標記用到了哪個Module,主要是看使用方需要哪些對象進行構造,然后將它的提供方@module寫在這里) 然后我們寫入一個void inject(MainActivity activity); 這里后面的參數,就是我們的使用方了。如此一來,我們在使用的地方,使用類似這種方式(DaggerMainActivityComponent.builder().build().inject(this);)的動作,將使用方類里面的標記 為@Inject的類初始化掉,完成自動初始化的動作。
結構如下:

為了更好的來學習它,我們來依次看看各種使用情況。
#1 常規使用方法



直接感受下,如何?
#2 帶一個參數的效果
----------

----------

----------

----------
我們來看一個代碼段,當我們創建兩個實例的時候,發現地址是獨立的。

如果我們想要一樣的地址呢?加上一句話,具體如下:

效果便是兩個共用實例啦。

#3 換種經常使用的方式
將提供的構造,放入@module里面,具體效果如下:

去掉標記的@singleton后

效果變成獨立的啦

#4 依賴一個組件的時候
有時我們需要依賴一個組件,這個最常見的用法是,我們App實例里面提供了比如獲取sharepreference的實例,和比如現在代碼里面的LocationManager的實例,我們Activity里面需要這些實例,我們該如何來做呢?看效果:
1:一個AndroidModule 模塊標記

這個模塊屬于AndroidcationComponent 組件里面

這里有個關鍵點,就是子組件需要這個里面的某個實例的時候,這里需要使用一個接口,將需要的實例做一個返回動作。這里是LocationManager這一行。
我們的子組件的代碼如下:

對應的Cmodule代碼如下:

再來看下Test3的代碼當前情況:

使用的地方:

----------
細心的你會發現這里多了一個注釋了,@PerActivity,它是個什么鬼呢?

這里我們看到它是使用了@Scope的一個注釋,這個注釋的意思就是作用域,在作用域內保持單例,可以直接理解為單例即可。
為什么要新增一個呢,主要是因為各個組件需要獨立出來,因此如果是依賴關系,則需要各自在不同的注釋作用域里面。
我們來看下在Cmodule里面,加上@perActivity注釋后的效果:


如果去掉呢?

我們突然發現,它和單例的注釋起的作用一樣啊。so。。。是不是發現什么啦。

因此我們得出一個結論,這里@Singleton
就是一個普通的作用域通道,使用了作用域@Scope注釋的代碼,會變成單例模式。為了驗證我們的思路,作如下測試:


我們將之前的@Singleton用新建的這個替換掉,驗證兩次的生成代碼,發現一模一樣,一模一樣,一模一樣,so。。。 就是這個樣子啦。
#5 自定義一個標記
為什么要自定義標記呢?這個標記不是使用@Scope注釋的哦,是使用@Qualifier 標記的,它的目標是,為了區分如果同時返回類型一樣,比如構造男孩,女孩的基本屬性,性別和名字時候,獲取男孩和女孩都是一個對象,我們該如何區分呢,這個就是關鍵啦。說這么多,真心很煩,直接栗子來啦。
這里稍安勿躁,先來看相同效果的另一個注釋,@Name,這個是Dagger2自帶的一個讓區分,效果如下:




這里@Name可以簡單的一個使用方式,就是它不是區分對象,而是限制使用時候必須加入這個注釋,否則報錯,目的就是讓使用者注意是否使用正確了。


我們使用自己的注釋再來一遍:





對比兩種方式,我們發現使用@Name的時候,后面的注釋名字會敲錯,而我們第二種方式呢,則不會耶,so。。。
我們看下自定義的標記,作為限制出錯,讓強制標注的例子。




#6 子組件(公共組件)
這個出現的目的是為了如果有一個組件,是每次創建實例提供給別人,而恰好其他組件(有多個)里面有需要它,如果只有一個,我們就用依賴搞定啦。那么它就可以定義成子組件,誰需要在誰的組件里面加一下,具體看例子:



#7 懶加載方式


#8 多個綁定方式
####1 第一種方式








####2 第二種方式



這里需要注意的就是,在組件里面加入多個綁定的時候,module的里面必須要有一個是@IntoSet 這個作為第一個標記,否則會出錯,可以多個@IntoSet標記。
如果是列表類型的,則使用@ElementsIntoSet就ok了。

#9 終極boss ,Map方式





再加一點,生成的代碼位置在\dagger2Demo\app\build\generated\source\apt\debug\com\xm\dagger2demo,可以直接看生成代碼,更好理解
如上,寫完啦。。
[回到開始地方](#begin)
實戰地方,可以參照 https://github.com/gzsll/TLint 來閱讀啦,收工。
需要交流,聯系微信:code_gg_boy
更多精彩,時時關注微信公眾號code_gg_home

- 0-發現
- AndroidInterview-Q-A
- Android能讓你少走彎路的干貨整理
- LearningNotes
- temp
- temp11
- 部分地址
- 0-待辦任務
- 待補充列表
- 0-未分類
- AndroidView事件分發與滑動沖突處理
- Spannable
- 事件分發機制詳解
- 1-Java
- 1-Java-01基礎
- 未歸檔
- 你應該知道的JDK知識
- 集合框架
- 1-Java-04合集
- Java之旅0
- Java之旅
- JAVA之旅01
- JAVA之旅02
- JAVA之旅03
- JAVA之旅04
- JAVA之旅05
- JAVA之旅06
- JAVA之旅07
- JAVA之旅08
- JAVA之旅09
- java之旅1
- JAVA之旅10
- JAVA之旅11
- JAVA之旅12
- JAVA之旅13
- JAVA之旅14
- JAVA之旅15
- JAVA之旅16
- JAVA之旅17
- JAVA之旅18
- JAVA之旅19
- java之旅2
- JAVA之旅20
- JAVA之旅21
- JAVA之旅22
- JAVA之旅23
- JAVA之旅24
- JAVA之旅25
- JAVA之旅26
- JAVA之旅27
- JAVA之旅28
- JAVA之旅29
- java之旅3
- JAVA之旅30
- JAVA之旅31
- JAVA之旅32
- JAVA之旅33
- JAVA之旅34
- JAVA之旅35
- 1-Java-05辨析
- HashMapArrayMap
- Java8新特性
- Java8接口默認方法
- 圖解HashMap(1)
- 圖解HashMap(2)
- 2-Android
- 2-Android-1-基礎
- View繪制流程
- 事件分發
- AndroidView的事件分發機制和滑動沖突解決
- 自定義View基礎
- 1-安卓自定義View基礎-坐標系
- 2-安卓自定義View基礎-角度弧度
- 3-安卓自定義View基礎-顏色
- 自定義View進階
- 1-安卓自定義View進階-分類和流程
- 10-安卓自定義View進階-Matrix詳解
- 11-安卓自定義View進階-MatrixCamera
- 12-安卓自定義View進階-事件分發機制原理
- 13-安卓自定義View進階-事件分發機制詳解
- 14-安卓自定義View進階-MotionEvent詳解
- 15-安卓自定義View進階-特殊形狀控件事件處理方案
- 16-安卓自定義View進階-多點觸控詳解
- 17-安卓自定義View進階-手勢檢測GestureDetector
- 2-安卓自定義View進階-繪制基本圖形
- 3-安卓自定義View進階-畫布操作
- 4-安卓自定義View進階-圖片文字
- 5-安卓自定義View進階-Path基本操作
- 6-安卓自定義View進階-貝塞爾曲線
- 7-安卓自定義View進階-Path完結篇偽
- 8-安卓自定義View進階-Path玩出花樣PathMeasure
- 9-安卓自定義View進階-Matrix原理
- 通用類介紹
- Application
- 2-Android-2-使用
- 2-Android-02控件
- ViewGroup
- ConstraintLayout
- CoordinatorLayout
- 2-Android-03三方使用
- Dagger2
- Dagger2圖文完全教程
- Dagger2最清晰的使用教程
- Dagger2讓你愛不釋手-終結篇
- Dagger2讓你愛不釋手-重點概念講解、融合篇
- dagger2讓你愛不釋手:基礎依賴注入框架篇
- 閱讀筆記
- Glide
- Google推薦的圖片加載庫Glide:最新版使用指南(含新特性)
- rxjava
- 這可能是最好的RxJava2.x入門教程完結版
- 這可能是最好的RxJava2.x入門教程(一)
- 這可能是最好的RxJava2.x入門教程(三)
- 這可能是最好的RxJava2.x入門教程(二)
- 這可能是最好的RxJava2.x入門教程(五)
- 這可能是最好的RxJava2.x入門教程(四)
- 2-Android-3-優化
- 優化概況
- 各種優化
- Android端秒開優化
- apk大小優化
- 內存分析
- 混淆
- 2-Android-4-工具
- adb命令
- 一鍵分析Android的BugReport
- 版本控制
- git
- git章節簡述
- 2-Android-5-源碼
- HandlerThread 源碼分析
- IntentService的使用和源碼分析
- 2-Android-9-辨析
- LRU算法
- 什么是Bitmap
- 常見圖片壓縮方式
- 3-Kotlin
- Kotlin使用筆記1-草稿
- Kotlin使用筆記2
- kotlin特性草稿
- Kotlin草稿-Delegation
- Kotlin草稿-Field
- Kotlin草稿-object
- 4-JavaScript
- 5-Python
- 6-Other
- Git
- Gradle
- Android中ProGuard配置和總結
- gradle使用筆記
- Nexus私服搭建
- 編譯提速最佳實踐
- 7-設計模式與架構
- 組件化
- 組件化探索(OKR)
- 1-參考列表
- 2-1-組件化概述
- 2-2-gradle配置
- 2-3-代碼編寫
- 2-4-常見問題
- 2-9-值得一讀
- 8-數據結構與算法
- 0臨時文件
- 漢諾塔
- 8-數據-1數據結構
- HashMap
- HashMap、Hashtable、HashSet 和 ConcurrentHashMap 的比較
- 遲到一年HashMap解讀
- 8-數據-2算法
- 1個就夠了
- Java常用排序算法(必須掌握的8大排序算法)
- 常用排序算法總結(性能+代碼)
- 必須知道的八大種排序算法(java實現)
- 9-職業
- 閱讀
- 書單
- 面試
- 面試-01-java
- Java面試題全集駱昊(上)
- Java面試題全集駱昊(下)
- Java面試題全集駱昊(中)
- 面試-02-android
- 40道Android面試題
- 面試-03-開源源碼
- Android圖片加載框架最全解析(二),從源碼的角度理解Glide的執行流程
- 面試-07-設計模式
- 面試-08-算法
- 面試-09-其他
- SUMMARY
- 版權說明
- temp111