smali语法积累记录
1.constructor
我们知道运行一个类的时候会先调用static方法中的内容,比如:
static { System.loadLibrary("qihooTest"); }
为什么呢?
上面这段代码用smali语言写出来是这样的:
.method static constructor <clinit>()V #不带参数的构造方法 .locals 1 .prologue .line 64 const-string v0, "qihooTest" invoke-static {v0}, Ljava/lang/System;->loadLibrary(Ljava/lang/String;)V .line 65 return-void .end method
可以看到这个方法是作为构造方法出现的,而Java的构造方法在类运行时是要自动运行的。至于<cliinit>和<init>,clinit是初始化静态的类变量,init是初始化实例变量。
2.invoke-virtual和invoke-direct
在做360无线攻防第二题的时候,这个函数:
public void onClick(View paramView) { MainActivity.this.packageNameCheck(); }
原文件反编译出来后是:
invoke-direct {p0}, Lcom/qihoo/crack/MainActivity;->packageNameCheck()V
而看雪上人家修改后的这个函数反编译出来后,是:
invoke-virtual {p0}, Lcom/qihoo/crack/MainActivity;->packageNameCheck()V
一开始很疑惑,后来一想他们之间的不同,发现:
原来的类调用的packageNameCheck方法就在这个类中;而修改后的类继承了另外一个类,且packageNameCheck在父类中。
另外,书上写的:
invoke-virtual 或 invoke-virtual/range 调用实例的虚方法
invoke-super 或 invoke-super/range 调用实例的父类方法
invoke-direct 或 invoke-direct/range 调用实例的直接方法
由此猜想,invoke-virtual代表引用非本类中的方法(那invoke-super行不行呢?)。
9月25日补充,在调用so的过程中通过另一个so作媒介也是需要把direct改成virtual的,详见上面的题目链接。