如何实现不安装xpoedinstall激活xposed模块

简介: xpoedinstall及安装器,它的作用有几个:安装xposed框架、激活xposed模块、搜索模块等等如果我们已经安装了xposed框架,或者说我们自己二次开发了这个框架,通过手动的方式安装。然后需要使用我们自己的xposed模块,这时xposedinstall的作用就只剩激活模块了,那么我们是否可以摆脱它来激活模块?

前言


xpoedinstall及安装器,它的作用有几个:安装xposed框架、激活xposed模块、搜索模块等等

如果我们已经安装了xposed框架,或者说我们自己二次开发了这个框架,通过手动的方式安装。然后需要使用我们自己的xposed模块,这时xposedinstall的作用就只剩激活模块了,那么我们是否可以摆脱它来激活模块?


源码


先来看看它如何激活模块的

在它的源码中有一个ModuleUtil类,其中


public void setModuleEnabled(String packageName, boolean enabled) {
    if (enabled)
        mPref.edit().putInt(packageName, 1).apply();
    else
        mPref.edit().remove(packageName).apply();
}
复制代码


可以看到将激活的模块的包名存到了sharepreference中,然后还有另外一个类


public synchronized void updateModulesList(boolean showToast) {
    try {
        Log.i(XposedApp.TAG, "updating modules.list");
        int installedXposedVersion = XposedApp.getInstalledXposedVersion();
        PrintWriter modulesList = new PrintWriter(MODULES_LIST_FILE);
        PrintWriter enabledModulesList = new PrintWriter(XposedApp.ENABLED_MODULES_LIST_FILE);
        List<InstalledModule> enabledModules = getEnabledModules();
        for (InstalledModule module : enabledModules) {
            if (module.minVersion > installedXposedVersion || module.minVersion < MIN_MODULE_VERSION)
                continue;
            modulesList.println(module.app.sourceDir);
            try {
                String installer = mPm.getInstallerPackageName(module.app.packageName);
                if (!PLAY_STORE_PACKAGE.equals(installer)) {
                    enabledModulesList.println(module.app.packageName);
                }
            } catch (IllegalArgumentException ignored) {
                // In rare cases, the package might not be installed anymore at this point,
                // so the PackageManager can't return its installer package name.
            }
        }
        modulesList.close();
        enabledModulesList.close();
        FileUtils.setPermissions(MODULES_LIST_FILE, 00664, -1, -1);
        FileUtils.setPermissions(XposedApp.ENABLED_MODULES_LIST_FILE, 00664, -1, -1);
        if (showToast)
            showToast(R.string.xposed_module_list_updated);
    } catch (IOException e) {
        Log.e(XposedApp.TAG, "cannot write " + MODULES_LIST_FILE, e);
        Toast.makeText(mApp, "cannot write " + MODULES_LIST_FILE + e, Toast.LENGTH_SHORT).show();
    }
}
复制代码


这里有一个getEnableModules函数,如下:


public List<InstalledModule> getEnabledModules() {
    LinkedList<InstalledModule> result = new LinkedList<InstalledModule>();
    for (String packageName : mPref.getAll().keySet()) {
        InstalledModule module = getModule(packageName);
        if (module != null)
            result.add(module);
        else
            setModuleEnabled(packageName, false);
    }
    return result;
}
复制代码


可以看到这个函数正是从刚才sharepreference中读取激活的模块包名

然后回到updateModulesList中,可以看到将这些激活的模块的sourceDir保存到MODULES_LIST_FILE这个文件中,即:


private static final String MODULES_LIST_FILE = XposedApp.BASE_DIR + "conf/modules.list";
复制代码


以上就是xposedinstaller激活模块的流程,可以看到就是将要激活的模块的apk地址保存到一个modules.list文件中。看来真正激活使用模块是在xposedbridge中,在它的main函数中(XposedBridge类)


protected static void main(String[] args) {
   // Initialize the Xposed framework and modules
   try {
      if (!hadInitErrors()) {
         initXResources();
         SELinuxHelper.initOnce();
         SELinuxHelper.initForProcess(null);
         runtime = getRuntime();
         XPOSED_BRIDGE_VERSION = getXposedVersion();
         if (isZygote) {
            XposedInit.hookResources();
            XposedInit.initForZygote();
         }
         XposedInit.loadModules();
      } else {
         Log.e(TAG, "Not initializing Xposed because of previous errors");
      }
   } catch (Throwable t) {
      Log.e(TAG, "Errors during Xposed initialization", t);
      disableHooks = true;
   }
   // Call the original startup code
   if (isZygote) {
      ZygoteInit.main(args);
   } else {
      RuntimeInit.main(args);
   }
}
复制代码


