本文屬代碼GG原創(chuàng),非經(jīng)本人同意,制止轉(zhuǎn)載。
github地址:https://github.com/luxiaoming/dagger2Demo
需要交換,聯(lián)系微信:code_gg_boy
更多精彩,時時關(guān)注微信公眾號code_gg_home
沒有更多開場白,直接說下我對它的理解。
Dagger2 是1個Android依賴注入框架。而android開發(fā)當(dāng)前非常流行的非MVP模式莫屬了,Dagger2的目標(biāo)便是將MVP中的V P 進(jìn)1步解耦,到達(dá)模塊化最大的解耦,使得代碼更容易保護(hù)。
舉個栗子:有個A對象 B對象 和C對象,如果C對象創(chuàng)建需要A和B,那末我們是否是需要構(gòu)造里面?zhèn)魅雲(yún)?shù)A和參數(shù)B,然后在使用的地方以下寫個代碼:
C c=new C(new A(),new B());
如果我們使用了Dagger2時候,我們就不需要管這些了,只需要關(guān)聯(lián)住能提供創(chuàng)建A 和 B的地方 ,然后在需要C的地方寫下:
@Inject
C c;
然后在這個類的初始化地方進(jìn)行注入便可。
我們初步來看,會發(fā)現(xiàn)Dagger2優(yōu)勢不大,沒甚么吸引人的,那末請你靜下心來,看完再得出結(jié)論。
閑話休敘,我們來直接上代碼:(常規(guī)寫法)
public class Test3 {
public Test3() {
}
}
public class MainActivity extends AppCompatActivity {
Test3 test3;
@Override
protected void onCreate(Bundle savedInstanceState) {
//.....
test3 = new Test3();
}
}
使用了注解方式,使得Dagger2能找到它。
public class Test3 {
//這里可以看到加入了注解方式
@Inject
public Test3() {
}
}
@Singleton
//用這個標(biāo)注標(biāo)識是1個連接器
@Component()
public interface MainActivityComponent {
//這個連接器要注入的對象。這個inject標(biāo)注的意思是,我后面的參數(shù)對象里面有標(biāo)注為@Inject的屬性,這個標(biāo)注的屬性是需要這個連接器注入進(jìn)來的。
void inject(MainActivity activity);
}
public class MainActivity extends AppCompatActivity {
//加入注解,標(biāo)注這個test3是需要注入的
@Inject
Test3 test3;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//使用組件進(jìn)行構(gòu)造,注入
DaggerMainActivityComponent.builder().build().inject(this);
}
這是最簡單的1種使用了。首先我們看到,第1印象是我去,這個更復(fù)雜了啊。我只能說確切,由于這個是它對的最基礎(chǔ)的使用,看起來很笨拙,但是當(dāng)它在大型項目里面,在依賴更多的情況下,則會產(chǎn)生質(zhì)的奔騰,會發(fā)現(xiàn)它非常好用,并且將你需要傳遞的參數(shù)都隱藏掉,來實(shí)現(xiàn)解耦。
我先說下Dagger2的注釋思路:關(guān)鍵的點(diǎn)是@Component,這個是個連接器,用來連接提供方和使用方的,所以它是橋梁。它使用在組件里面標(biāo)記使用的Module(標(biāo)記用到了哪一個Module,主要是看使用方需要哪些對象進(jìn)行構(gòu)造,然后將它的提供方@module寫在這里) 然后我們寫入1個void inject(MainActivity activity); 這里后面的參數(shù),就是我們的使用方了。如此1來,我們在使用的地方,使用類似這類方式(DaggerMainActivityComponent.builder().build().inject(this);)的動作,將使用方類里面的標(biāo)記 為@Inject的類初始化掉,完成自動初始化的動作。
結(jié)構(gòu)以下:
為了更好的來學(xué)習(xí)它,我們來順次看看各種使用情況。
直接感受下,如何?
我們來看1個代碼段,當(dāng)我們創(chuàng)建兩個實(shí)例的時候,發(fā)現(xiàn)地址是獨(dú)立的。
如果我們想要1樣的地址呢?加上1句話,具體以下:
效果便是兩個共用實(shí)例啦。
將提供的構(gòu)造,放入@module里面,具體效果以下:
去掉標(biāo)記的@singleton后
效果變成獨(dú)立的啦
有時我們需要依賴1個組件,這個最多見的用法是,我們App實(shí)例里面提供了比如獲得sharepreference的實(shí)例,和比如現(xiàn)在代碼里面的LocationManager的實(shí)例,我們Activity里面需要這些實(shí)例,我們該如何來做呢?看效果:
1:1個AndroidModule 模塊標(biāo)記
這個模塊屬于AndroidcationComponent 組件里面
這里有個關(guān)鍵點(diǎn),就是子組件需要這個里面的某個實(shí)例的時候,這里需要使用1個接口,將需要的實(shí)例做1個返回動作。這里是LocationManager這1行。
我們的子組件的代碼以下:
對應(yīng)的Cmodule代碼以下:
再來看下Test3的代碼當(dāng)前情況:
使用的地方:
仔細(xì)的你會發(fā)現(xiàn)這里多了1個注釋了,@PerActivity,它是個甚么鬼呢?
這里我們看到它是使用了@Scope的1個注釋,這個注釋的意思就是作用域,在作用域內(nèi)保持單例,可以直接理解為單例便可。
為何要新增1個呢,主要是由于各個組件需要獨(dú)立出來,因此如果是依賴關(guān)系,則需要各自在不同的注釋作用域里面。
我們來看下在Cmodule里面,加上@perActivity注釋后的效果:
如果去掉呢?
我們突然發(fā)現(xiàn),它和單例的注釋起的作用1樣啊。so。。。是否是發(fā)現(xiàn)甚么啦。
因此我們得出1個結(jié)論,這里@Singleton
就是1個普通的作用域通道,使用了作用域@Scope注釋的代碼,會變成單例模式。為了驗(yàn)證我們的思路,作以下測試:
我們將之前的@Singleton用新建的這個替換掉,驗(yàn)證兩次的生成代碼,發(fā)現(xiàn)1模1樣,1模1樣,1模1樣,so。。。 就是這個模樣啦。
為何要自定義標(biāo)記呢?這個標(biāo)記不是使用@Scope注釋的哦,是使用@Qualifier 標(biāo)記的,它的目標(biāo)是,為了辨別如果同時返回類型1樣,比如構(gòu)造男孩,女孩的基本屬性,性別和名字時候,獲得男孩和女孩都是1個對象,我們該如何辨別呢,這個就是關(guān)鍵啦。說這么多,真心很煩,直接栗子來啦。
這里稍安勿躁,先來看相同效果的另外一個注釋,@Name,這個是Dagger2自帶的1個讓辨別,效果以下:
這里@Name可以簡單的1個使用方式,就是它不是辨別對象,而是限制使用時候必須加入這個注釋,否則報錯,目的就是讓使用者注意是不是使用正確了。
我們使用自己的注釋再來1遍:
對照兩種方式,我們發(fā)現(xiàn)使用@Name的時候,后面的注釋名字會敲錯,而我們第2種方式呢,則不會耶,so。。。
我們看下自定義的標(biāo)記,作為限制出錯,讓強(qiáng)迫標(biāo)注的例子。
這個出現(xiàn)的目的是為了如果有1個組件,是每次創(chuàng)建實(shí)例提供給他人,而恰好其他組件(有多個)里面有需要它,如果只有1個,我們就用依賴弄定啦。那末它就能夠定義成子組件,誰需要在誰的組件里面加1下,具體看例子:
如上,寫完啦。。
實(shí)戰(zhàn)地方,可以參照 https://github.com/gzsll/TLint 來瀏覽啦,收工。
需要交換,聯(lián)系微信:code_gg_boy
更多精彩,時時關(guān)注微信公眾號code_gg_home
上一篇 為什么清浮動?清浮動的方法是什么?哪一種是終極方法?
下一篇 異步下載照片墻