注意:本篇是一个以方法论为导向的文章。java
Q1:Smali是什么。android
Smali是一种宽松式的Jasmin/dedexer语法.git
简单来讲就是咱们用java写的代码编译成class打包成dex文件后使用baksmali程序逆向回来的一种语法。github
Q2:为何要学习Smali。
首先,提到smali就不得不说逆向。早在尚未android以前,各大平台和语言上就有对应的逆向一说。那么到目前为止,逆向一个apk一般是安全工程师(逆向工程师)和作破解等恶意分子由于某些利益在作(apk二次打包插入广告、破解收费应用、恶意代码植入、剽窃api等)。api
技术是一把双刃剑,怎么用在人。而不在技术自己上。那么咱们说为何应用层开发者也要学smali呢?我能想到如下几点供参考。数组
1.借鉴 当咱们发现其余应用有一个很牛逼功能,而咱们想不明白如何实现的时候。拿不到源码能够选择逆向。
2.安全 咱们写的app须要考虑安全性,可是咱们可能只知道混淆和第三方加固,须要明白别人是怎么破解咱们的应用。
3.适配 当咱们发现api在某些手机上被弃用,而其余应用或系统应用又能实现该功能的时候。关于这点我以前写过一篇逆向小米作适配的文章。安全
喂,差很少够了吧?还不能打动你学吗?给你升职加薪怎么样? :)bash
噗,坏蛋!!!,我学 我学,还不行吗?app
Q3:Smali难不难?
不难。也许你很早以前看过一些文章。或者也经常使用一些工具去打开反编译后的代码。看着一团麻的指令和一些你从未见过的关键字、代码格式风格,赖不住性子就潦草的关掉了。但其实是你没有找对方法来学习它。ide
Q4:怎么学
我一贯的风格都是不爱把知识生拉硬套的往脑子里塞,我更加习惯从实践中去分析,然后反过来作总结。如今给你们推荐一款好用的Smali学习工具插件。咱们打开AndroidStudio找到插件安装的位置。以下图
这个步骤之后,咱们就能够愉快的将任何java代码在androidStudio中直接转换成smali来学习里。步骤以下。
1.编写一个最简单的java文件。好比下面这样的。
而后咱们点击Build->Compile to smali
稍等几秒钟后就会获得smali文件。
接下来咱们就能够对照着java代码来逐行分析这个smali文件。若是是第一次看咱们可能会被一些没见过关键字干扰到。其实这里有个很简单的办法。注意.line关键字就是用来描述当前代码在java源文件中的行数。而后你能够经过对照两组代码的方法进行反推。这样就能够很轻松的学会看smali文件。
好下面是一个示例代码,供参考。
原java代码
public AA methodAReturn(AA mAA, AA sAA) {
return mAA;
}
AA aa= new AA();
//调用
methodAReturn(aa, aa);复制代码
Smali代码
.method public methodAReturn(Lcom/bolex/AA;Lcom/bolex/AA;)Lcom/bolex/AA;
.registers 3
.param p1, "mAA" # Lcom/bolex/AA;
.param p2, "sAA" # Lcom/bolex/AA;
.prologue
.line 34
return-object p1
.end method
.line 21
new-instance v0, Lcom/bolex/AA;
invoke-direct {v0}, Lcom/bolex/AA;-><init>()V
.line 22
invoke-virtual {p0, v0, v0}, Lcom/bolex/seamAct;->methodAReturn(Lcom/bolex/AA;Lcom/bolex/AA;)Lcom/bolex/AA;复制代码
.line 34复制代码
表示当前代码在源java文件中的行数。
.method public methodAReturn(Lcom/bolex/AA;Lcom/bolex/AA;)Lcom/bolex/AA;复制代码
表示来自公共方法methodAReturn返回值是一个对象com.bolex.AA
.registers 3复制代码
表示该函数上须要使用3个寄存器
.param p1, "mAA" # Lcom/bolex/AA;
.param p2, "sAA" # Lcom/bolex/AA;复制代码
表示接收两个入参都是AA对象,并标记寄存器p1和p2
.prologue复制代码
表示函数内执行的起始标记。直译为开场白的意思。
.line 34复制代码
表示在源代码中的第34行。
return-object p1复制代码
表示 返回寄存器上p1对象
.end method复制代码
表示函数结束标记
new-instance v0, Lcom/bolex/AA;复制代码
建立一个AA对象
invoke-direct {v0}, Lcom/bolex/AA;-><init>()V复制代码
表示使用无参构造方法直接调用
invoke-virtual {p0, v0, v0}, Lcom/bolex/seamAct;->methodAReturn(Lcom/bolex/AA;Lcom/bolex/AA;)Lcom/bolex/AA;复制代码
表示为虚拟方法
就是这个样子的,有没有很简单呢?
以上只举例了部分关键字,更多的关键字能够自行依赖两组文件反推。其实有时候更加讲究的是一个方法。我以为这个方法就挺不错的,因此就分享给你们咯,咱也不须要刻意去背下来。熟能生巧,玩多了岂能不是老司机?
关于smali的知识还有不少本文并未详细阐述,如寄存器、类型(原始类型、对象类型)、数组方法的表示形式。如读者还须要进一步深刻挖。能够参考官方文档。里面有详细的解释,已翻译成中文版了。
source.android.com/devices/tec…
本文发自掘金&简书-香脆的大鸡排。原创文章,未受权禁止转载。