主要是XposedInit.loadModules()这句,这个函数的代码:


static void loadModules() throws IOException {
   final String filename = BASE_DIR + "conf/modules.list";
   BaseService service = SELinuxHelper.getAppDataFileService();
   if (!service.checkFileExists(filename)) {
      Log.e(TAG, "Cannot load any modules because " + filename + " was not found");
      return;
   }
   ClassLoader topClassLoader = XposedBridge.BOOTCLASSLOADER;
   ClassLoader parent;
   while ((parent = topClassLoader.getParent()) != null) {
      topClassLoader = parent;
   }
   InputStream stream = service.getFileInputStream(filename);
   BufferedReader apks = new BufferedReader(new InputStreamReader(stream));
   String apk;
   while ((apk = apks.readLine()) != null) {
      loadModule(apk, topClassLoader);
   }
   apks.close();
}
复制代码


这里可以看到从modules.list文件中读取已激活模块的apk地址,然后执行loadModule将其加载进来,loadModule中会有一些判断,比如说是否有assets/xposed_init文件等等,如果都正常就会将这个apk的class加载进内存中,后续就会调用他们,实际上就是将模块的代码注入到xposedbridge中来使用。这部分代码就不细说了。


(实际上,真正hook都是在xposedbridge中的,如果我们直接在这里写hook代码也是可以的,只是这样就无法灵活修改了。所以我们将hook代码写在apk中,然后在这里将这些类加载进来,然后xposedbridge再调用它的hookLoadPackage之类的代码就可以了)


结论


所以通过上面我们可以知道,我们完全可以抛开xposedinstaller,当我们安装app并打开时,可以自己创建或修改modules.list文件,将自己app加进入就可以了。当然还要重启手机,从上面可以看到,因为加载模块的代码是在main函数中执行的,所以只有重启手机才会再次执行。


当然这个默认的modules.list我们可能需要root权限才能读写,但是如果我们自己二次开发xposed,可以在xposedbridge中将这个文件路径改成sd卡下可以任意读写的路径,这样我们的app就可以随意进行改写了。


或者我们提前准好这个文件手动放到手机中,文件内容是一个未安装的apk的路径,比如/sdcard/123.apk,这样不需要安装应该就可以直接使用。这个我没有测试,效果待验证。


目录
相关文章
|
Go Android开发 开发者
关于Xposed和Magisk的各方面比较(附Xposed框架各版本卡刷包及安装器)
说到搞机神器,不得不提江湖老大哥Xposed和后起之秀Magisk这两个框架,文章简单的介绍一下两者相似和不同之处
1589 0
|
2月前
|
Java 开发者
javafx jlink 遇到的非模块化的依赖打包报错“模块异常”的问题和处理
【9月更文挑战第18天】在使用JavaFX的jlink进行应用打包时,非模块化依赖可能导致“模块异常”报错。此文档详细分析了该问题的原因,并提供了四种解决方案:模块化依赖、自动模块转换、手动创建模块描述符及检查模块依赖关系。通过这些方法,可以有效解决此类问题,提高项目的可维护性和扩展性。建议开发者优先选用模块化设计。
205 1
|
6月前
|
编译器 异构计算 Windows
LabVIEWCompactRIO 开发指南27 创建模块化、可重复使用的子VI
LabVIEWCompactRIO 开发指南27 创建模块化、可重复使用的子VI
37 2
|
6月前
|
XML Java API
安卓逆向 -- Xposed模块编写
安卓逆向 -- Xposed模块编写
67 0
|
6月前
|
算法 关系型数据库 编译器
[项目配置] 配置Qt函数库和ui界面库的封装并调用的项目(一)
[项目配置] 配置Qt函数库和ui界面库的封装并调用的项目
187 0
|
6月前
|
C++
[项目配置] 配置Qt函数库和ui界面库的封装并调用的项目(二)
[项目配置] 配置Qt函数库和ui界面库的封装并调用的项目
102 0
CTK框架 - 通信 - 插件服务注册和调用
接口就是虚函数(也可以是纯虚函数),也就是最终的服务的前身。 接口对外暴露功能,比如我们给之前写的mainwindow 加一个界面类,并且对外暴露一个popWindow()的接口
131 0
|
开发者
Magisk模块:Shamiko模块 V0.5.0 更新:更好的隐藏 Zygisk
Magisk模块:Shamiko模块 V0.5.0 更新:更好的隐藏 Zygisk
4173 0
|
算法
【Magisk模块】自动内存优化模块v1.2
【Magisk模块】自动内存优化模块v1.2
865 0
Magisk模块:Scene附加模块(二)修改版
Magisk模块:Scene附加模块(二)修改版
4505 0