基於booster給方法套層try..catch (Demo)

如果不自己寫一個plugin插件的話,可以現有開源庫比如Lancet或者AspectJ這兩個庫功能強大,用於給方法套一層try...catch自然是輕而易舉,可以網上搜索下有很多的文章。

本文默認你已經熟悉了採用ASM實現gradle plugin 以及熟悉booster...

那麼回到我們的話題上:自己如何使用ASM技術給方法套一層try...catch呢?
舉個例子,如何給如下代碼中的printStr方法套一層try...catch呢?

public class HookTest {

    public void printStr(String s) {
        Log.i("HookTest", s);
    }
}

通過查看該方法對應的字節碼:

     // 沒有套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

可以看到插入之後主要是多了對應的try...catch的字節碼 以及多了一些L0..3。

開始編寫代碼:
仍然是基於booster的框架進行改造:

@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))
                    // 調用方法 e.printStackTrace
                    insertBefore(it, MethodInsnNode(Opcodes.INVOKEVIRTUAL, "java/lang/Exception","printStackTrace", "()V", false))

                    insertBefore(it, returnLabelNode)
                }
            }
        }
        return super.transform(context, klass)
    }
}

本文主要是基於ASM中的Tree api,參照字節碼寫起來還算順利。
最後,我們看下編譯出來的是不是想要的代碼:

public class HookTest {
    public HookTest() {
    }
    public void printStr(String s) {
        try {
            Log.i("HookTest", s);
        } catch (Exception var3) {
            var3.printStackTrace();
        }
    }
}

沒錯了,就是我們想要的結果。

由於本文只是簡單的demo,如果想要在自己項目中使用 可以增加配置文件如xxx類.xxx方法 或者 是通過註解的方式找到要套try..catch。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章