【Android插件化】启动没有在Manifest中注册的Activity

2024-09-03 14:58

本文主要是介绍【Android插件化】启动没有在Manifest中注册的Activity,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1 概述

如果要启动没有在Manifest中注册的Activity,应该从startActivity着手。一般启动Activity的方式有两种,一种是startActivity,一种是startActivityForResult。其实startActivity最终调用的也是startActivityForResult,如下所示:

    //Activity.java@Overridepublic void startActivity(Intent intent, @Nullable Bundle options) {if (options != null) {startActivityForResult(intent, -1, options);} else {// Note we want to go through this call for compatibility with// applications that may have overridden the method.startActivityForResult(intent, -1);}}

当我们启动一个Activity时,AMS会对这个Activity是否在AndroidManifest中声明注册进行检查,如果没有注册,则会报错。要想启动没有在AndroidManifest中注册的Activity,则需要“欺骗AMS”,让其以为我们已经在AndroidManifest中注册了。

基本思路是:

  1. 发送要启动的Activity信息给AMS之前,把这个Activity替换为一个在AndroidManifest中声明的StubActivity,这样就能绕过AMS的检查,在替换过程中,要把原来的Activity信息保存起来。
  2. AMS通知App启动StubActivity时,我们再将保存的Activity替换StubActivity,这样就能启动没有在AndroidManifest中注册的Activity。

修改Activity启动流程如图所示(图片来自网络)

image.png

2 代码实现

AndroidManifest.xml

    <applicationandroid:name=".MainApplication"android:allowBackup="true"android:icon="@mipmap/ic_launcher"android:label="@string/app_name"android:roundIcon="@mipmap/ic_launcher_round"android:supportsRtl="true"android:theme="@style/AppTheme"><activity android:name=".MainActivity"><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity><activity android:name=".SubActivity"/></application>

一共两个Activity,分别是MainActivity和SubActivity,一个MainApplication,SubActivity是一个壳Activity,用于欺骗AMS的。

MainActivity.java

    private View.OnClickListener mListener = new View.OnClickListener() {@Overridepublic void onClick(View v) {Intent intent = new Intent();intent.setClass(MainActivity.this, PluginTestActivity.class);startActivity(intent);}};

PluginTestActivity并没有在AndroidManifest中注册,正常情况下会报错,所以我们需要欺骗AMS。

欺骗AMS的方式有几种,这里我们选择对ActivityThread的mInstrumentation进行hook。代码如下:

