Android Transform+ASM添加try catch并返回默认值之明白讲义

Transform这个部分这里不作讲解,直接使用hunter库就行,里面对Transform遍历处理class文件都做好了封装,我们继承实现对应的方法就行。在build.gradle的dependencies里面添加依赖:

   // 使用hunter框架
    implementation('com.quinn.hunter:hunter-transform:0.9.3') {
        // 排除hunter带来的gradle传递依赖,以便自定义应用的gradle版本
        exclude group: 'com.android.tools.build'
    }

本文源代码:DxKit一个基于ASM的开发工具集:https://github.com/Dawish/DxKit

1. 查看try catch的生成原理

我们使用ASM Bytecode Viewer插件查看一个简单的try catch方法对应的ASM代码。在使用ASM Bytecode Viewer之前,在设置中记得把skip debug和skip frames勾选上,不勾选上,可能对产生一堆我们写ASM代码用不上的frame相关操作,我们自己在写ASM代码时,不建议直接操作frame。

ASM_Bytecode_Viewer设置.png

java源码:

       public int tryTest() {
        try {
            //1 try里面的方法执行
            int i = 3 / 0;
            //2 try里面正常,返回结果值
            return i;
        } catch (Exception e) {
            //3 发生异常
            ExceptionHandler.handleException(e);
            //4 异常处理完成返回默认值0
            return 0;
        }
    }

对应的 ASM代码

{
    // tryTest方法信息,方法签名为"()I"
    methodVisitor = classWriter.visitMethod(ACC_PUBLIC, "tryTest", "()I", null, null);
    methodVisitor.visitCode();
    
    Label labelStart = new Label();     // try开始
    Label labelEnd = new Label();       // try结束
    Label labelHandler = new Label();   // catch开始处理异常
    
    // 添加TryCatch固定操作,传入上面几个label
    methodVisitor.visitTryCatchBlock(labelStart, labelEnd, labelHandler, "java/lang/Exception");
    //1 try里面的方法执行
    methodVisitor.visitLabel(labelStart);
    methodVisitor.visitInsn(ICONST_3);
    methodVisitor.visitInsn(ICONST_0);
    methodVisitor.visitInsn(IDIV);
    methodVisitor.visitVarInsn(ISTORE, 1);
    methodVisitor.visitVarInsn(ILOAD, 1);
    //2 try里面正常,返回结果值
    methodVisitor.visitLabel(labelEnd);
    methodVisitor.visitInsn(IRETURN);
    
    //3 发生异常,进入labelHandler
    methodVisitor.visitLabel(labelHandler);
    // 存储和加载Exception实例变量
    methodVisitor.visitVarInsn(ASTORE, 1);
    methodVisitor.visitVarInsn(ALOAD, 1);
    // 调用自定义异常处理方法
    methodVisitor.visitMethodInsn(INVOKESTATIC, "com/dxkit/library/utils/ExceptionHandler", "handleException", "(Ljava/lang/Exception;)V", false);
    //4 异常处理完成返回默认值0
    methodVisitor.visitInsn(ICONST_0);
    methodVisitor.visitInsn(IRETURN);
    
    // 方法结束固定操作
    methodVisitor.visitMaxs(2, 2);
    methodVisitor.visitEnd();
}

这简单的示例代码分四个步骤,java对应的ASM操作码,我都有注释。

methodVisitor = classWriter.visitMethod(ACC_PUBLIC, "tryTest", "()I", null, null);

这句我们就能看出来,tryTest方法是public的,方法描述符是"()I" 说明方法没得参数,方法返回类型为int。

方法签名格式如下:

(参数类型描述符)返回值类型描述符

类型描述符:

类型描述符.png

举例方法签名:
方法描述符.png

这个跟我们写JNI开发的描述符是一样的。
在我们继承实现ClassVisitor的时候,在visitMethod方法中,会把方法签名descriptor回调给我们,通过descriptor我们就能判断任何方法的默认返回值是是啥。

2. 添加try catch把整个方法保护住

上面,我们知道了方法的ASM码与java代码的对应关系,也知道了方法描述符是怎么得来的,接下来就是找到切入点,把整个方法用try catch保护住。大概思路就是:

在方法开始之前,把try加上,在方法结束之前把catch以及异常处理添加上。

我们还是看上面的ASM代码:


ASM分段说明.png

methodVisitor.visitCode()标志方法开始进入。
methodVisitor.visitMaxs(2, 2)这里标志方法已经结束。

在AdviceAdapter对应的位置就是onMethodEntervisitMaxs,我们分别在这两个方法里面分别加上try和catch的代码就行。或者你想添加在visitCodevisitEnd方法里也行,最终代码都是一样的。
这样我们最终的代码为:

/**
 * @author danxingxi
 */
public class TryCatchClassVisitor extends ClassVisitor {

    private List<String> methodList;
    TryCatchClassVisitor(ClassVisitor classVisitor, List<String> methodList) {
        super(Opcodes.ASM6, classVisitor);
        this.methodList = methodList;
    }

