Android插件化系列第(五)篇---Activity的插件化方案(代理模式)

2024-09-02 14:58

本文主要是介绍Android插件化系列第(五)篇---Activity的插件化方案(代理模式),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这篇文章介绍Activity的插件化方案,Activity的插件化方案不止今天介绍的这一种。建议在看本文之前,先看我的前两篇博客,如果前两篇有认真看过,那么阅读本文至多十分钟完事儿!

  • Android插件化系列第(一)篇—Hook技术之Activity的启动过程拦截
  • Android插件化系列第(二)篇—动态加载技术之apk换肤
  • Android插件化系列第(四)篇—插件加载机制两种方案

看过上面文章,我们知道了怎么加载一个插件中的Activity类了,想做Activity的插件化,还需要解决几个问题。

  • 1、插件中所有的Activity都没有在宿主中注册,怎么欺骗AMS去启动一个清单文件不存在的Activity;
  • 2、把一个Activity类加载之后,怎么使插件里的Activity具有生命周期;
  • 3、插件apk中用过的各种资源,如何动态的加载资源。

关于问题1,在本系列的第一篇已经说过,首先在宿主中声明一个ProxyActivity,然后启动ProxyActivity,即:ProxyActivity + 插件中没注册的Activity = 标准的Activity

关于问题2,其实在本系列第一篇也有体现,准备在下一篇写的更清楚一点。

关于问题3,在本系列的第二篇也写过,以一个换肤的例子说明怎么样去加载插件中的资源。

本篇博客主要讲述的是第二个问题,把一个Activity类加载之后,怎么使插件里的Activity具有生命周期。

这里使用Activity代理模式。老套路,在宿主APK注册一个ProxyActivity(代理Activity),就是作为占坑使用。每次打开插件APK里的某一个Activity的时候,都是在宿主里使用启动ProxyActivity,然后在ProxyActivity的生命周期里方法中,调用插件中的Activity实例的生命周期方法,从而执行插件APK的业务逻辑。所以思路就来了。
第一、ProxyActivity中需要保存一个Activity实例,该实例记录着当前需要调用插件中哪个Activity的生命周期方法。
第二、ProxyActivity如何调用插件apk中Activity的所有生命周期的方法,使用反射呢?还是其他方式。

