如何实现不安装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,这样不需要安装应该就可以直接使用。这个我没有测试,效果待验证。


目录
相关文章
|
6月前
|
JSON 数据格式 Python
我们需要安装`authlib`模块(如果尚未安装):
我们需要安装`authlib`模块(如果尚未安装):
|
8月前
|
XML Java API
Android 浅度解析:系统框架层修改,编译,推送相关操作
Android 浅度解析:系统框架层修改,编译,推送相关操作
292 0
|
Java API Android开发
你有没有想过自己写一个Xposed模块?教程来了~(一)
在互联网上,关于Xposed模块编写的教程可谓是一抓一大把。但由于时间的推移,很多工具和方法都发生了变化(如Eclipse退出安卓编程舞台,AndroidStudio 不断升级导致其一些设置也随之变化等)也正因此,网上的教程往往有一些时限性,比如现如今 provide 这个关键字已经被舍弃了却仍有人在用,还有些说要把jar包放到lib文件夹而非libs文件夹……种种错误或者落伍的教程对新手产生了很大的误导。
399 0
|
开发者
Magisk模块:Shamiko模块 V0.5.0 更新:更好的隐藏 Zygisk
Magisk模块:Shamiko模块 V0.5.0 更新:更好的隐藏 Zygisk
4502 0
|
存储 Java Android开发
制作内置XPosed模块应用(详细教程,有手就会,无需Root即可使用)
众所周知,如今许多的安卓app,为了获利,为了赚钱,为了老板买别墅换豪车,加入了非常多的广告及推广内容以及软件模块。随着安卓的时代发展,Xposed的出现,在一定程度上改变了这种困境。
3871 0
|
Android开发
【Magisk模块】Hchai触控3.4Pro(经典重置)
【Magisk模块】Hchai触控3.4Pro(经典重置)
1359 0
|
Java Android开发
你有没有想过自己写一个Xposed模块?教程来了~(四)
在互联网上,关于Xposed模块编写的教程可谓是一抓一大把。但由于时间的推移,很多工具和方法都发生了变化(如Eclipse退出安卓编程舞台,AndroidStudio 不断升级导致其一些设置也随之变化等)也正因此,网上的教程往往有一些时限性,比如现如今 provide 这个关键字已经被舍弃了却仍有人在用,还有些说要把jar包放到lib文件夹而非libs文件夹……种种错误或者落伍的教程对新手产生了很大的误导。
132 0
|
Java API Android开发
你有没有想过自己写一个Xposed模块?教程来了~(二)
在互联网上,关于Xposed模块编写的教程可谓是一抓一大把。但由于时间的推移,很多工具和方法都发生了变化(如Eclipse退出安卓编程舞台,AndroidStudio 不断升级导致其一些设置也随之变化等)也正因此,网上的教程往往有一些时限性,比如现如今 provide 这个关键字已经被舍弃了却仍有人在用,还有些说要把jar包放到lib文件夹而非libs文件夹……种种错误或者落伍的教程对新手产生了很大的误导。
424 0
|
JSON Java atlas
动态部署、远程 bundle 、单模块部署演示 | 学习笔记
简介:快速学习动态部署、远程 bundle 、单模块部署演示
198 0
|
Android开发
【Android 插件化】VirtualApp 接入 ( 在 VirtualApp 工程下创建 Module | 添加依赖 | 启动 VirtualApp 插件引擎 )(二)
【Android 插件化】VirtualApp 接入 ( 在 VirtualApp 工程下创建 Module | 添加依赖 | 启动 VirtualApp 插件引擎 )(二)
220 0
【Android 插件化】VirtualApp 接入 ( 在 VirtualApp 工程下创建 Module | 添加依赖 | 启动 VirtualApp 插件引擎 )(二)

热门文章

最新文章