电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

Android 插件化處理方案詳解

瀏覽:56日期:2022-09-19 18:15:08
插件化啟動Activity的過程

Android 插件化處理方案詳解

在宿主里面的AndroidManifest.xml里面注冊一個空的activity

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

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

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

@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被修改過 // 這里再把Action還原到原始的Action intent.setAction(targetClassName[1]); } else { intent.setAction(null); } // 添加一個標記符 intent.addCategory(RELAUNCH_FLAG + className);} else { throw new ClassNotFoundException('pluginClassName : ' + pluginClassName, new Throwable());} } else if (PluginManagerHelper.isExact(className, PluginDescriptor.ACTIVITY)) {// 這個邏輯是為了支持外部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 {// 進入這個分支可能是因為activity重啟了,比如橫豎屏切換,由于上面的分支已經(jīng)把Action還原到原始到Action了// 這里只能通過之前添加的標記符來查找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); }}

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

@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組件就被順序的啟動起來了,并且是系統(tǒng)在維護具備完整的生命周期。 組件service、Receiver也是一樣的,只是這兩個組件的攔截點在ActivityThread的Handler成員的回調(diào)Callback里面進行的。Application和provider在插件啟動的時候進行加載。

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

Android 插件化處理方案詳解

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

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

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

標簽: Android
相關文章:
主站蜘蛛池模板: 定制防伪标签_防伪标签印刷_防伪标签厂家-510品保防伪网 | IPO咨询公司-IPO上市服务-细分市场研究-龙马咨询 | pH污水传感器电极,溶解氧电极传感器-上海科蓝仪表科技有限公司 | 【直乐】河北石家庄脊柱侧弯医院_治疗椎间盘突出哪家医院好_骨科脊柱外科专业医院_治疗抽动症/关节病骨伤权威医院|排行-直乐矫形中医医院 | 液压升降货梯_导轨式升降货梯厂家_升降货梯厂家-河南东圣升降设备有限公司 | 上海租车公司_上海包车_奔驰租赁_上海商务租车_上海谐焕租车 | 专业生产动态配料系统_饲料配料系统_化肥配料系统等配料系统-郑州鑫晟重工机械有限公司 | 一体化预制泵站-一体化提升泵站-一体化泵站厂家-山东康威环保 | 高压管道冲洗清洗机_液压剪叉式升降机平台厂家-林君机电 | 缠膜机|缠绕包装机|无纺布包装机-济南达伦特机械设备有限公司 | CE认证_FCC认证_CCC认证_MFI认证_UN38.3认证-微测检测 CNAS实验室 | 大_小鼠elisa试剂盒-植物_人Elisa试剂盒-PCR荧光定量试剂盒-上海一研生物科技有限公司 | 冷藏车-东风吸污车-纯电动环卫车-污水净化车-应急特勤保障车-程力专汽厂家-程力专用汽车股份有限公司销售二十一分公司 | 石家庄装修设计_室内家装设计_别墅装饰装修公司-石家庄金舍装饰官网 | 烟雾净化器-滤筒除尘器-防爆除尘器-除尘器厂家-东莞执信环保科技有限公司 | 盐城网络公司_盐城网站优化_盐城网站建设_盐城市启晨网络科技有限公司 | 心得体会网_心得体会格式范文模板 | 防火门-专业生产甲级不锈钢钢质防火门厂家资质齐全-广东恒磊安防设备有限公司 | WTB5光栅尺-JIE WILL磁栅尺-B60数显表-常州中崴机电科技有限公司 | 石英粉,滑石粉厂家,山东滑石粉-莱州市向阳滑石粉有限公司 | 千斤顶,液压千斤顶-力良企业,专业的液压千斤顶制造商,shliliang.com | 警方提醒:赣州约炮论坛真的安全吗?2025年新手必看的网络交友防坑指南 | 石膏基自流平砂浆厂家-高强石膏基保温隔声自流平-轻质抹灰石膏粉砂浆批发-永康市汇利建设有限公司 | 杭州公司变更法人-代理记账收费价格-公司注销代办_杭州福道财务管理咨询有限公司 | 贵州水玻璃_-贵阳花溪闽兴水玻璃厂 | 广东风淋室_广东风淋室厂家_广东风淋室价格_广州开源_传递窗_FFU-广州开源净化科技有限公司 | 澳门精准正版免费大全,2025新澳门全年免费,新澳天天开奖免费资料大全最新,新澳2025今晚开奖资料,新澳马今天最快最新图库 | 塑料薄膜_PP薄膜_聚乙烯薄膜-常州市鑫美新材料包装厂 | 新疆散热器,新疆暖气片,新疆电锅炉,光耀暖通公司 | 正压密封性测试仪-静态发色仪-导丝头柔软性测试仪-济南恒品机电技术有限公司 | 心得体会网_心得体会格式范文模板 | 长城人品牌官网 | 新疆系统集成_新疆系统集成公司_系统集成项目-新疆利成科技 | 郑州宣传片拍摄-TVC广告片拍摄-微电影短视频制作-河南优柿文化传媒有限公司 | 齿轮减速电机一体机_蜗轮蜗杆减速马达-德国BOSERL齿轮减速机带电机生产厂家 | 电采暖锅炉_超低温空气源热泵_空气源热水器-鑫鲁禹电锅炉空气能热泵厂家 | 济南ISO9000认证咨询代理公司,ISO9001认证,CMA实验室认证,ISO/TS16949认证,服务体系认证,资产管理体系认证,SC食品生产许可证- 济南创远企业管理咨询有限公司 郑州电线电缆厂家-防火|低压|低烟无卤电缆-河南明星电缆 | 上海盐水喷雾试验机_两厢式冷热冲击试验箱-巨怡环试 | 百度关键词优化_网站优化_SEO价格 - 云无限好排名 | YAGEO国巨电容|贴片电阻|电容价格|三星代理商-深圳市巨优电子有限公司 | 滚珠丝杆升降机_螺旋升降机_丝杠升降机-德迈传动|