Как внедрить Java-компилятор и DEX-конвертер в приложение для Android?
Пытаясь найти ответ на Android Jasper Reporting, я обнаружил, что есть еще два вопроса, на которые мне нужно ответить, как вопрос, а не как ответ;):
Мои вопросы теперь: "Есть ли какой-нибудь компилятор, который можно использовать непосредственно на устройстве?" И "как выполнить такое без рутирования устройства. Если кто-нибудь может дать мне подсказку, я был бы очень признателен…"
Я немного подумал об этом подходе и нашел приложения, которые позволяют создавать APK-файлы непосредственно на Android-устройстве, которое НЕ имеет root-прав:
- TerminalIDE - https://play.google.com/store/apps/details?id=com.spartacusrex.spartacuside&hl=de
- JavaIDEdroid - http://code.google.com/p/java-ide-droid/
- AIDE - https://play.google.com/store/apps/details?id=com.aide.ui&hl=en
Похоже, они используют компилятор из eclipse и перенесенный dex конвертер. Сейчас я пытаюсь понять, как сделать то же самое.
Конечно: получите исходный код и посмотрите на него. Но хотя у меня возникают любопытные проблемы с подключением к серверам и я пытаюсь их решить, я следую просьбе задать этот вопрос здесь. Надеясь как помочь другим с этим, а также получить ответ для себя;)
Я взял 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% использованием......
В это время я не мог понять, что происходит и почему. И потому, что у меня есть другая работа, чтобы сделать эту часть, нужно немного подождать.
1 ответ
Я преуспел после того, как черпал вдохновение из источника JavaIDEdroid и понял, что я тупой (какое-то время я пытался использовать компилятор с классами dexified framework на устройстве - что, естественно, не могло работать).
После того, как мне удалось скомпилировать мой Test.java с копией ADT android-jar на SDCard, мне просто нужно было загрузить классы с помощью DexClassLoader.
Сообщая myselft о том, как это сделать, я нашел эту замечательную статью Custom Class Loading в 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!";
}
Чтобы запустить его, вам нужны jars jdt-compiler-xxxjar (находится в каталоге плагинов eclipse) и dx.jar (находится в каталоге platform-tools/lib Android SDK)
Не очень сложно;)
А теперь я выясню, что нужно изменить в источнике JasperReports, чтобы он работал на наших любимых устройствах Android:D