Android組件化、插件化詳細(xì)講解
什么是組件化(通俗易懂)
通俗易懂來(lái)講就是,拆成多個(gè)module開(kāi)發(fā)就是組件化。
App的部分功能模塊在打包時(shí)并不以傳統(tǒng)?式打包進(jìn)apk?件中,?是以另?種形式?次封裝進(jìn)apk內(nèi)部,或者放在?絡(luò)上適時(shí)下載,在需要的時(shí)候動(dòng)態(tài)對(duì)這些功能模塊進(jìn)?加載,稱(chēng)之為插件化。這些單獨(dú)?次封裝的功能模塊apk,就稱(chēng)作插件,初始安裝的apk稱(chēng)作宿主。插件化是組件化的更進(jìn)?步推進(jìn)。
插件化基礎(chǔ)之反射:
反射的寫(xiě)法
try { Class utilClass = Class.forName("com.hencoder.demo.hidden.Util"); Constructor utilConstructor = utilClass.getDeclaredConstructors()[0]; utilConstructor.setAccessible(true); Object util = utilConstructor.newInstance(); Method shoutMethod = utilClass.getDeclaredMethod("shout"); shoutMethod.setAccessible(true); shoutMethod.invoke(util); } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (NoSuchMethodException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InstantiationException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); }
反射的?的
Java既然提供了可?性關(guān)鍵字public、private等等,?來(lái)限制代碼之間的可?性,為什么?要提供反射功能?可?性特性的?持不是為了代碼不被壞?使?,?是為了程序開(kāi)發(fā)的簡(jiǎn)潔性。安全性的話,可?性的?持提供的是Safety 的安全,?不是Security的安全。即,可?性的?持讓程序更不容易寫(xiě)出bug,?不是更不容易被??侵。反射的?持可以讓開(kāi)發(fā)者在可?性的例外場(chǎng)景中,可以突破可?性限制來(lái)調(diào)???需要的API。這是基于對(duì)開(kāi)發(fā)者在使?反射時(shí)已經(jīng)?夠了解和謹(jǐn)慎的假設(shè)的。所以,可?性的?持不是為了防御外來(lái)者?侵,因此反射功能的?持并沒(méi)有什么不合理。
關(guān)于DEX:
- class:java編譯后的?件,每個(gè)類(lèi)對(duì)應(yīng)?個(gè)class?件
- dex:Dalvik EXecutable把class打包在?起,?個(gè)dex可以包含多個(gè)class?件
- odex:Optimized DEX針對(duì)系統(tǒng)的優(yōu)化,例如某個(gè)?法的調(diào)?指令,會(huì)把虛擬的調(diào)?轉(zhuǎn)換為使?具體的index,這樣在執(zhí)?的時(shí)候就不?再查找了
- oat:Optimized Androidfile Type。使?AOT策略對(duì)dex預(yù)先編譯(解釋?zhuān)┏杀镜刂噶?,這樣再運(yùn)?階段就不需再經(jīng)歷?次解釋過(guò)程,程序的運(yùn)?可以更快
- AOT:Ahead-Of-Time compilation預(yù)先編譯
插件化原理:動(dòng)態(tài)加載
通過(guò)?定義ClassLoader來(lái)加載新的dex?件,從?讓程序員原本沒(méi)有的類(lèi)可以被使?,這就是插件化的原理。
例如:把Utils拆到單獨(dú)的項(xiàng)?,打包apk作為插件引?:
File f = new File(getCacheDir() + "/demo-debug.apk"); if (!f.exists()) { try { InputStream is = getAssets().open("apk/demo-debug.apk"); int size = is.available(); byte[] buffer = new byte[size]; is.read(buffer); is.close(); FileOutputStream fos = new FileOutputStream(f); fos.write(buffer); fos.close(); } catch (Exception e) { throw new RuntimeException(e); } } DexClassLoader classLoader = new DexClassLoader(f.getPath(), getCodeCacheDir().getPath(), null, null); try { Class oldClass = classLoader.loadClass("com.hencoder.demo.hidden.Util"); Constructor utilConstructor = oldClass.getDeclaredConstructors()[0]; utilConstructor.setAccessible(true); Object util = utilConstructor.newInstance(); Method shoutMethod = oldClass.getDeclaredMethod("shout"); shoutMethod.setAccessible(true); shoutMethod.invoke(util); Class activityClass = classLoader.loadClass("com.hencoder.demo.MainActivity"); startActivity(new Intent(this, activityClass)); } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (NoSuchMethodException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InstantiationException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); }
問(wèn)題?:未注冊(cè)的組件(例如Activity)不能打開(kāi)
- 解決?式?:代理Activity
- 解決?式?:欺騙系統(tǒng)
- 解決?式三:重寫(xiě)gradle打包過(guò)程,合并AndroiManifest.xml
問(wèn)題?:資源?件?法加載
解決?式:?定義AssetManager和Resources對(duì)象
private AssetManager createAssetManager (String dexPath) { try { AssetManager assetManager = AssetManager.class.newInstance(); Method addAssetPath = assetManager.getClass().getMethod("addAssetPath", String.class); addAssetPath.invoke(assetManager, dexPath); return assetManager; } catch (Exception e) { e.printStackTrace(); return null; } }
private Resources createResources(AssetManager assetManager) { Resources superRes = mContext.getResources(); Resources resources = new Resources(assetManager, superRes.getDisplayMetrics(), superRes.getConfiguration()); return resources; }
插件化有什么用?
- 早期:解決dex 65535問(wèn)題。?歌后來(lái)也出了multidex?具來(lái)專(zhuān)?解決
- 懶加載來(lái)減少軟件啟動(dòng)速度:有可能,實(shí)質(zhì)上未必會(huì)快
- 減?安裝包??:可以
- 項(xiàng)?結(jié)構(gòu)拆分,依賴(lài)完全隔離,?便多團(tuán)隊(duì)開(kāi)發(fā)和測(cè)試,解決了組件化耦合度太?的問(wèn)題:這個(gè)使?模塊化就夠了,況且模塊化解耦不夠的話,插件化也解決不了這個(gè)問(wèn)題
- 動(dòng)態(tài)部署:可以
- 熱修復(fù):可以
到此這篇關(guān)于Android組件化、插件化詳細(xì)講解的文章就介紹到這了,更多相關(guān)Android組件化,插件化內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android中Service實(shí)時(shí)向Activity傳遞數(shù)據(jù)實(shí)例分析
這篇文章主要介紹了Android中Service實(shí)時(shí)向Activity傳遞數(shù)據(jù)的方法,實(shí)例分析了Service組件基于線程操作實(shí)現(xiàn)數(shù)值實(shí)時(shí)傳遞的相關(guān)技巧,需要的朋友可以參考下2015-09-09Android仿IOS ViewPager滑動(dòng)進(jìn)度條
這篇文章主要為大家詳細(xì)介紹了Android仿IOS ViewPager滑動(dòng)進(jìn)度條的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01Android中兩個(gè)類(lèi)讓你再也不用實(shí)現(xiàn)onActivityResult()
這篇文章主要給大家介紹了關(guān)于Android中兩個(gè)類(lèi)讓你再也不用實(shí)現(xiàn)onActivityResult()的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)各位Android開(kāi)發(fā)者們具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起看看吧2018-08-08Android 關(guān)于“NetworkOnMainThreadException”問(wèn)題的原因分析及解決辦法
這篇文章主要介紹了Android 關(guān)于“NetworkOnMainThreadException”的相關(guān)知識(shí),本文介紹的非常詳細(xì),具有參考借鑒價(jià)值,感興趣的朋友一起學(xué)習(xí)吧2016-02-02Android viewpager自動(dòng)輪播和小圓點(diǎn)聯(lián)動(dòng)效果
這篇文章主要為大家詳細(xì)介紹了Android viewpager自動(dòng)輪播和小圓點(diǎn)聯(lián)動(dòng)效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-10-10Android 應(yīng)用中插入廣告詳解及簡(jiǎn)單實(shí)例
這篇文章主要介紹了Android 應(yīng)用中插入廣告詳解及簡(jiǎn)單實(shí)例的相關(guān)資料,需要的朋友可以參考下2016-10-10Android開(kāi)發(fā)之App widget用法實(shí)例分析
這篇文章主要介紹了Android開(kāi)發(fā)之App widget用法,結(jié)合實(shí)例形式詳細(xì)分析了Android開(kāi)發(fā)中使用App widget組件的具體步驟與相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06