    @Override
    public MethodVisitor visitMethod(int access, String name, String descriptor, String signature, String[] exceptions) {
        MethodVisitor methodVisitor = super.visitMethod(access, name, descriptor, signature, exceptions);
        LogUtil.log("TryCatchClassVisitor:" + "name : " + name);
        if (isNeedTryCatch(name)) {
            return new TryCatchMethodVisitor(Opcodes.ASM6, methodVisitor, access, name, descriptor);
        } else {
            return methodVisitor;
        }
    }

    /**
     * 方法是否需要加try catch
     */
    private boolean isNeedTryCatch(String methodName) {
        if (methodList != null && methodList.contains(methodName)) {
            return true;
        }
        return false;
    }

    /**
     * 方法执行顺序
     * onMethodEnter
     * visitCode
     * onMethodExit
     * visitMaxs
     * visitEnd
     *
     * @author danxingxi
     */
    public class TryCatchMethodVisitor extends AdviceAdapter {

        // 方法返回值类型描述符
        private String methodDesc;

        private String exceptionHandleClass;

        private String exceptionHandleMethod;

        private Label startLabel = new Label(),   // 开头
                endLabel = new Label(),           // 结尾
                handlerLabel = new Label(),       // 处理
                returnLabel = new Label();        // 返回

        TryCatchMethodVisitor(int api, MethodVisitor methodVisitor, int access, String name, String descriptor) {
            super(api, methodVisitor, access, name, descriptor);

            LogUtil.log("TryCatchMethodVisitor:" + "descriptor : " + descriptor);
            methodDesc = descriptor;

            Map<String, String> exceptionHandler = TryCatchExtension.exceptionHandler;
            if (exceptionHandler != null && !exceptionHandler.isEmpty()) {
                exceptionHandler.entrySet().forEach(entry -> {
                    exceptionHandleClass = entry.getKey();
                    exceptionHandleMethod = entry.getValue();
                });
            }

        }

        // 开始执行方法,插入的代码会onMethodEnter插入的代码之后,在本来执行代码之前。
        @Override
        public void visitCode() {
            super.visitCode();
        }

        // 方法进入
        @Override
        protected void onMethodEnter() {
            super.onMethodEnter();
            // 1标志:try块开始位置
            mv.visitTryCatchBlock(startLabel,
                    endLabel,
                    handlerLabel,
                    "java/lang/Exception");
            mv.visitLabel(startLabel);
        }

        @Override
        protected void onMethodExit(int opcode) {
            super.onMethodExit(opcode);
        }

        @Override
        public void visitMaxs(int maxStack, int maxLocals) {
            // 2标志:try块结束
            mv.visitLabel(endLabel);

            // 3标志:catch块开始位置
            mv.visitLabel(handlerLabel);
            mv.visitFrame(Opcodes.F_SAME1, 0, null, 1, new Object[]{"java/lang/Exception"});
            // 0代表this, 1 第一个参数,异常信息保存到局部变量
            mv.visitVarInsn(ASTORE, 1);
            // 从local variables取出局部变量到operand stack
            mv.visitVarInsn(ALOAD, 1);
            // 自定义异常处理
            if (exceptionHandleClass != null && exceptionHandleMethod != null) {
                mv.visitMethodInsn(INVOKESTATIC, exceptionHandleClass,
                        exceptionHandleMethod, "(Ljava/lang/Exception;)V", false);

            } else {
                // 没提供处理类就直接抛出异常
                mv.visitInsn(ATHROW);
            }

            // 顺序向下执行,可以不要GOTO
            //mv.visitJumpInsn(Opcodes.GOTO, returnLabel);
            // 返回label
            // mv.visitLabel(returnLabel);

            // catch结束,方法返回默认值收工
            Pair<Integer, Integer> defaultVo = ASMUtil.getDefaultByDesc(methodDesc);
            int value = defaultVo.getKey();
            int opcode = defaultVo.getValue();
            if (value >= 0) {
                mv.visitInsn(value);
            }
            mv.visitInsn(opcode);
            super.visitMaxs(maxStack, maxLocals);

        }

        @Override
        public void visitEnd() {
            super.visitEnd();
        }
    }

}

3. 发生异常时返回默认值

首先我们看看java中各种类型对应的默认值

数据类型 默认值
byte 0
short 0
int 0
long 0
float 0.0
double 0.0
boolean false
String null

