赞
踩
可以利用android studio中的dx工具。
可以看到android的tool安装位置:
在此路径下的如下目录有dx.bat,这个正是我们需要使用的工具。
D:\sdk\build-tools\30.0.3
将dx.bat添加到环境变量
> dx --dex --output 输出路径 待转化的jar包
C:\Users\liyd\Desktop\快手> dx --dex --output dex1.dex .\com.gsc.pub-r-classes.jar
此时如果jar包的方法数过多,可能会报如下错误。
trouble writing output: Too many field references to fit in one dex file: 98482; max is 65536.
You may try using multi-dex. If multi-dex is enabled then the list of classes for the main dex list is too large.
在现有dx编译指令上添加
–multi-dex
dx --dex --multi-dex --output ./dex .\com.gsc.pub-r-classes.jar
dx --dex --multi-dex --output dex编译后的路径 待编译的jar包或者文件夹
dx --dex --multi-dex --main-dex-list=multidex-keep.txt --minimal-main-dex --output ../release ../release
在分包的基础上添加
--set-max-idx-number=48000
设置每个dex包的最大方法数为48000
dx --dex --multi-dex --set-max-idx-number=48000 --output . .\com.gsc.pub-r-classes.jar
dx --dex --multi-dex --set-max-idx-number=48000 --min-sdk-version >= 26 --output .\dex .\jar\*.jar
dx编译时,可能存在的问题。
Uncaught translation error: com.android.dx.cf.code.SimException: ERROR in com.gsc.base.utils.CaptureUtils.screenshotV26:(Landroid/app/Activity;Lcom/gsc/base/interfaces/Callback;)V: invalid opcode ba - invokedynamic requires --min-sdk-version >= 26 (currently 13)
此问题要求dx中声明 min-sdk-version>=26,声明方式如下:
dx --dex --multi-dex --set-max-idx-number=48000 --min-sdk-version >= 26 --output .\dex .\jar\*.jar
需要确认的是,这个min-sdk-version对应的是什么版本。android中有三个版本,dx命令中的版本对应的是其中的哪个?
进行版本声明后,在dex转化为smali文件时是否会有问题。
d8是将jar、class等转化为dex文件的工具类。
在android studio中可以找到d8.jar,路径如下:
d8 是一种命令行工具,Android Studio 和 Android Gradle 插件使用该工具来将项目的 Java 字节码编译为在 Android 设备上运行的 DEX 字节码。d8 支持您在应用的代码中使用 Java 8 语言功能。
d8 还作为独立工具纳入了 Android 构建工具 28.0.1 及更高版本中:android_sdk/build-tools/version/。
D:\sdk\build-tools\28.0.3\lib
找到d8工具之后,可以根据d8类的一系列内置方法,完成jar文件到dex文件的转化。d8的相关操作可见,d8,d8工具使用.
d8示例如下:
java -jar d8.jar --output .\dex\ .\jar\*.jar --lib .\android.jar
将d8添加到环境变量后,语法为:
d8 --output .\dexd8\ .\jar\*.jar --lib .\android.jar
D:\work\tkbackground\jjchannelpackage\ToolConfigPath\tools\dx
指定您希望 DEX 输出文件支持的最低 API 级别。此API的等级可以理解为支持的最低Android版本。
Android 公开三个 Android API 级别项目设置:
目标框架
最低 Android 版本
目标 Android 版本
使用d8命令行进行文件编译时,会对java8语言进行脱糖操作,进行脱糖操作会使用android.jar。
android.jar的路径如下:
D:\SDK\platforms\android-32
d8操作时,Try supplying a main-dex list. fields: 125587 > 65536
Android 5.0(API 级别 21)及更高版本使用名为 ART 的运行时,它本身支持从 APK 文件加载多个 DEX 文件。ART在应用安装时执行预编译,这会扫描查找 classesN.dex 文件,并将它们编译成单个 OAT 文件,以供 Android设备执行。因此,如果您的 minSdkVersion 为 21 或更高版本,系统会默认启用 MultiDex,并且您不需要 MultiDex 库。
从这句话,结合–min-api指令,可以尝试编译jar文件时申明最低版本的api,此时指令如下:
d8 .\com.gsc.pub-r-classes.jar --min-api 21 --output .\d8\ --lib .\android.jar
实践中,该方法可以实现单d8命令行进行dex包分包。而Android API 21对应的版本为Android5.0,最低版本也适用于当前的Android版本分布情况。
编译class文件时,使用javac直接编译
javac .\Hello.java
后续使用d8指令编译生成的class文件,提示
d8 .\Hello.class
获得下面报错
PS C:\Users\liyd\Desktop\1> d8 .\Hello.class
Compilation failed with an internal error.
java.lang.IllegalArgumentException
at com.android.tools.r8.org.objectweb.asm.ClassReader.<init>(ClassReader.java:160)
at com.android.tools.r8.org.objectweb.asm.ClassReader.<init>(ClassReader.java:143)
at com.android.tools.r8.org.objectweb.asm.ClassReader.<init>(ClassReader.java:418)
at com.android.tools.r8.graph.JarClassFileReader.read(JarClassFileReader.java:66)
at com.android.tools.r8.dex.ApplicationReader$ClassReader.lambda$readClassSources$1(ApplicationReader.java:231)
at java.util.concurrent.ForkJoinTask$AdaptedCallable.exec(Unknown Source)
at java.util.concurrent.ForkJoinTask.doExec(Unknown Source)
at java.util.concurrent.ForkJoinPool$WorkQueue.runTask(Unknown Source)
at java.util.concurrent.ForkJoinPool.runWorker(Unknown Source)
at java.util.concurrent.ForkJoinWorkerThread.run(Unknown Source)
解决此问题,需设置javac编译时的版本:
javac -source 8 -target 8 .\Hello.java
紧接着使用d8指令就能成功
PS C:\Users\liyd\Desktop\1> javac -source 8 -target 8 .\Hello.java
警告: [options] 未与 -source 8 一起设置引导类路径
1 个警告
PS C:\Users\liyd\Desktop\1> d8 .\Hello.class
PS C:\Users\liyd\Desktop\1>
此工具的链接。
java -jar baksmali.jar d .\bilibili1.dex
使用此指令后,可以在当前目录生成一个output文件,该文件中为dex转化的smali文件。
java -jar baksmali-2.5.2.jar --version
输出:
baksmali 2.5.2 (http://smali.org)
Copyright (C) 2010 Ben Gruver (JesusFreke@JesusFreke.com)
BSD license (http://www.opensource.org/licenses/bsd-license.php)
java -jar baksmali-2.5.2.jar help
输出:
usage: baksmali [--version] [--help] [<command [<args>]]
Options:
--help,-h,-? - Show usage information
--version,-v - Print the version of baksmali and then exit
Commands:
deodex(de,x) - Deodexes an odex/oat file
disassemble(dis,d) - Disassembles a dex file.
dump(du) - Prints an annotated hex dump for the given dex file
help(h) - Shows usage information
list(l) - Lists various objects in a dex file.
java -jar baksmali-2.5.2.jar d .\bilibili1.dex
Exception in thread "main" org.jf.util.ExceptionWithContext: .\classes.dex is not an apk, dex file or odex file.
java -jar baksmali-2.5.2.jar d .\bilibili1.dex -o .\out\
- dx --dex --multi-dex --set-max-idx-number=48000 --min-sdk-version >= 26 --output .\dex .\jar*.jar
- java -jar baksmali-2.5.2.jar d .\dex\classes.dex
.java文件是给人看的,.class是给计算机看的。程序员编写的是.java文件,经编译后生成.class文件给计算执行。
.java转化为.class使用到了javac工具。
PS C:\Users\liyd\Desktop\快手> javac -help
用法: javac <options> <source files>
其中, 可能的选项包括:
-g 生成所有调试信息
-g:none 不生成任何调试信息
-g:{lines,vars,source} 只生成某些调试信息
-nowarn 不生成任何警告
-verbose 输出有关编译器正在执行的操作的消息
-deprecation 输出使用已过时的 API 的源位置
-classpath <路径> 指定查找用户类文件和注释处理程序的位置
-cp <路径> 指定查找用户类文件和注释处理程序的位置
-sourcepath <路径> 指定查找输入源文件的位置
-bootclasspath <路径> 覆盖引导类文件的位置
-extdirs <目录> 覆盖所安装扩展的位置
-endorseddirs <目录> 覆盖签名的标准路径的位置
-proc:{none,only} 控制是否执行注释处理和/或编译。
-processor <class1>[,<class2>,<class3>...] 要运行的注释处理程序的名称; 绕过默认的搜索进程
-processorpath <路径> 指定查找注释处理程序的位置
-parameters 生成元数据以用于方法参数的反射
-d <目录> 指定放置生成的类文件的位置
-s <目录> 指定放置生成的源文件的位置
-h <目录> 指定放置生成的本机标头文件的位置
-implicit:{none,class} 指定是否为隐式引用文件生成类文件
-encoding <编码> 指定源文件使用的字符编码
-source <发行版> 提供与指定发行版的源兼容性
-target <发行版> 生成特定 VM 版本的类文件
-profile <配置文件> 请确保使用的 API 在指定的配置文件中可用
-version 版本信息
-help 输出标准选项的提要
-A关键字[=值] 传递给注释处理程序的选项
-X 输出非标准选项的提要
-J<标记> 直接将 <标记> 传递给运行时系统
-Werror 出现警告时终止编译
@<文件名> 从文件读取选项和文件名
转化命令如下图所示:
转化当前目录下的R.java文件,转化后的文件放在当前路径下的class文件夹下,并且当前处理文件即R.java文件的编码类型为UTF-8。
javac .\R.java -d .\class\ -encoding UTF-8
使用javac进行转化时,可能会报以下错误:
PS C:\Users\liyd\Desktop\快手> javac .\WXPayEntryActivity.java -d .\class\ -encoding UTF-8
.\WXPayEntryActivity.java:3: 错误: 程序包com.bsgamesdk.android.activity不存在
import com.bsgamesdk.android.activity.BaseWXEntryActivity;
^
.\WXPayEntryActivity.java:5: 错误: 找不到符号
public class WXPayEntryActivity extends BaseWXEntryActivity{
此时需要指定我们需要编译文件中的 java 依赖库,指令如下:
-cp .\lib
javac .\R.java -cp .\lib -d .\class\ -encoding UTF-8
apk签名。
使用 release.jks(密钥库中唯一的密钥)为 APK 签名:
apksigner sign --ks release.jks app.apk
zipalign 是一种 zip 归档文件对齐工具,有助于确保归档文件中的所有未压缩文件相对于文件开头对齐。这样一来,您便可直接通过 mmap(2) 访问这些文件,而无需在 RAM 中复制这些数据并减少了应用的内存用量。
注意:您必须在构建流程的特定时间点使用 zipalign。该时间点取决于您使用的应用签名工具:
如果您使用的是 apksigner,则必须在为 APK 文件签名之前使用 zipalign。如果您在使用 apksigner 为 APK 签名之后对 APK 做出了进一步更改,签名便会失效。
如果您使用的是 jarsigner(不推荐),则必须在为 APK 文件签名之后使用 zipalign。
如果您的 APK 包含共享库(.so 文件),请使用 -p 来确保它们与适合 mmap(2) 的 4KiB 页面边界对齐。对于其他文件(其对齐方式由 zipalign 的必选对齐参数确定),Android Studio 将在 32 位和 64 位系统中对齐到 4 个字节。
如需对齐 infile.apk 并将其保存为 outfile.apk,请运行以下命令:
zipalign -p -f -v 4 infile.apk outfile.apk
如需确认 existing.apk 的对齐情况,请使用以下命令。如果您使用 Android Studio 或 AGP 进行构建,则应使用该命令来验证 APK 是否已对齐。
zipalign -c -v 4 existing.apk
d2j-dex2jar .\classes.dex
使用此指令后,将classes.dex转化为了classes-dex2jar.jar
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。