您好,登錄后才能下訂單哦!
原由
移動(dòng)開(kāi)發(fā)中,隨著項(xiàng)目不斷的跌代,需求越來(lái)越復(fù)雜后。項(xiàng)目工程也越來(lái)越龐大。那么此時(shí)的分module的開(kāi)發(fā),則是必然的選擇了。在最終的組件化之路上,不妨把單一工程比如石器時(shí)代,那么接下來(lái)簡(jiǎn)單的拆分工程分多個(gè)moudle開(kāi)來(lái)就是銅器時(shí)代。
銅器時(shí)代之簡(jiǎn)單分module
演進(jìn)
由于從復(fù)雜的單工程拆分了多個(gè)module了,達(dá)到了代碼及資源的初步的隔離,或需求模塊的開(kāi)發(fā)人員,開(kāi)始專注于自己的需求模塊module的開(kāi)發(fā)了。但是隨著部分需求有相關(guān)性,需要相互調(diào)用時(shí)。那么問(wèn)題來(lái)了,在AXXX module中
api project(':BXXX')
而在BXXX module中
api project(':AXXX')
這時(shí)出現(xiàn)了相互依賴,首先編譯器會(huì)能不過(guò),會(huì)出現(xiàn)Circular dependency,循環(huán)相互依賴的問(wèn)題,這是絕不允許的。
為了解決上述的問(wèn)題,將AXXX module與BXXX module需要對(duì)外提供服務(wù)能力支持的,進(jìn)行封裝與抽象。將需要對(duì)外暴露接口/協(xié)議地方,對(duì)其抽象出接口出來(lái)。把些這接口獨(dú)立放在BaseXXXX module中,這樣AXXX module與BXXX module,都分別去
api project(':BaseXXXX')
通過(guò)BaseXXXX中間module通信去解決AXXX module與BXXX module相互依賴調(diào)用通信。
初步的解決方法
為了在BaseXXXX module中,搭建起AXXX module與BXXX module相互通信的橋梁,可以在BaseXXXX module 定義一個(gè)通信標(biāo)識(shí)接口:
/** * * 跨module通訊的 標(biāo)識(shí) interface接口 */ public interface IModuleApi { }
然后主要通過(guò)ModuleApiHelper進(jìn)行通信
public class ModuleApiHelper { private static Map<Class<? extends IModuleApi>,IModuleApi> moduleApiMap = new HashMap<>(); private static Map<Class<? extends IModuleApi>,List<IModuleApi>> moduleApiListMap = new HashMap<>(); /** * 跨module 注冊(cè)進(jìn) IKWModuleApi接口,及實(shí)現(xiàn) * 通常可以在 其它的module中 注冊(cè)此接口的實(shí)現(xiàn),在用的module中g(shù)etModuleApi拿到接口實(shí)現(xiàn) * 這樣,用的module 不是 必須依賴compile其它module了 * @param clazz * @param iModuleApi */ public static void register(Class<? extends IModuleApi> clazz, IModuleApi iModuleApi){ if (null != iModuleApi && null != clazz){ moduleApiMap.put(clazz, iModuleApi); } } public static void unregister(Class<? extends IModuleApi> clazz){ if (moduleApiMap.containsKey(clazz)){ moduleApiMap.remove(clazz); } } public static void register2List(Class<? extends IModuleApi> clazz, IModuleApi iModuleApi){ if (null != iModuleApi && null != clazz){ if (moduleApiListMap.containsKey(clazz)){ List<IModuleApi> iModuleApis = moduleApiListMap.get(clazz); iModuleApis.add(iModuleApi); }else{ List<IModuleApi> iModuleApis = new ArrayList<>(); iModuleApis.add(iModuleApi); moduleApiListMap.put(clazz, iModuleApis); } } } public static void unregister2List(Class<? extends IModuleApi> clazz){ if (moduleApiListMap.containsKey(clazz)){ moduleApiListMap.remove(clazz); } } public static void unregisterAll(Class<? extends IModuleApi> clazz){ unregister(clazz); unregister2List(clazz); } public static <T extends IModuleApi> List<T> getModuleListApi(Class<T> clazz){ if (null != clazz){ if (moduleApiListMap.containsKey(clazz)){ List<IModuleApi> iModuleApis = moduleApiListMap.get(clazz); return (List<T>) iModuleApis; }else{ return null; } }else{ return null; } } /** * 獲取注冊(cè)綁定過(guò)來(lái)的IKWModuleApi 實(shí)現(xiàn) * @param clazz * @param <T> * @return */ public static <T extends IModuleApi> T getModuleApi(Class<T> clazz){ if (null != clazz){ if (moduleApiMap.containsKey(clazz)){ return (T) moduleApiMap.get(clazz); }else{ return null; } }else{ return null; } } }
這樣比如在AXXX module中將原有AServiceData類是如下的:
public class AServiceData { public String getSomeData(){ return "this is some data"; } public void sayHello(){ System.out.println("hello"); } }
改造成
public interface IAServiceData extends IModuleApi { String getSomeData(); void sayHello(); } public class AServiceData implements IAServiceData{ @Override public String getSomeData(){ return "this is some data"; } @Override public void sayHello(){ System.out.println("hello"); } }
將IAServiceData接口定義在BaseXXXX module 中。然后AXXX module中進(jìn)行register相應(yīng)的服務(wù)
public class AModuleService { public void init(){ ModuleApiHelper.register(IAServiceData.class,new AServiceData()); } }
這樣調(diào)用AModuleService的init方法,即可對(duì)IAServiceData服務(wù)進(jìn)行注冊(cè)了。然后即下來(lái),在BXXX module中進(jìn)行g(shù)etXXX得到服務(wù)即可調(diào)用相應(yīng)的方法了.
在任何需要此服務(wù)的方法可如下調(diào)用:
IAServiceData iaServiceData = ModuleApiHelper.getModuleApi(IAServiceData.class);
注意
1> register注冊(cè)時(shí)機(jī),需要越早越好,一般建議在各module的有類似的application的onCreate時(shí)注冊(cè)最好。
2> IModuleApi與ModuleApiHelper,和各extends繼承IModuleApi接口的接口,需要放在中間通信BaseXXX Module中。各需要通信的module去 compile/api BaseXXX Module即可。
問(wèn)題
為了保證IModuleApi接口注冊(cè)有效,需要越早越好進(jìn)行注冊(cè)。這樣隨著項(xiàng)目越來(lái)越復(fù)雜,需要通信的地方越來(lái)越多。統(tǒng)一的ModuleApiHelper,注冊(cè)的地方將越來(lái)越多帶的問(wèn)題也多起來(lái)。
1> 注冊(cè)Map容器占用的內(nèi)存不斷的增多。
2> register注冊(cè)的地方不統(tǒng)一,有些放在各module的類Application的onCreate中,有些可能是放在其它的類中.
3> 不支持ui頁(yè)面的跳轉(zhuǎn),由AXXX module的AxxActtivy頁(yè)面跳轉(zhuǎn)到BXXX module的BxxActivity頁(yè)面中。
4> 不支持多進(jìn)程中應(yīng)用。
為了解決上述問(wèn)題,引入了蒸汽時(shí)代之ARoute到來(lái)。
蒸汽時(shí)代之ARoute
由于遍幅的原因,總體概述不詳細(xì)細(xì)述ARoute,下遍再剖析ARoute??傮w來(lái)說(shuō)在多module通信中解決了:
1> 解決了ui頁(yè)面的跳轉(zhuǎn)問(wèn)題。
2> 根據(jù)需要進(jìn)行register的問(wèn)題,且register通過(guò)靜態(tài)注解來(lái)的,所以register地方統(tǒng)一比如容易維護(hù)。
但是依然不能解決多進(jìn)程中的應(yīng)用。
電器時(shí)代之Andromeda
Andromeda解決了多進(jìn)程,跨進(jìn)程ipc之間的通信過(guò)程,同樣也支持單進(jìn)程的通信...
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。