当前位置:   article > 正文

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

jar转dex

jar转成dex文件

dx

可以利用android studio中的dx工具。
在这里插入图片描述
可以看到android的tool安装位置:
在此路径下的如下目录有dx.bat,这个正是我们需要使用的工具。

D:\sdk\build-tools\30.0.3
  • 1

在这里插入图片描述
将dx.bat添加到环境变量

基本指令

> dx  --dex --output 输出路径 待转化的jar包
  • 1
 C:\Users\liyd\Desktop\快手> dx --dex --output dex1.dex .\com.gsc.pub-r-classes.jar
  • 1

此时如果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.
  • 1
  • 2

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

在现有dx编译指令上添加

–multi-dex

dx --dex --multi-dex --output ./dex .\com.gsc.pub-r-classes.jar
  • 1
dx --dex --multi-dex --output dex编译后的路径  待编译的jar包或者文件夹
  • 1
dx --dex --multi-dex --main-dex-list=multidex-keep.txt --minimal-main-dex --output ../release ../release
  • 1

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

在分包的基础上添加

--set-max-idx-number=48000
  • 1

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

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

指定SDK最小版本

dx --dex --multi-dex --set-max-idx-number=48000 --min-sdk-version >= 26 --output .\dex .\jar\*.jar
  • 1

最小版本问题

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)
  • 1

此问题要求dx中声明 min-sdk-version>=26,声明方式如下:

dx --dex --multi-dex --set-max-idx-number=48000 --min-sdk-version >= 26 --output .\dex .\jar\*.jar
  • 1

需要确认的是,这个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
  • 1

Android dx迁移到d8。

找到d8工具之后,可以根据d8类的一系列内置方法,完成jar文件到dex文件的转化。d8的相关操作可见,d8d8工具使用.
d8示例如下:

java -jar d8.jar --output .\dex\ .\jar\*.jar --lib .\android.jar
  • 1

将d8添加到环境变量后,语法为:

d8 --output .\dexd8\ .\jar\*.jar --lib .\android.jar
  • 1

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
  • 1

方法数超过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
  • 1

实践中,该方法可以实现单d8命令行进行dex包分包。而Android API 21对应的版本为Android5.0,最低版本也适用于当前的Android版本分布情况。

d8常见问题

编译class文件时,使用javac直接编译

javac .\Hello.java
  • 1

后续使用d8指令编译生成的class文件,提示

d8 .\Hello.class
  • 1

获得下面报错

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)
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13

解决此问题,需设置javac编译时的版本:

javac -source 8 -target 8 .\Hello.java
  • 1

紧接着使用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>
  • 1
  • 2
  • 3
  • 4
  • 5

在这里插入图片描述

dex文件转化为smali文件

使用baksmali.jar工具

此工具的链接

基本使用

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

使用此指令后,可以在当前目录生成一个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)
  • 1
  • 2
  • 3
  • 4
  • 5

查看命令

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.
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

dex -> smali

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

问题

.\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.
  • 1

指定路径

java -jar baksmali-2.5.2.jar d .\bilibili1.dex -o .\out\
  • 1

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                    出现警告时终止编译
  @<文件名>                     从文件读取选项和文件名
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34

转化命令如下图所示:
转化当前目录下的R.java文件,转化后的文件放在当前路径下的class文件夹下,并且当前处理文件即R.java文件的编码类型为UTF-8。

javac .\R.java -d .\class\ -encoding UTF-8
  • 1

使用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{
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

此时需要指定我们需要编译文件中的 java 依赖库,指令如下:

-cp .\lib

javac .\R.java -cp .\lib -d .\class\ -encoding UTF-8
  • 1

apksigner签名

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

apksigner sign --ks release.jks app.apk
  • 1

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
  • 1

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

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/weixin_40725706/article/detail/789764
推荐阅读
相关标签
  

闽ICP备14008679号