- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我想创建一个应用程序,允许用户根据我提供的一些静态代码生成 APK。 Java 代码不会改变,我只需要修改 AndroidManifest.xml 和资源(修改 resources.arsc 或从头开始创建一个新的,因为我有可用的 R.java)。
资源的数量和名称不会改变,只是字符串内容和drawables内容有变化(意味着R.java不需要重新生成)
那么,假设我可以预编译所有内容并将其放入应用程序中,有没有办法在设备上没有 Java 编译器的情况下执行此操作?我可以将 aapt 和 aidl 二进制文件放入 APK。最后唱歌不是问题。
最佳答案
查看此应用:ThemeDIY , 你也可以查看演示 here .
此应用程序允许用户选择各种资源(drawable 资源、ttf 文件等)来自定义他们的主题,然后将它们编译和组装在一起以获得它们的 apk 文件。
它使用 kellinwood 的 ZipSigner用于签署 apk 和 lingala Zip4j 的库用于创建 zip 文件。
我想这和你想做的一样,选择/修改资源(保持资源名称相同)并生成 apk。
因此,您的解决方案是在文件夹中准备好静态内容(即 apk 的内容,如 list 文件、jar 文件、dex 文件、res 文件夹等),创建一个副本作为临时文件夹,修改根据用户需要资源内容(而不是名称),然后最终创建此临时文件夹的 zip 并将其签名以获得所需的 .apk 文件(那时不需要 java 编译器)。
希望对您有所帮助!
好的,在评论中讨论后,我想出了以下(详细的整个过程如下所述):
我们需要从我们的应用程序创建一个 apk(在修改资源之后)。因此,apk 生成所需的东西:
所需工具:
除第一步外,以下所有步骤均在应用程序运行时在 android 中完成。
dx --dex --output=classes.dex SomeJar1.jar SomeJar2.jar ..
这些 jar 是您编译的类(此步骤可以在 windows/linux 中完成,因为这些是静态内容)。
现在,在运行时将我们生成 apk 所需的东西放入 android 应用程序缓存目录的临时文件夹中:
getApplictionContext().getCacheDir() + "/tmp/"
将classes.dex、AndroidManifest.xml 和res/文件夹复制到上面的文件夹中。
(您可能希望最初将 classes.dex、AndroidManifest.xml 和 res 文件夹保留在应用程序的原始文件夹中,然后在运行时使用 FileInputStream 类将其复制(就像我们在 java 中所做的那样)到上面的 tmp 文件夹中。)
现在,根据需要修改此缓存的 tmp 目录中的 res 文件夹。
修改后,是时候打包初始 apk 了,我们需要从我们的 android 应用程序运行以下命令:
aapt package -f -M AndroidManifest.xml -S res/ -I android.jar -F MyProject.apk.unaligned
因此,代码现在变成了 android 的代码:
StringBuilder sb= new StringBuilder();
sb.append("aapt package -f -M");
sb.append(getApplicationContext().getCacheDir());
sb.append("/tmp/AndroidManifest.xml -S ");
sb.append(getApplicationContext().getCacheDir());
sb.append("/tmp/res -I ");
sb.append(getApplicationContext().getCacheDir());
sb.append("/tmp/android.jar -F ");
sb.append(getApplicationContext().getCacheDir());
sb.append("/apk/MyProject.apk.unaligned");
sb = Runtime.getRuntime().exec(sb.toString());
所以,现在我们有了带有 resources.arsc 的初始 apk 和这个 apk 中的加密 list 。但是我们需要添加 classes.dex 文件,所以我们运行以下命令:
aapt add -f MyProject.apk.unaligned classes.dex
从 android 我们写:
StringBuilder sb = new StringBuilder();
sb.append("aapt add -f");
sb.append(getApplicationContext().getCacheDir());
sb.append("/apk/MyProject.apk.unaligned ");
sb.append(getApplicationContext().getCacheDir());
sb.append("/tmp/classes.dex");
sb = Runtime.getRuntime().exec(sb.toString());
对包进行签名,使用命令:
jarsigner -storepass <keystore password> -keystore <keystore filename> MyProject.apk.unaligned <key name>
(可以转成android代码,同上)
最后,对齐它,使用命令:
zipalign 4 MyProject.apk.unaligned MyProject.apk
(同样,这可以转换为android代码,同上)
所以,现在我们已经完全准备好 android apk,可以根据需要安装它或将其复制到 sd 卡。
上面的所有代码都已经过测试并且工作正常!
希望这对您有所帮助!
关于java - 修改或生成Android resources.arsc和编译 list ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40545028/
我最近在/ drawable中添加了一些.gifs,以便可以将它们与按钮一起使用。这个工作正常(没有错误)。现在,当我重建/运行我的应用程序时,出现以下错误: Error: Gradle: Execu
Android 中有返回内部存储数据路径的方法吗? 我有 2 部 Android 智能手机(Samsung s2 和 s7 edge),我在其中安装了一个应用程序。我想使用位于这条路径中的 sqlit
这个问题在这里已经有了答案: What's the difference between "?android:" and "@android:" in an android layout xml f
我只想知道 android 开发手机、android 普通手机和 android root 手机之间的实际区别。 我们不能从实体店或除 android marketplace 以外的其他地方购买开发手
自Gradle更新以来,我正在努力使这个项目达到标准。这是一个团队项目,它使用的是android-apt插件。我已经进行了必要的语法更改(编译->实现和apt->注释处理器),但是编译器仍在告诉我存在
我是android和kotlin的新手,所以请原谅要解决的一个非常简单的问题! 我已经使用导航体系结构组件创建了一个基本应用程序,使用了底部的导航栏和三个导航选项。每个导航选项都指向一个专用片段,该片
我目前正在使用 Facebook official SDK for Android . 我现在正在使用高级示例应用程序,但我不知道如何让它获取应用程序墙/流/状态而不是登录的用户。 这可能吗?在那种情
我在下载文件时遇到问题, 我可以在模拟器中下载文件,但无法在手机上使用。我已经定义了上网和写入 SD 卡的权限。 我在服务器上有一个 doc 文件,如果用户单击下载。它下载文件。这在模拟器中工作正常但
这个问题在这里已经有了答案: What is the difference between gravity and layout_gravity in Android? (22 个答案) 关闭 9
任何人都可以告诉我什么是 android 缓存和应用程序缓存,因为当我们谈论缓存清理应用程序时,它的作用是,缓存清理概念是清理应用程序缓存还是像内存管理一样主存储、RAM、缓存是不同的并且据我所知,缓
假设应用程序 Foo 和 Eggs 在同一台 Android 设备上。任一应用程序都可以获取设备上所有应用程序的列表。一个应用程序是否有可能知道另一个应用程序是否已经运行以及运行了多长时间? 最佳答案
我有点困惑,我只看到了从 android 到 pc 或者从 android 到 pc 的例子。我需要制作一个从两部手机 (android) 连接的 android 应用程序进行视频聊天。我在想,我知道
用于使用 Android 以编程方式锁定屏幕。我从 Stackoverflow 之前关于此的问题中得到了一些好主意,并且我做得很好,但是当我运行该代码时,没有异常和错误。而且,屏幕没有锁定。请在这段代
文档说: android:layout_alignParentStart If true, makes the start edge of this view match the start edge
我不知道这两个属性和高度之间的区别。 以一个TextView为例,如果我将它的layout_width设置为wrap_content,并将它的width设置为50 dip,会发生什么情况? 最佳答案
这两个属性有什么关系?如果我有 android:noHistory="true",那么有 android:finishOnTaskLaunch="true" 有什么意义吗? 最佳答案 假设您的应用中有
我是新手,正在尝试理解以下 XML 代码: 查看 developer.android.com 上的文档,它说“starStyle”是 R.attr 中的常量, public static final
在下面的代码中,为什么当我设置时单选按钮的外观会发生变化 android:layout_width="fill_parent" 和 android:width="fill_parent" 我说的是
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 9
假设我有一个函数 fun myFunction(name:String, email:String){},当我调用这个函数时 myFunction('Ali', 'ali@test.com ') 如何
我是一名优秀的程序员,十分优秀!