当前位置: 首页 > news >正文

网站建设分类wordpress缓存方法 张戈

网站建设分类,wordpress缓存方法 张戈,专业设计服务,上海制作网站的公司有哪些1 前言 前面章节我们学习了【Android Framework系列】第5章 AMS启动流程和【Android Framework系列】第6章 AMS原理之Launcher启动流程#xff0c;大概了解了AMS的原理及启动流程#xff0c;这一章节我们通过反射和动态代理对不同Android版本下的AMS进行Hook#xff0c;实现…1 前言 前面章节我们学习了【Android Framework系列】第5章 AMS启动流程和【Android Framework系列】第6章 AMS原理之Launcher启动流程大概了解了AMS的原理及启动流程这一章节我们通过反射和动态代理对不同Android版本下的AMS进行Hook实现登录页面的跳转。 这里我们只简单介绍一下HookAMS思路和重点代码需要详细了解的请到文末处项目地址下载查看。 1.1 实现打开统一登录页面 我们在Android的APP中一般会有登录状态。如果非登录状态下除了闪屏页和登录页外其他页面打开需要先登录。往往登录状态也会有有效期的说法如果在有效期到了跳转我们则需要跳转到登录页面而不是继续打开页面这种情况下我们通过HookAMS可以实现。 1.2 实现打开动态插件下发页面 另外通过HookAMS还可以实现动态下发插件的功能比如动态下发的Activity在AndroidManifest.xml里是没有注册要想打开则需要通过HookAMS的方式使用代理页面在AndroidManifest.xml注册在跳转时动态切换到下发下来的插件内Activity 2 实现 2.1 实现思路 通过动态代理的方式将AMS中startActivity方法拦截下来把要跳转的意图替换成我们要打开的Activity。由于不同的Android版本AMS源码有所差别所以这里区分SDK23、SDK28、SDK28这三种情况做HookAMS适配。下面我们来看看项目结构 2.2 项目结构 上图我们可以看到项目结构如下 // Config 常量配置类 // HookAMSApplication Application进行HookAMS初始化 // HookAMSUtils HookAMS工具类主要的Hook逻辑 // ListActivity 数据页面登录后才可打开 // LoginActivity 登录页 // MainActivity 首页这里打开数据页面 // ProxyActivity 代理页用于欺瞒AMS跳转时动态替换为真正Activity首先我们来看是怎么HookAMSApplication 2.3 HookAMSApplication package com.yvan.hookams;import android.app.Application; import android.os.Handler; import android.os.Looper;/*** author yvan* date 2023/7/28* description*/ public class HookAMSApplication extends Application {private final Handler handler new Handler(Looper.getMainLooper());Overridepublic void onCreate() {super.onCreate();handler.post(this::hookAMS);}public void hookAMS() {try {HookAMSUtils hookUtils new HookAMSUtils(this, ProxyActivity.class);hookUtils.hookAms();} catch (Exception e) {e.printStackTrace();}} }在App启动时Application的onCreate()方法内通过Handler的post方法进行对HookAMSUtils类的hookAms()方法调用。为什么要使用Handler呢初始化Application的onCreate()初始化还没完成直接调hookAms()方法会崩溃这里加了post将任务加入到主线程的队列里这样就不会出现崩溃异常。 我们继续看HookAMSUtils类 2.4 HookAMSUtils package com.yvan.hookams;import static android.os.Build.VERSION.SDK_INT;import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.SharedPreferences; import android.os.Handler; import android.os.Message; import android.util.Log;import java.lang.reflect.Field; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import java.util.List;/*** author yvan* date 2023/7/28* description Hook AMS工具类*/ public class HookAMSUtils {private static final String TAG HookAMSUtils.class.getSimpleName();private Context context;private Class? proxyActivity;/*** proxyActivity 传入一个有注册在AndroidManifest的就行** param context* param proxyActivity*/public HookAMSUtils(Context context, Class? proxyActivity) {this.context context;this.proxyActivity proxyActivity;}public void hookAms() throws Exception {if (SDK_INT 23) {hookAmsFor6();} else if (SDK_INT 28) {hookAmsFor9();} else {hookAmsFor10();}hookSystemHandler();}public void hookAmsFor10() throws Exception {Class? iActivityManagerClazz Class.forName(android.app.IActivityTaskManager);Class? clazz Class.forName(android.app.ActivityTaskManager);Field singletonField clazz.getDeclaredField(IActivityTaskManagerSingleton);singletonField.setAccessible(true);Object singleton singletonField.get(null);Class? singletonClass Class.forName(android.util.Singleton);Field mInstanceField singletonClass.getDeclaredField(mInstance);mInstanceField.setAccessible(true);final Object mInstance mInstanceField.get(singleton);Object proxyInstance Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(), new Class[]{iActivityManagerClazz}, new AmsInvocationHandler(mInstance));mInstanceField.setAccessible(true);mInstanceField.set(singleton, proxyInstance);}public void hookAmsFor9() throws Exception {// 1.反射获取类ActivityTaskManager这个就是AMS实例Class ActivityManagerClz Class.forName(android.app.ActivityManager);// 2.获取IActivityManagerSingleton并设置访问权限Field iActivityManagerSingletonFiled ActivityManagerClz.getDeclaredField(IActivityManagerSingleton);iActivityManagerSingletonFiled.setAccessible(true);// 因为是静态变量所以获取的到的是默认值final Object iActivityManagerSingletonObj iActivityManagerSingletonFiled.get(null);// 3.现在创建我们的AMS实例// 由于IActivityManager是一个接口那么其实我们可以使用Proxy类来进行代理对象的创建// 结果被摆了一道IActivityManager这玩意居然还是个AIDL动态生成的类编译器还不认识这个类怎么办反射咯// 反射创建一个Singleton的classClass SingletonClz Class.forName(android.util.Singleton);Field mInstanceField SingletonClz.getDeclaredField(mInstance);mInstanceField.setAccessible(true);// 4.获取AMS ProxyObject iActivityManagerObj mInstanceField.get(iActivityManagerSingletonObj);// 5.获取需要实现的接口IActivityManager实现类Class iActivityManagerClz Class.forName(android.app.IActivityManager);// 6.动态生成接口对象// 构建代理类需要两个东西用于创建伪装的Intent// 拿到AMS实例然后用代理的AMS换掉真正的AMS代理的AMS则是用 假的Intent骗过了 activity manifest检测.Object proxyIActivityManager Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),new Class?[]{iActivityManagerClz}, new AmsInvocationHandler(iActivityManagerObj));mInstanceField.setAccessible(true);// 7.替换掉系统的变量mInstanceField.set(iActivityManagerSingletonObj, proxyIActivityManager);}public void hookAmsFor6() throws Exception {//1.反射获取类ActivityManagerNativeClass ActivityManagerClz Class.forName(android.app.ActivityManagerNative);//2.获取变量gDefaultField IActivityManagerSingletonFiled ActivityManagerClz.getDeclaredField(gDefault);//2.1 设置访问权限IActivityManagerSingletonFiled.setAccessible(true);//3. 获取变量的实例值Object IActivityManagerSingletonObj IActivityManagerSingletonFiled.get(null);//4.获取mInstanceClass SingletonClz Class.forName(android.util.Singleton);Field mInstanceField SingletonClz.getDeclaredField(mInstance);mInstanceField.setAccessible(true);//5.获取AMS ProxyObject AMSProxy mInstanceField.get(IActivityManagerSingletonObj);//6.由于不能去手动实现IActivityManager实现类// 所以只能通过动态代理去动态生成实现类//6.1 获取需要实现的接口Class IActivityManagerClz Class.forName(android.app.IActivityManager);//6.2 动态生成接口对象Object proxyIActivityManager Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),new Class[]{IActivityManagerClz}, new AmsInvocationHandler(AMSProxy));mInstanceField.setAccessible(true);//7.替换掉系统的变量mInstanceField.set(IActivityManagerSingletonObj, proxyIActivityManager);}private class AmsInvocationHandler implements InvocationHandler {private Object iActivityManagerObject;public AmsInvocationHandler(Object iActivityManagerObject) {this.iActivityManagerObject iActivityManagerObject;}Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {if (startActivity.contains(method.getName())) {Intent intent null;int index 0;for (int i 0; i args.length; i) {Object arg args[i];if (arg instanceof Intent) {intent (Intent) args[i]; // 原意图过不了安检index i;break;}}Intent proxyIntent new Intent();ComponentName componentName new ComponentName(context, proxyActivity);proxyIntent.setComponent(componentName);proxyIntent.putExtra(realIntent, intent);//替换原有的intent为我们自己生成的为了骗过PMS//为跳到我们的传入的proxyActivityargs[index] proxyIntent;}return method.invoke(iActivityManagerObject, args);}}//上面的主要是替换成我们自己的intent骗过系统//下面的主要是将我们上面替换的intent中取出我们真正的意图也就是正在要启动的Activity////下面是为了拿到mH对象但是mH是一个非static 的值那我们就只能拿到他的持有对象也就是ActivityThread//正好发现在ActivityThread类中有一个static变量sCurrentActivityThread值可以拿到ActivityThread类那我们就从他入手public void hookSystemHandler() throws Exception {//1.反射ActivityThreadClass ActivityThreadClz Class.forName(android.app.ActivityThread);//2. 获取sCurrentActivityThread 是一个static变量Field field ActivityThreadClz.getDeclaredField(sCurrentActivityThread);field.setAccessible(true);//3.获取ActivityThread对象Object ActivityThreadObj field.get(null);//4.通过ActivityThreadObj获取到mH变量Field mHField ActivityThreadClz.getDeclaredField(mH);mHField.setAccessible(true);//5.获取到mH的对象Handler mHObj (Handler) mHField.get(ActivityThreadObj);//ok当前的mH拿到了//到这里获取到mH的对象了那我们怎么去监听他的方法调用呢//能不能通过动态代理不能因为它不是个接口//由于在Handler的源码中我们知道如果mCallback如果不等于空就会调用mCallback的handleMessage方法。//6.获取mH的mCallbackField mCallbackField Handler.class.getDeclaredField(mCallback);mCallbackField.setAccessible(true);//7.创建我们自己的Callback自己处理handleMessageHandler.Callback proxyMHCallback getMHCallback();//8.给系统的mHHandler的mCallback设值proxyMHCallbackmCallbackField.set(mHObj, proxyMHCallback);}private Handler.Callback getMHCallback() {if (SDK_INT 23) {return new ProxyHandlerCallbackFor6();} else if (SDK_INT 28) {return new ProxyHandlerCallbackFor();} else {return new ProxyHandlerCallbackFor();}}private class ProxyHandlerCallbackFor6 implements Handler.Callback {private int LAUNCH_ACTIVITY 100;Overridepublic boolean handleMessage(Message msg) {if (msg.what LAUNCH_ACTIVITY) {try {Class ActivityClientRecord Class.forName(android.app.ActivityThread$ActivityClientRecord);//判断传过来的值msg.obj是不是ClientTransaction对象if (!ActivityClientRecord.isInstance(msg.obj)) return false;//获取ActivityClientRecord的intent变量Field intentField ActivityClientRecord.getDeclaredField(intent);intentField.setAccessible(true);if (intentField null) return false;Intent mIntent (Intent) intentField.get(msg.obj);if (mIntent null) return false;//获取我们之前传入的realIntent也就是我们真正要打开的ActivityIntent realIntent mIntent.getParcelableExtra(realIntent);if (realIntent null) {return false;}realStartActivity(mIntent, realIntent);} catch (Exception e) {e.printStackTrace();}}return false;}}private class ProxyHandlerCallbackFor implements Handler.Callback {private int EXECUTE_TRANSACTION 159;Overridepublic boolean handleMessage(Message msg) {if (msg.what EXECUTE_TRANSACTION) {try {Class ClientTransactionClz Class.forName(android.app.servertransaction.ClientTransaction);//判断传过来的值msg.obj是不是ClientTransaction对象if (!ClientTransactionClz.isInstance(msg.obj)) return false;Class LaunchActivityItemClz Class.forName(android.app.servertransaction.LaunchActivityItem);//获取ClientTransaction的mActivityCallbacks变量Field mActivityCallbacksField ClientTransactionClz.getDeclaredField(mActivityCallbacks);//ClientTransaction的成员//设值成mActivityCallbacksField.setAccessible(true);//获取到ASM传递过来的值ClientTransaction对象里的mActivityCallbacks变量Object mActivityCallbacksObj mActivityCallbacksField.get(msg.obj);List list (List) mActivityCallbacksObj;if (list.size() 0) return false;Object LaunchActivityItemObj list.get(0);if (!LaunchActivityItemClz.isInstance(LaunchActivityItemObj)) return false;//获取mIntent变量Field mIntentField LaunchActivityItemClz.getDeclaredField(mIntent);mIntentField.setAccessible(true);//获取mIntent对象Intent mIntent (Intent) mIntentField.get(LaunchActivityItemObj);//获取我们之前传入的realIntent也就是我们真正要打开的ActivityIntent realIntent mIntent.getParcelableExtra(realIntent);if (realIntent null) {return false;}realStartActivity(mIntent, realIntent);} catch (Exception e) {e.printStackTrace();}}return false;}}private void realStartActivity(Intent mIntent, Intent realIntent) {//登录判断SharedPreferences share context.getSharedPreferences(Config.SP_NAME,Context.MODE_PRIVATE);if (share.getBoolean(Config.SP_KEY_LOGIN, false)) {mIntent.setComponent(realIntent.getComponent());} else {Log.i(TAG, handleLauchActivity: realIntent.getComponent().getClassName());ComponentName componentName new ComponentName(context, LoginActivity.class);mIntent.putExtra(extraIntent, realIntent.getComponent().getClassName());mIntent.setComponent(componentName);}}} 从上面代码我们能看到 hookAms()方法分别是SDK23、SDK28、SDK28三种情况进行HookAMS其实都是大同小异。实际上是获取到IActivityManager对象通过动态代理Proxy.newProxyInstance()Hook到其所有方法通过AmsInvocationHandler进行方法调用的回调。hookSystemHandler()方法在hookAms()方法调用后立刻执行通过反射获取android.app.ActivityThread类对象的sCurrentActivityThread属性和Handler实例mH将Handler的回调handleMessage()方法进行拦截。根据SDK23、SDK28、SDK28三种情况不同来区别处理。1中的AmsInvocationHandler负责hookAms()内Hook到的方法调用的处理在Hook到的Callback中判断为startActivity()方法则拦截下来将我们真正要跳转的Activity意图存在Extra内由于在原来的Intent中隐藏了真正的Activity意图所以只需要将真正的意图拿出来替换将其Intent的意图替换为要打开的Activity。2中handleMessage()其实就是将startActivity()方法进行拦截判断如果是未登录状态则将真正要跳转的Activity从Extra内拿出来进行跳转已登录则不替换Intent意图跳转。 3 总结 文章只做核心HookAMS代码思路的分析这里是项目地址小伙伴可以自行下载查看别忘了点Star喔谢谢
http://www.tj-hxxt.cn/news/225532.html

相关文章:

  • 哪个网站可以做计算机二级的题wordpress 媒体文件库
  • 在哪个网站做ppt模板赚钱优化营商环境指什么
  • 专门做黄漫的网站建立一个购物网站平台费用
  • 网站服务器租用还是自买东莞市疾控中心官网
  • 营口电商平台网站建设wordpress指定分类文章
  • 模板网站可以做优化新人怎么做跨境电商
  • 网站开发款计入什么科目网站开发 python 工具
  • 大唐网站首页广东建设工程备案网站
  • 机械网站案例网站开发课设个人总结
  • 珠海网站建设熊掌号怎样免费做一个网站
  • 石家庄网站编辑郑州 发布会
  • 自由型网站网站建设好弄不好弄
  • 58同城 网站建设 推广排名教务系统管理
  • 建设企业网站需要使用哪些技术网站建设方案行业
  • 怎么可以联系到网站开发者网站备案产品信息错误
  • 网站建设与运营市场开拓方案大连三丰建设集团公司网站
  • 网站建设开什么名目电子商务网站开发与应用论文
  • 做网站要主机还是服务器那些做电影的网站赚钱吗
  • 网站建设有什么注意宿州科技网站建设
  • 美食网站html静态域名解析要登入哪个网站做
  • 宁波哪里有做网站的网站的建设原始代码
  • 北京最大做网站的公司网页版qq邮箱登录
  • 甘肃建网站长沙微信网站建设
  • 做淘宝客网站流量选择福州移动网站建设
  • 在万网申请的域名_需要把万网的账户密码给做网站的吗淄博网站设计制作
  • 福建网站开发有限开发公司wordpress 504
  • 东莞网站建设公司辉煌大厦怎样创建网站和网页
  • 天津网站建设好公司做网站投资太大 网站也没搞起来
  • 电子商务网站规划 分析 设计阜阳网页
  • 做网站需要备注号码沧州网站设计报价