andorid jar/库源码解析之HotXposed
HotXposed:
作用:
免重启手机,实现Xposed hook更新。(当然app是要重启的)
栗子:
入口:
// Hook 入口
public void handleLoadPackage(XC_LoadPackage.LoadPackageParam lpparam) throws Throwable { hook(HookerDispatcher.class, lpparam); } public static void hook(Class clazz, XC_LoadPackage.LoadPackageParam lpparam) throws Exception { String packageName = clazz.getName().replace("." + clazz.getSimpleName(), ""); Log.i(tag, "packageName " + packageName); File apkFile = getApkFile(packageName); Log.i(tag, "apkFile.getAbsolutePath " + apkFile.getAbsolutePath()); if (!apkFile.exists()) { Log.i(tag, "apk file not found"); return; } Log.i(tag, "去掉 xposed 通知"); filterNotify(lpparam); PathClassLoader classLoader = new PathClassLoader(apkFile.getAbsolutePath(), lpparam.getClass().getClassLoader()); Log.i(tag, "classLoader " + classLoader); XposedHelpers.callMethod(classLoader.loadClass(clazz.getName()).newInstance(), "dispatch", lpparam); }
热更新的Hook:
public class HookerDispatcher implements IHookerDispatcher { private static String tag = "HotXposed"; @Override public void dispatch(final XC_LoadPackage.LoadPackageParam lpparam) { if(!Constants.TARGET_PACKAGE_NAME.equals(lpparam.packageName)){ return; } Log.i(tag, "dispatch 被调用 热更新代码从这里开始"); if (lpparam.packageName.contains(Constants.TARGET_PACKAGE_NAME) && lpparam.processName.equals(lpparam.packageName)) { findAndHookMethod(Application.class, "attach", Context.class, new XC_MethodHook() { @Override protected void afterHookedMethod(final MethodHookParam param) throws Throwable { final ClassLoader cl = ((Context) param.args[0]).getClassLoader(); Log.i(tag, "enter 2 " + lpparam.packageName); } }); } } }
源码解读:
首先来看入口实现。
1、通过集成 IXposedHookLoadPackage 实现他的 handleLoadPackage方法来实现他的入口。
2、首先,确定 HookerDispatcher 在包名下面,然后通过去掉他自身的名称,即是包名,也可以写死包名。
3、通过包名,得到当前安装模块(即Apk)的,路径,一般是(/data/app/包名-1/2/base.apk).这个路径下面。通过判断文件是否存在,可以知道是1还是2
4、然后 因为得到了apk的路径,可以通过构造一个 PathClassLoader对象,入参传入apk路径和当前的ClassLoader,来创建一个PathClassLoader对象。(下一步大有用处)
5、把这个ClassLoader,作为入口,调用Xposed的Hook入口方法,即;用于热更新的部分(这里有点绕,我详细解释一下。)
5.1、因为xposed,载入是在apk加载的时候,这个时候,上面的源码1.是不会被热更新的。
5.2、而源码2中,是通过加载apk得到的ClassLoader的调用,因为app安装之后,产生了变化,是可以被热更新到的。
源码:https://github.com/githubwing/HotXposed
引入:
implementation 'com.github.githubwing:HotXposed:v1.0.0
'
RT