3

我正在构建一个家庭自动化应用程序。我正在尝试添加一个插件系统。作为一个测试,我将一个测试类(即 Button 的子类)导出为 APK 文件并将其放在我的应用程序的文件目录中。我能够创建这个类的一个新实例,并使用 and 将它放在我的视图DexClassLoader.loadClass

下一步是扫描此目录中的所有 APK 并获取其中类的名称。

我发现 DexFile 类就是这样做的,但是它抛出了以下异常:

04-18 17:26:15.697: E/dalvikvm(726): Can't open dex cache '/data/dalvik-cache/data@data@com.strutton.android.testplugin@files@testloadclass.apk@classes.dex': No such file or directory

04-18 17:26:15.705: I/dalvikvm(726): Unable to open or create cache for /data/data/com.strutton.android.testplugin/files/testloadclass.apk (/data/dalvik-cache/data@data@com.strutton.android.testplugin@files@testloadclass.apk@classes.dex)

它似乎试图在系统缓存中找到优化的 DexFile,但我没有缓存目录的权限。据我所知,这可能是设计使然,我对此没有问题。是否有另一种方法来解析 DEX 文件并从中获取类名?

我查看了一些 dex 反编译器项目的源代码。我必须推出自己的解决方案吗?

这是我的测试应用程序代码(来自我的 Activity OnCreate),以防我错过了什么:

    try {
        ArrayList<String> UIPlugins = new ArrayList<String>();
        final File filesDir = this.getFilesDir();
        final File tmpDir = getDir("dex", 0);
        final DexClassLoader classloader = new DexClassLoader( filesDir.getAbsolutePath()+"/testloadclass.apk",
                tmpDir.getAbsolutePath(),
                null, this.getClass().getClassLoader());
        final Class<Button> classToLoad = 
                (Class<Button>) classloader.loadClass("com.strutton.android.testloadclass.MyTestClass_IRDroidUIPlugIn");
        Button mybutton = classToLoad.getDeclaredConstructor(Context.class).newInstance(this);
        mybutton.setId(2);
        mybutton.setOnClickListener(this);
        main.addView(mybutton);
        // Up to here everything works as expected
        // This line throws the exceptions
        DexFile mDexFile = new DexFile(filesDir.getAbsolutePath()+"/testloadclass.apk";);
        Enumeration<String> classNames = mDexFile.entries();
        while (classNames.hasMoreElements()){
            String className = classNames.nextElement();
            if (className.endsWith("IRDroidUIPlugIn")){
                UIPlugins.add(className);
            }
        }
        final Class<Button> classToLoad = 
                (Class<Button>) classloader.loadClass("com.strutton.android.testloadclass.MyTestClass_IRDroidUIPlugIn");
        Button mybutton = classToLoad.getDeclaredConstructor(Context.class).newInstance(this);
        mybutton.setId(2);
        mybutton.setOnClickListener(this);
        main.addView(mybutton);
        btn.setText(tmpDir.getAbsolutePath());
      } catch (Exception e) {
        e.printStackTrace();
    }
4

2 回答 2

4

代替:

DexFile mDexFile = new DexFile(filesDir.getAbsolutePath()+"/testloadclass.apk";);

尝试:

DexFile mDexFile = DexFile.loadDex(filesDir.getAbsolutePath()+"/testloadclass.apk", tmpDir.getAbsolutePath()+"/testloadclass.odex", 0);
于 2012-04-18T17:50:32.040 回答
0

您可能会考虑的另一种方法是将插件安装为单独的应用程序。您可以使用您定义的特定意图让插件应用程序公开服务。您还需要服务将实现的预定义aidl 接口。然后,在您的主应用程序中,您可以解析该服务意图,以查找任何插件服务。

于 2012-04-18T18:51:45.053 回答