这种Activity的插件化思想最初来自dynamic-load-apk,dynamic-load-apk调用插件apk中Activity的所有生命周期的方法最初使用的是反射的方式,因为反射会影响效率,最后换成接口的方式。先看反射这种方式,在代理activity中,第一步要获取将要反射所有生命周期方法。

    protected void instantiateLifecircleMethods(Class<?> localClass) {String[] methodNames = new String[] {"onRestart","onStart","onResume","onPause","onStop","onDestory"};for (String methodName : methodNames) {Method method = null;try {method = localClass.getDeclaredMethod(methodName, new Class[] { });method.setAccessible(true);} catch (NoSuchMethodException e) {e.printStackTrace();}//mActivityLifecircleMethods是一个map集合mActivityLifecircleMethods.put(methodName, method);}Method onCreate = null;try {onCreate = localClass.getDeclaredMethod("onCreate", new Class[] { Bundle.class });onCreate.setAccessible(true);} catch (NoSuchMethodException e) {e.printStackTrace();}mActivityLifecircleMethods.put("onCreate", onCreate);Method onActivityResult = null;try {onActivityResult = localClass.getDeclaredMethod("onActivityResult",new Class[] { int.class, int.class, Intent.class });onActivityResult.setAccessible(true);} catch (NoSuchMethodException e) {e.printStackTrace();}mActivityLifecircleMethods.put("onActivityResult", onActivityResult);}

第二步,反射所有Activity的生命周期方法

    @Overrideprotected void onResume() {super.onResume();Method onResume = mActivityLifecircleMethods.get("onResume");if (onResume != null) {try {onResume.invoke(mRemoteActivity, new Object[] { });} catch (Exception e) {e.printStackTrace();}}}@Overrideprotected void onPause() {Method onPause = mActivityLifecircleMethods.get("onPause");if (onPause != null) {try {onPause.invoke(mRemoteActivity, new Object[] { });} catch (Exception e) {e.printStackTrace();}}super.onPause();}protected void setRemoteActivity(Object activity) {try {//插件中Activity对象mRemoteActivity = (Activity) activity;} catch (ClassCastException e) {e.printStackTrace();}}

这是反射的方式,后面dynamic-load-apk经过优化,改成接口的方式,将activity的生命周期方法封装一个接口,代理activity中实现一下这个接口,然后还是通过代理activity去调用插件activity实现的生命周期方法。

public interface DLPlugin {public void onStart();public void onRestart();public void onActivityResult(int requestCode, int resultCode, Intent data);public void onResume();public void onPause();public void onStop();public void onDestroy();public void onCreate(Bundle savedInstanceState);public void setProxy(Activity proxyActivity, String dexPath);public void onSaveInstanceState(Bundle outState);public void onNewIntent(Intent intent);public void onRestoreInstanceState(Bundle savedInstanceState);public boolean onTouchEvent(MotionEvent event);public boolean onKeyUp(int keyCode, KeyEvent event);public void onWindowAttributesChanged(LayoutParams params);public void onWindowFocusChanged(boolean hasFocus);
}
    @Overrideprotected void onStart() {mRemoteActivity.onStart();super.onStart();}@Overrideprotected void onRestart() {mRemoteActivity.onRestart();super.onRestart();}@Overrideprotected void onResume() {mRemoteActivity.onResume();super.onResume();}@Overrideprotected void onPause() {mRemoteActivity.onPause();super.onPause();}

这就是dynamic-load-apk之中Activity插件化方案,如果要使用这种方案,我们开发插件需要遵循一定的规范,dynamic-load-apk要求遵循的规范如下:

  • 1、慎用this(接口除外):因为this指向的是当前对象,即apk中的activity,但是由于activity已经不是常规意义上的activity,所以this是没有意义的,但是如果this表示的是一个接口而不是context,比如activity实现了而一个接口,那么this继续有效。

  • 2、使用that:既然this不能用,那就用that,that是apk中activity的基类BaseActivity中的一个成员,它在apk安装运行的时候指向this,而在未安装的时候指向宿主程序中的代理activity,anyway,that is better than this。

-3、activity的成员方法调用问题:原则来说,需要通过that来调用成员方法,但是由于大部分常用的api已经被重写,所以仅仅是针对部分api才需要通过that去调用用。同时,apk安装以后仍然可以正常运行。

  • 4、启动新activity的约束:启动外部activity不受限制,启动apk内部的activity有限制,首先由于apk中的activity没注册,所以不支持隐式调用,其次必须通过BaseActivity中定义的新方法startActivityByProxy和startActivityForResultByProxy,还有就是不支持LaunchMode。

  • 5、目前暂不支持Service、BroadcastReceiver等需要注册才能使用的组件,但广播可以采用代码动态注册。

可以看到限制非常大,现在估计不会有谁采用这种方式做插件化开发了,但是2014年那个时候,dynamic-load-apk这种思想还是非常先进的,相信这为后来的插件化方案提供了很宝贵的经验,下一篇博客讲解用Hook方式,实现业界比较倡导的一种Activity插件化方案。

Please accept mybest wishes for your happiness and success !

参考:https://github.com/singwhatiwanna/dynamic-load-apk

这篇关于Android插件化系列第(五)篇---Activity的插件化方案(代理模式)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1130275

相关文章

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

IDEA常用插件之代码扫描SonarLint详解

《IDEA常用插件之代码扫描SonarLint详解》SonarLint是一款用于代码扫描的插件,可以帮助查找隐藏的bug,下载并安装插件后,右键点击项目并选择“Analyze”、“Analyzewit... 目录SonajavascriptrLint 查找隐藏的bug下载安装插件扫描代码查看结果总结Sona

Java解析JSON的六种方案

《Java解析JSON的六种方案》这篇文章介绍了6种JSON解析方案,包括Jackson、Gson、FastJSON、JsonPath、、手动解析,分别阐述了它们的功能特点、代码示例、高级功能、优缺点... 目录前言1. 使用 Jackson:业界标配功能特点代码示例高级功能优缺点2. 使用 Gson:轻量

Redis KEYS查询大批量数据替代方案

《RedisKEYS查询大批量数据替代方案》在使用Redis时,KEYS命令虽然简单直接,但其全表扫描的特性在处理大规模数据时会导致性能问题,甚至可能阻塞Redis服务,本文将介绍SCAN命令、有序... 目录前言KEYS命令问题背景替代方案1.使用 SCAN 命令2. 使用有序集合(Sorted Set)

MyBatis延迟加载的处理方案

《MyBatis延迟加载的处理方案》MyBatis支持延迟加载(LazyLoading),允许在需要数据时才从数据库加载,而不是在查询结果第一次返回时就立即加载所有数据,延迟加载的核心思想是,将关联对... 目录MyBATis如何处理延迟加载?延迟加载的原理1. 开启延迟加载2. 延迟加载的配置2.1 使用

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

无人叉车3d激光slam多房间建图定位异常处理方案-墙体画线地图切分方案

墙体画线地图切分方案 针对问题:墙体两侧特征混淆误匹配,导致建图和定位偏差,表现为过门跳变、外月台走歪等 ·解决思路:预期的根治方案IGICP需要较长时间完成上线,先使用切分地图的工程化方案,即墙体两侧切分为不同地图,在某一侧只使用该侧地图进行定位 方案思路 切分原理:切分地图基于关键帧位置,而非点云。 理论基础:光照是直线的,一帧点云必定只能照射到墙的一侧,无法同时照到两侧实践考虑:关

高效+灵活,万博智云全球发布AWS无代理跨云容灾方案!

摘要 近日,万博智云推出了基于AWS的无代理跨云容灾解决方案,并与拉丁美洲,中东,亚洲的合作伙伴面向全球开展了联合发布。这一方案以AWS应用环境为基础,将HyperBDR平台的高效、灵活和成本效益优势与无代理功能相结合,为全球企业带来实现了更便捷、经济的数据保护。 一、全球联合发布 9月2日,万博智云CEO Michael Wong在线上平台发布AWS无代理跨云容灾解决方案的阐述视频,介绍了

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo