国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Android 插件化處理方案詳解

瀏覽:49日期:2022-09-19 18:15:08
插件化啟動(dòng)Activity的過(guò)程

Android 插件化處理方案詳解

在宿主里面的AndroidManifest.xml里面注冊(cè)一個(gè)空的activity

從開(kāi)始執(zhí)行execStartActivity到最終將Activity對(duì)象new出來(lái)這個(gè)過(guò)程,系統(tǒng)層會(huì)去校驗(yàn)需要啟動(dòng)的activity的合法性[就是是否有在某個(gè)應(yīng)用的AndroidManifest.xml里面注冊(cè)]以及按啟動(dòng)要求創(chuàng)建activity對(duì)象。清晰了這點(diǎn)我們就可以很好的繞過(guò)系統(tǒng)的約束,達(dá)到我們的目的:【插件中的組件擁有真正生命周期,完全交由系統(tǒng)管理、非反射代理】。 簡(jiǎn)單來(lái)說(shuō)方案就兩步: Step1、在開(kāi)始startActivity的時(shí)候?qū)⑿枰獑?dòng)的插件組件替換成宿主預(yù)先聲明號(hào)的。

public ActivityResult execStartActivity(Context who, IBinder contextThread, IBinder token, Activity target, Intent intent, int requestCode, Bundle options) { //如果啟動(dòng)的是插件的activity組件,這里面將會(huì)被替換成宿主預(yù)先聲明的 PluginIntentResolver.resolveActivity(intent); return hackInstrumentation.execStartActivity(who, contextThread, token, target, intent, requestCode, ptions);}

Step2、在最終創(chuàng)建activity對(duì)象的時(shí)候改回成插件組件的。

@Overridepublic Activity newActivity(ClassLoader cl, String className, Intent intent) throws InstantiationException, IllegalAccessException, ClassNotFoundException { ClassLoader orignalCl = cl; String orginalClassName = className; String orignalIntent = intent.toString(); if (ProcessUtil.isPluginProcess()) { // 將PluginStubActivity替換成插件中的activity if (PluginManagerHelper.isStub(className)) { String action = intent.getAction(); if (action != null && action.contains(PluginIntentResolver.CLASS_SEPARATOR)) {String[] targetClassName = action.split(PluginIntentResolver.CLASS_SEPARATOR);String pluginClassName = targetClassName[0];final String pid = intent.getStringExtra(PluginIntentResolver.INTENT_EXTRA_PID).trim();PluginDescriptor pluginDescriptor = TextUtils.isEmpty(pid) ? PluginManagerHelper.getPluginDescriptorByClassName(pluginClassName) : PluginManagerHelper.getPluginDescriptorByPluginId(pid);Class<?> clazz = PluginLoader.loadPluginClassByName(pluginDescriptor, pluginClassName);if (clazz != null) { className = pluginClassName; cl = clazz.getClassLoader(); intent.setExtrasClassLoader(cl); if (targetClassName.length > 1) { // 之前為了傳遞classNae,intent的action被修改過(guò) // 這里再把Action還原到原始的Action intent.setAction(targetClassName[1]); } else { intent.setAction(null); } // 添加一個(gè)標(biāo)記符 intent.addCategory(RELAUNCH_FLAG + className);} else { throw new ClassNotFoundException('pluginClassName : ' + pluginClassName, new Throwable());} } else if (PluginManagerHelper.isExact(className, PluginDescriptor.ACTIVITY)) {// 這個(gè)邏輯是為了支持外部app喚起配置了stub_exact的插件ActivityPluginDescriptor pluginDescriptor = PluginManagerHelper.getPluginDescriptorByClassName(className);if (pluginDescriptor != null) { boolean isRunning = PluginLauncher.instance().isRunning(pluginDescriptor.getPackageName()); if (!isRunning) { return waitForLoading(pluginDescriptor); }}Class<?> clazz = PluginLoader.loadPluginClassByName(pluginDescriptor, className);if (clazz != null) { cl = clazz.getClassLoader();} else { throw new ClassNotFoundException('className : ' + className, new Throwable());} } else {// 進(jìn)入這個(gè)分支可能是因?yàn)閍ctivity重啟了,比如橫豎屏切換,由于上面的分支已經(jīng)把Action還原到原始到Action了// 這里只能通過(guò)之前添加的標(biāo)記符來(lái)查找classNameboolean found = false;Set<String> category = intent.getCategories();if (category != null) { Iterator<String> itr = category.iterator(); while (itr.hasNext()) { String cate = itr.next(); if (cate.startsWith(RELAUNCH_FLAG)) { className = cate.replace(RELAUNCH_FLAG, ''); PluginDescriptor pluginDescriptor = PluginManagerHelper.getPluginDescriptorByClassName(className); if (pluginDescriptor != null) {boolean isRunning = PluginLauncher.instance().isRunning( pluginDescriptor.getPackageName());if (!isRunning) { return waitForLoading(pluginDescriptor);} } Class<?> clazz = PluginLoader.loadPluginClassByName(pluginDescriptor, className); cl = clazz.getClassLoader(); found = true; break; } }}if (!found) { throw new ClassNotFoundException( 'className : ' + className + ', intent : ' + intent.toString(), new Throwable());} } } else { if (cl instanceof PluginClassLoader) {PluginIntentResolver.resolveActivity(intent); } else {// Do Nothing } } } try { Activity activity = super.newActivity(cl, className, intent); if (activity instanceof PluginContainer) { ((PluginContainer) activity).setPluginId(intent.getStringExtra(PluginContainer.FRAGMENT_PLUGIN_ID)); } return activity; } catch (ClassNotFoundException e) { // 收集狀態(tài),便于異常分析 throw new ClassNotFoundException(' orignalCl : ' + orignalCl.toString() + ', orginalClassName : '+ orginalClassName + ', orignalIntent : ' + orignalIntent + ', currentCl : ' + cl.toString()+ ', currentClassName : ' + className + ', currentIntent : ' + intent.toString() + ', process : '+ ProcessUtil.isPluginProcess() + ', isStubActivity : '+ PluginManagerHelper.isStub(orginalClassName) + ', isExact : '+ PluginManagerHelper.isExact(orginalClassName, PluginDescriptor.ACTIVITY), e); }}

方案確實(shí)很簡(jiǎn)單,不過(guò)還有一些收尾工作,就是將創(chuàng)建好的[插件]組件進(jìn)行一些必要的init操作,比如:在聲明周期onCreate之前進(jìn)行上下文替換等操作,這些都在插件框架提供的PluginInstrumentionWrapper里面進(jìn)行完成的,看一下代碼片段:

@Overridepublic void callActivityOnCreate(Activity activity, Bundle icicle) { PluginInjector.injectActivityContext(activity); Intent intent = activity.getIntent(); if (intent != null) { intent.setExtrasClassLoader(activity.getClassLoader()); } if (icicle != null) { icicle.setClassLoader(activity.getClassLoader()); } if (ProcessUtil.isPluginProcess()) { installPluginViewFactory(activity); if (activity instanceof WaitForLoadingPluginActivity) { // NOTHING } else { } if (activity.isChild()) { // 修正TabActivity中的Activity的ContextImpl的packageName Context base = activity.getBaseContext(); while (base instanceof ContextWrapper) {base = ((ContextWrapper) base).getBaseContext(); } if (HackContextImpl.instanceOf(base)) {HackContextImpl impl = new HackContextImpl(base);String packageName = PluginLoader.getApplication().getPackageName();// String packageName1 = activity.getPackageName();impl.setBasePackageName(packageName);impl.setOpPackageName(packageName); } } } super.callActivityOnCreate(activity, icicle); monitor.onActivityCreate(activity);}

到這插件activity組件就被順序的啟動(dòng)起來(lái)了,并且是系統(tǒng)在維護(hù)具備完整的生命周期。 組件service、Receiver也是一樣的,只是這兩個(gè)組件的攔截點(diǎn)在ActivityThread的Handler成員的回調(diào)Callback里面進(jìn)行的。Application和provider在插件啟動(dòng)的時(shí)候進(jìn)行加載。

資源沖突的解決方案resources.arsc資源描述符詳解

Android 插件化處理方案詳解

packageId: 包名id 資源類型id:string,drawable,layout,color 偏移:某一種類型的偏移值 解決沖突的方案

由于每個(gè)插件的包名是不一致的,可以事先規(guī)定某個(gè)插件的packageId的值固定,然后修改aapt對(duì)其進(jìn)行編譯固定,就可以保證每個(gè)插件分配的值不一樣了。

以上就是Android 插件化處理方案詳解的詳細(xì)內(nèi)容,更多關(guān)于Android 插件化處理方案的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Android
相關(guān)文章:
主站蜘蛛池模板: 久久精品a一国产成人免费网站 | 中文国产成人精品久久一 | 国产亚洲高清视频 | 欧美a欧美1级 | 久久久久久久性高清毛片 | 日鲁夜鲁鲁狠狠综合视频 | 新版天堂中文资源8在线 | 欧美成人鲁丝片在线观看 | 亚洲欧美久久 | 日韩一级片网址 | 国产成人综合网在线观看 | 国产一级一片免费播放 | 日韩毛片免费视频一级特黄 | 精品一区二区三区中文 | 欧美日韩专区国产精品 | 黄色a三级免费看 | 一级毛片在线观看视频 | 日韩欧美一区二区三区免费观看 | 欧美成人一级毛片 | 欧美一级二级三级 | 国产三级a三级三级天天 | 亚洲国产中文字幕 | 亚洲国产成+人+综合 | 日本a级在线| 国产成人高清亚洲一区久久 | 国产高清在线观看 | 亚洲第一页视频 | 在线观看亚洲免费 | 九九热国产精品视频 | 亚洲精品一二区 | 亚洲精品一区二区综合 | 免费黄网在线观看 | 成人高清在线观看 | 91精品91 | 日韩综合色 | 99久久99久久精品免费看子伦 | 男人添女人下面免费毛片 | 日本韩国一级片 | 蘑菇午夜三级 | 91亚洲精品国产第一区 | 久久久9999久久精品小说 |