public class HookHelper {public static final String PLUGIN_INTENT = "plugin_intent";public static void hookInstrumentation(Context context,String className) throws Exception{//获取ActivityThread的字节码对象Class<?> clazz = Class.forName("android.app.ActivityThread");//获取ActivityThread中的sCurrentActivityThread字段Field sCurrentActivityThreadField = ReflectUtils.getField(clazz,"sCurrentActivityThread");//获取ActivityThread中的mInstrumentation字段Field mInstrumentationField = ReflectUtils.getField(clazz,"mInstrumentation");//获取ActivityThread中的sCurrentActivityThread的值并赋值给currentActivityThreadObject currentActivityThread = sCurrentActivityThreadField.get(clazz);//获取ActivityThread中的mInstrumentation的值并赋值给instrumentationInstrumentation instrumentation = (Instrumentation) mInstrumentationField.get(currentActivityThread);//创建一个PackageManager对象packageManagerPackageManager packageManager = context.getPackageManager();//创建InstrumentationProxy对象,InstrumentationProxy是继承Instrumentation的类,下面详细介绍InstrumentationProxy instrumentationProxy = new InstrumentationProxy(instrumentation,packageManager,className);//将ActivityThread中的mInstrumentation赋值为instrumentationProxyReflectUtils.setFieldObject(clazz,currentActivityThread,"mInstrumentation",instrumentationProxy);}
}

InstrumentationProxy类如下

package com.example.plugintest;import android.app.Activity;
import android.app.Instrumentation;
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo;
import android.os.Bundle;
import android.os.IBinder;
import android.text.TextUtils;import java.lang.reflect.Method;
import java.util.List;/*** Created by yds* on 2020/2/14.*/
public class InstrumentationProxy extends Instrumentation{private Instrumentation mInstrumentation;private PackageManager mPackageManager;private String className;public InstrumentationProxy(Instrumentation mInstrumentation, PackageManager mPackageManager, String className) {this.mInstrumentation = mInstrumentation;this.mPackageManager = mPackageManager;this.className = className;}public ActivityResult execStartActivity(Context who, IBinder contextThread, IBinder token, Activity target,Intent intent, int requestCode, Bundle options) {List<ResolveInfo> infos = mPackageManager.queryIntentActivities(intent, PackageManager.MATCH_ALL);if (infos == null || infos.size() == 0) {intent.putExtra(HookHelper.PLUGIN_INTENT,intent.getComponent().getClassName());intent.setClassName(who,className);}try {Method execMethod = Instrumentation.class.getDeclaredMethod("execStartActivity",Context.class,IBinder.class,IBinder.class,Activity.class,Intent.class,int.class,Bundle.class);return (ActivityResult) execMethod.invoke(mInstrumentation,who,contextThread,token,target,intent,requestCode,options);} catch (Exception e) {e.printStackTrace();}return null;}public Activity newActivity(ClassLoader cl,String className,Intent intent) throwsInstantiationException,IllegalAccessException,ClassNotFoundException{String intentName = intent.getStringExtra(HookHelper.PLUGIN_INTENT);if(!TextUtils.isEmpty(intentName)){return super.newActivity(cl,intentName,intent);}return super.newActivity(cl,className,intent);}
}

只要将InstrumentationProxy赋值给ActivityThread的mInstrumentation,调用startActivity时,最终会调用InstrumentationProxy的execStartActivity和newActivity。我们只要在execStartActivity中将未注册的Activity保存起来,并使用已注册的SubActivity,然后在newActivity中将保存起来未注册的Activity替换SubActivity就可以了。

源码地址:https://github.com/Yedongsheng/PluginTest01

这篇关于【Android插件化】启动没有在Manifest中注册的Activity的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

kkFileView启动报错:报错2003端口占用的问题及解决

《kkFileView启动报错:报错2003端口占用的问题及解决》kkFileView启动报错因office组件2003端口未关闭,解决:查杀占用端口的进程,终止Java进程,使用shutdown.s... 目录原因解决总结kkFileViewjavascript启动报错启动office组件失败,请检查of

SQL Server安装时候没有中文选项的解决方法

《SQLServer安装时候没有中文选项的解决方法》用户安装SQLServer时界面全英文,无中文选项,通过修改安装设置中的国家或地区为中文中国,重启安装程序后界面恢复中文,解决了问题,对SQLSe... 你是不是在安装SQL Server时候发现安装界面和别人不同,并且无论如何都没有中文选项?这个问题也

Linux下在线安装启动VNC教程

《Linux下在线安装启动VNC教程》本文指导在CentOS7上在线安装VNC,包含安装、配置密码、启动/停止、清理重启步骤及注意事项,强调需安装VNC桌面以避免黑屏,并解决端口冲突和目录权限问题... 目录描述安装VNC安装 VNC 桌面可能遇到的问题总结描js述linux中的VNC就类似于Window

linux下shell脚本启动jar包实现过程

《linux下shell脚本启动jar包实现过程》确保APP_NAME和LOG_FILE位于目录内,首次启动前需手动创建log文件夹,否则报错,此为个人经验,供参考,欢迎支持脚本之家... 目录linux下shell脚本启动jar包样例1样例2总结linux下shell脚本启动jar包样例1#!/bin

Spring Boot Maven 插件如何构建可执行 JAR 的核心配置

《SpringBootMaven插件如何构建可执行JAR的核心配置》SpringBoot核心Maven插件,用于生成可执行JAR/WAR,内置服务器简化部署,支持热部署、多环境配置及依赖管理... 目录前言一、插件的核心功能与目标1.1 插件的定位1.2 插件的 Goals(目标)1.3 插件定位1.4 核

SpringBoot整合Dubbo+ZK注册失败的坑及解决

《SpringBoot整合Dubbo+ZK注册失败的坑及解决》使用Dubbo框架时,需在公共pom添加依赖,启动类加@EnableDubbo,实现类用@DubboService替代@Service,配... 目录1.先看下公共的pom(maven创建的pom工程)2.启动类上加@EnableDubbo3.实

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注