jar转成dex文件 dex文件转化为smali文件 java转化为class 打包/dx/d8 apksigner签名

82 篇文章 1 订阅

jar转成dex文件

dx

可以利用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.

方法数超过65535时的处理,分包

在现有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

限制每个dex包的最大方法数

在分包的基础上添加

--set-max-idx-number=48000

设置每个dex包的最大方法数为48000

dx --dex --multi-dex --set-max-idx-number=48000 --output . .\com.gsc.pub-r-classes.jar

指定SDK最小版本

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

d8是什么

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/。

d8工具在哪

D:\sdk\build-tools\28.0.3\lib

Android dx迁移到d8。

找到d8工具之后,可以根据d8类的一系列内置方法,完成jar文件到dex文件的转化。d8的相关操作可见,d8d8工具使用.
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
在这里插入图片描述在这里插入图片描述

指定SDK最小版本

指定您希望 DEX 输出文件支持的最低 API 级别。此API的等级可以理解为支持的最低Android版本。

Android 公开三个 Android API 级别项目设置:
目标框架
最低 Android 版本
目标 Android 版本

–classpath

使用d8命令行进行文件编译时,会对java8语言进行脱糖操作,进行脱糖操作会使用android.jar。
android.jar的路径如下:

D:\SDK\platforms\android-32

方法数超过65535时的处理

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版本分布情况。

d8常见问题

编译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>

在这里插入图片描述

dex文件转化为smali文件

使用baksmali.jar工具

此工具的链接

基本使用

java -jar baksmali.jar d .\bilibili1.dex

使用此指令后,可以在当前目录生成一个output文件,该文件中为dex转化的smali文件。
在这里插入图片描述
在这里插入图片描述

baksmali.jar 2.5.2版本

查看版本

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.

dex -> smali

java -jar baksmali-2.5.2.jar d .\bilibili1.dex

问题

.\classes.dex is not an apk, dex file or odex file

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\

jar转化为smali文件

  1. dx --dex --multi-dex --set-max-idx-number=48000 --min-sdk-version >= 26 --output .\dex .\jar*.jar
  2. java -jar baksmali-2.5.2.jar d .\dex\classes.dex

.java文件转化为.class文件

.java文件是给人看的,.class是给计算机看的。程序员编写的是.java文件,经编译后生成.class文件给计算执行。

javac

javac的基本使用

.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

apksigner签名

apk签名。
使用 release.jks(密钥库中唯一的密钥)为 APK 签名:

apksigner sign --ks release.jks app.apk

zipalign

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

dex2jar

d2j-dex2jar .\classes.dex

使用此指令后,将classes.dex转化为了classes-dex2jar.jar
在这里插入图片描述

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

学知识拯救世界

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值