java - 如何将 Java 编译器和 DEX 转换器实现到 Android 应用程序中?

标签 java android dex

在尝试寻找 Android Jasper Reporting 的答案时我发现还有另外两个问题需要回答,我被要求作为问题而不是答案来问;):

我现在的问题是:“是否有任何编译器可以直接在设备上使用”和“如何在不对设备进行 root 操作的情况下执行这些编译器。 如果有人能给我一个提示,我将非常感激......


我对这种方法进行了一段时间的研究,发现了可以直接在未获得 root 权限的 Android 设备上创建 APK 的应用:

看起来他们正在使用来自 eclipse 的编译器和一个移植的 dex 转换器。现在我正试图弄清楚如何做同样的事情。

当然:获取源代码并研究它。但是,当我在连接到服务器并尝试解决它时遇到奇怪的问题时,我按照请求在这里提出这个问题。希望既能帮助别人,也能为自己找到答案;)


我从我的 indigo 的插件目录中取出了 org.eclipse.jdt.core_3.7.3.v20120119-1537.jar 并尝试了以下代码:

     org.eclipse.jdt.internal.compiler.batch.Main ecjMain = new org.eclipse.jdt.internal.compiler.batch.Main(new PrintWriter(System.out), new PrintWriter(System.err), false/*noSystemExit*/, null, progress);
     System.err.println("compiling...");
     ecjMain.compile(new String[] {"-classpath", "/system/framework", storage.getAbsolutePath()+"/Test.java"});
     ecjMain.compile(new String[] {storage.getAbsolutePath()+"/Test.java"});
     System.err.println("compile succeeded!!!");

有时会抛出找不到 java.lang.Object 的异常,有时它会在以 100% 使用率加热我的处理器时卡住什么也不做......

此时我无法弄清楚发生了什么以及为什么。由于我还有其他工作要做,这部分必须稍等一下。

最佳答案

我从 JavaIDEdroid 的来源获得灵感后成功了并意识到我很笨(有一段时间我尝试将编译器与设备上的 dexified 框架类一起使用——这自然无法工作)。
在我使用 SD 卡上的 ADT android-jar 副本成功编译我的 Test.java 后,我只需要使用 DexClassLoader 加载类。
在告诉自己如何做的同时,我发现了这篇不错的文章 Custom Class Loading in Dalvik 这至少启发了我编写这段代码:

    File storage = getDir("all41", Context.MODE_PRIVATE);


    System.err.println("copying the android.jar from asssets to the internal storage to make it available to the compiler");
    BufferedInputStream bis = null;
    OutputStream dexWriter = null;
    int BUF_SIZE = 8 * 1024;
    try {
          bis = new BufferedInputStream(getAssets().open("android.jar"));
          dexWriter = new BufferedOutputStream(
              new FileOutputStream(storage.getAbsolutePath() + "/android.jar"));
          byte[] buf = new byte[BUF_SIZE];
          int len;
          while((len = bis.read(buf, 0, BUF_SIZE)) > 0) {
              dexWriter.write(buf, 0, len);
          }
          dexWriter.close();
          bis.close();

    } catch (Exception e) {
        System.err.println("Error while copying from assets: " + e.getMessage());
        e.printStackTrace();
    }


    System.err.println("instantiating the compiler and compiling the java file"); 
    org.eclipse.jdt.internal.compiler.batch.Main ecjMain = new org.eclipse.jdt.internal.compiler.batch.Main(new PrintWriter(System.out), new PrintWriter(System.err), false/*noSystemExit*/, null);
    ecjMain.compile(new String[] {"-classpath", storage.getAbsolutePath()+"/android.jar", Environment.getExternalStorageDirectory().getAbsolutePath() + "/Test.java"});


    System.err.println("calling DEX and dexifying the test class"); 
    com.android.dx.command.Main.main(new String[] {"--dex", "--output=" + storage.getAbsolutePath() + "/Test.zip", Environment.getExternalStorageDirectory().getAbsolutePath() + "/./Test.class"});


    System.err.println("instantiating DexClassLoader, loading class and invoking toString()");
    DexClassLoader cl = new DexClassLoader(storage.getAbsolutePath() + "/Test.zip", storage.getAbsolutePath(), null, getClassLoader());
    try {
        Class libProviderClazz = cl.loadClass("Test");
        Object instance = libProviderClazz.newInstance();
        System.err.println(instance.toString());
    } catch (Exception e) {
        System.err.println("Error while instanciating object: " + e.getMessage());
        e.printStackTrace();
    }

Test.java 只包含一个方法:

public String toString() {
    return "Hallo Welt!";
}

要使其运行,您需要 jdt-compiler-x.x.x.jar(位于 eclipse 的插件目录中)和 dx.jar(位于 Android SDK 的 platform-tools/lib 目录中)


并不难 ;)
现在我将找出 JasperReports 源代码中需要更改的内容,以使其在我们心爱的 Android 设备上运行 :D

关于java - 如何将 Java 编译器和 DEX 转换器实现到 Android 应用程序中?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11223408/

相关文章:

java - Spring Security getPrincipal() 方法返回 anonymousUser

java - 如何使用 JSTL 检查 session 中是否存在登录用户?

java - 验证,如果无效则通知,返回循环

java - 简单的线程管理 - Java - Android

android - 创建自定义 XML 数据类型?

android - 为什么新的 Android 功能(Proto DataStore)仍在生成 Java 代码而不是 Kotlin?

java - 如何防止在 Android 应用程序中创建多个数据库?

java - 使用 dex2jar 将 .dex 文件转换为 .jar 文件

android - 多个dex文件定义lorg/apache/commons/io/IOUtils;

android - 在 Android Studio/gradle 中构建时,API 版本 23 中未调用 dx.bat