其实只要是引用类型,就是对象,默认返回值就是null,不用管时数组还是多维数组,默认值也是 null。知道这个我们就可以根据方法描述符的返回类型来确认了。
具体操作的方法如下:

  /**
     * 根据方法描述符获取返回类型和默认值
     *
     * @param methodDesc
     * @return
     */
    public static Pair<Integer, Integer> getDefaultByDesc(String methodDesc) {
        Pair<Integer, Integer> pair = null;
        int value = -1;
        int opcode = -1;

        if (methodDesc.endsWith("[Z") ||
                methodDesc.endsWith("[I") ||
                methodDesc.endsWith("[S") ||
                methodDesc.endsWith("[B") ||
                methodDesc.endsWith("[C")) {
            value = Opcodes.ACONST_NULL;
            opcode = Opcodes.ARETURN;

        } else if (methodDesc.endsWith("Z") ||
                methodDesc.endsWith("I") ||
                methodDesc.endsWith("S") ||
                methodDesc.endsWith("B") ||
                methodDesc.endsWith("C")) {
            value = Opcodes.ICONST_0;
            opcode = Opcodes.IRETURN;

        } else if (methodDesc.endsWith("J")) {
            value = Opcodes.LCONST_0;
            opcode = Opcodes.LRETURN;

        } else if (methodDesc.endsWith("F")) {
            value = Opcodes.FCONST_0;
            opcode = Opcodes.FRETURN;

        } else if (methodDesc.endsWith("D")) {
            value = Opcodes.DCONST_0;
            opcode = Opcodes.DRETURN;

        } else if (methodDesc.endsWith("V")) {
            opcode = Opcodes.RETURN;

        } else {
            value = Opcodes.ACONST_NULL;
            opcode = Opcodes.ARETURN;
        }

        pair = new Pair<>(value, opcode);
        return pair;
    }

对于ASM来说,就是分为三大类,第一个就是为0,只不过不同类型的0对于的ASM中Opcodes码不一样,做一下区分就行,第二个就是为null,第三个就是无返回值的。
我们获取到的是一个 Pair,第一个是默认值,第二个是操作码,当默认值 >=0时,说明有默认值,当为-1时说明是无返回值的方法。

  Pair<Integer, Integer> defaultVo = ASMUtil.getDefaultByDesc(methodDesc);
  int value = defaultVo.getKey();
  int opcode = defaultVo.getValue();
  if (value >= 0) {
      // 有默认值,加载
      mv.visitInsn(value);
  }
  // 针对不同的默认值执行不同操作码
  mv.visitInsn(opcode);

其中你要知道,就算是默认值都是0或者是 0.0, 但是在ASM中,或者是jvm指令中,对应的返回值和返回操作码都是不一样的。比如floatdouble,对应的默认返回值为FCONST_0DCONST_0,返回操作码为: FRETURNDRETURN,剩下的我就不一一说明了。
下面是方法验证:

源代码
    public Object getObj() {
        Object object = new Object();
        return object;
    }
    
    public double getDouble() {
        double a = 2.3d;
        return a;
    }

    public long getLong() {
        long a = 1234567L;
        return a;
    }

    public float getFloat() {
        float a = 231234213.45f;
        return a;
    }

    public short getShort() {
        short a = 32767;
        return a;
    }

    public byte getByte() {
        byte a = 127;
        return a;
    }
ASM添加try catch处理后
    public Object getObj() {
        try {
            Object object = new Object();
            return object;
        } catch (Exception var2) {
            ExceptionHandler.handleException(var2);
            return null;
        }
    }
    public double getDouble() {
    try {
        double a = 2.3D;
        return a;
    } catch (Exception var3) {
        ExceptionHandler.handleException(var3);
        return 0.0D;
    }

    public long getLong() {
        try {
            long a = 1234567L;
            return a;
        } catch (Exception var3) {
            ExceptionHandler.handleException(var3);
            return 0L;
        }
    }

    public float getFloat() {
        try {
            float a = 2.31234208E8F;
            return a;
        } catch (Exception var2) {
            ExceptionHandler.handleException(var2);
            return 0.0F;
        }
    }

    public short getShort() {
        try {
            short a = 32767;
            return a;
        } catch (Exception var2) {
            ExceptionHandler.handleException(var2);
            return 0;
        }
    }

    public byte getByte() {
        try {
            byte a = 127;
            return a;
        } catch (Exception var2) {
            ExceptionHandler.handleException(var2);
            return 0;
        }
    }

想了解更多请查看:本文源代码:DxKit一个基于ASM的开发工具集:https://github.com/Dawish/DxKit

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容

  • 一、ASM 的优势和逆势 使用 ASM 操作字节码的优势与逆势都 比较明显,其分别如下所示。 1、ASM 的优势 ...
    waiwaaa阅读 362评论 0 1
  • AspectJ 非常强大,但是它也只能实现 50% 的字节码操作场景,如果想要实现 100% 的字节码操作场景,那...
    凯玲之恋阅读 874评论 0 4
  • 成为一名优秀的Android开发,需要一份完备的知识体系[https://github.com/Android-A...
    字节跳不动阅读 575评论 0 0
  • 前言 前面一篇文章 ASM 简介[https://www.jianshu.com/p/a85e8f83fa14] ...
    Whyn阅读 11,500评论 1 22
  • 16宿命:用概率思维提高你的胜算 以前的我是风险厌恶者,不喜欢去冒险,但是人生放弃了冒险,也就放弃了无数的可能。 ...
    yichen大刀阅读 6,044评论 0 4