如果不自己寫一個(gè)plugin插件的話旦万,可以現(xiàn)有開源庫比如Lancet
或者AspectJ
這兩個(gè)庫功能強(qiáng)大谜嫉,用于給方法套一層try...catch自然是輕而易舉萎坷,可以網(wǎng)上搜索下有很多的文章。
本文默認(rèn)你已經(jīng)熟悉了采用ASM實(shí)現(xiàn)gradle plugin 以及熟悉booster...
那么回到我們的話題上:自己如何使用ASM技術(shù)給方法套一層try...catch呢沐兰?
舉個(gè)例子哆档,如何給如下代碼中的printStr
方法套一層try...catch呢?
public class HookTest {
public void printStr(String s) {
Log.i("HookTest", s);
}
}
通過查看該方法對(duì)應(yīng)的字節(jié)碼:
// 沒有套try..catch
public printStr(Ljava/lang/String;)V
LDC "HookTest"
ALOAD 1
INVOKESTATIC android/util/Log.i (Ljava/lang/String;Ljava/lang/String;)I
POP
RETURN
MAXSTACK = 2
MAXLOCALS = 2
// 套try..catch之后
public printStr(Ljava/lang/String;)V
TRYCATCHBLOCK L0 L1 L2 java/lang/Exception
L0
LDC "HookTest"
ALOAD 1
INVOKESTATIC android/util/Log.i (Ljava/lang/String;Ljava/lang/String;)I
POP
L1
GOTO L3
L2
ASTORE 2
ALOAD 2
INVOKEVIRTUAL java/lang/Exception.printStackTrace ()V
L3
RETURN
可以看到插入之后主要是多了對(duì)應(yīng)的try...catch的字節(jié)碼 以及多了一些L0..3住闯。
開始編寫代碼:
仍然是基于booster
的框架進(jìn)行改造:
@AutoService(ClassTransformer::class)
class TestTryTransformer: ClassTransformer {
override fun transform(context: TransformContext, klass: ClassNode): ClassNode {
val className = klass.name
if (className == "com/remote/neacy/HookTest") {
val method = klass.methods.find {
"${it.name}${it.desc}" == "printStr(Ljava/lang/String;)V"
}
val start = LabelNode()
val end = LabelNode()
val catch = LabelNode()
val returnLabelNode = LabelNode()
// 定義try catch
val tryNode = TryCatchBlockNode(start, end, catch, "java/lang/Exception")
method?.tryCatchBlocks?.add(tryNode)
method?.instructions?.iterator()?.asIterable()?.filter {
it.opcode == Opcodes.RETURN
}?.forEach {
method.instructions?.apply {
// L0
insert(start)
// L1
insertBefore(it, end)
// goto L3
insertBefore(it, JumpInsnNode(Opcodes.GOTO, returnLabelNode))
// L2
insertBefore(it, catch)
insertBefore(it, VarInsnNode(Opcodes.ASTORE, 2))
insertBefore(it, VarInsnNode(Opcodes.ALOAD, 2))
// 調(diào)用方法 e.printStackTrace
insertBefore(it, MethodInsnNode(Opcodes.INVOKEVIRTUAL, "java/lang/Exception","printStackTrace", "()V", false))
insertBefore(it, returnLabelNode)
}
}
}
return super.transform(context, klass)
}
}
本文主要是基于ASM中的Tree api
瓜浸,參照字節(jié)碼寫起來還算順利。
最后比原,我們看下編譯出來的是不是想要的代碼:
public class HookTest {
public HookTest() {
}
public void printStr(String s) {
try {
Log.i("HookTest", s);
} catch (Exception var3) {
var3.printStackTrace();
}
}
}
沒錯(cuò)了插佛,就是我們想要的結(jié)果。
由于本文只是簡(jiǎn)單的demo量窘,如果想要在自己項(xiàng)目中使用 可以增加配置文件如xxx類.xxx方法 或者 是通過注解的方式找到要套try..catch雇寇。