美文网首页
ASM(初探使用)

ASM(初探使用)

作者: 火火说技术 | 来源:发表于2018-05-07 23:20 被阅读0次

ASM

背景

写文章特别喜欢写背景,感觉如果不写背景就没法回忆出来当时为什么要搞这个东西。好了,因为之前参与的一个项目的故障演练模块觉得做的只是基于spring的Bean的代理做的,这样对业务的侵入性比较强,如果业务没有依赖于spring肿么办呢?现在我们来看看ASM是怎么做的

ASM可以干什么

大名鼎鼎的CGLIB其实底层就是ASM。通过ASM的字节码操作,可以动态创建新的类型,可以为类增加新的功能呢。虽然可以使用CGLIB这些高级的库也可以完成大量的工作,但是如果直接使用ASM还是有很多好处的,例如ASM的性能是最好的,灵活度是最好的,功能也是最为强大的,可以将操作粒度控制到每一条指令。

简单的进行字节码织入操作

例如我们只有一个简单的Account类,该类也只有一个方法operation方法。

public class Account {
    public void operation() {
        System.out.println("operation ...");
    }
}

现在我们要在这个操作之前进行一定的验证,例如加入一些检查权限的操作checkSecurity()。我们将添加一个名称为SecurityChecker的类。这个类中的方法可以帮助我们进行一些权限校验。

public class SecurityChecker {
    public static boolean checkSecurity() {
        System.out.println("SecurityChecker.checkSecurity ...");
        if ((System.currentTimeMillis() & 0x1) == 0) {
            return false;
        } else {
            return true;
        }
    }
}

我们在不修改原来Account的代码的前提下,如何增加校验操作呢?我们可以直接修改类的字节码进行代码织入操作,从而改变Account代码的执行状态。

我们先看一下单独运行Account类的执行效果把~

image.png

可以看到只是打印出来了operation ...

进行改写字节码

通过如下代码我们可以将代码的字节码进行修改,并且覆盖掉原来的编译好的字节码,从而改变类的执行状态。

首先我们需要几个类对象,第一个

  • 负责Class改写的适配器类AddSecurityCheckClassAdapter继承自ClassVisitor

  • 负责Method改写的适配器类AddSecurityCheckMethodAdapter继承自MethodVisitor

  • 负责调用Adapter的SecurityWeaveGeneratior

AddSecurityCheckClassAdapter

负责修改类文件中的字节码

public class AddSecurityCheckClassAdapter extends ClassVisitor {

    public AddSecurityCheckClassAdapter(ClassVisitor classVisitor) {
        super(Opcodes.ASM5, classVisitor);
    }

    @Override
    public MethodVisitor visitMethod(int i, String s, String s1, String s2, String[] strings) {


        MethodVisitor mv = super.visitMethod(i, s, s1, s2, strings);

        MethodVisitor wrappedMv = mv;

        if (mv != null){
            if (s.equals("operation")){
                wrappedMv = new AddSecurityCheckMethodAdapter(mv);
            }
        }
        return wrappedMv;
    }
}

AddSecurityCheckMethodAdapter

负责修改某个method中的字节码

public class AddSecurityCheckMethodAdapter extends MethodVisitor {
    public AddSecurityCheckMethodAdapter(MethodVisitor mv) {
        super(Opcodes.ASM5, mv);
    }

    @Override
    public void visitCode() {
        Label continueLabel = new Label();
        visitMethodInsn(Opcodes.INVOKESTATIC, "com/wsqandgy/asm/SecurityChecker", "checkSecurity", "()Z");
        visitJumpInsn(Opcodes.IFNE, continueLabel);
        visitInsn(Opcodes.RETURN);
        visitLabel(continueLabel);
        super.visitCode();
    }
}

SecurityWeaveGeneratior

读取类信息,进行字节码织入

public class SecurityWeaveGeneratior {

    public static void main(String[] args) throws Exception {
        String className = Account.class.getName();
        ClassReader classReader = new ClassReader(className);
        ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_FRAMES | ClassWriter.COMPUTE_MAXS);
        AddSecurityCheckClassAdapter classAdapter = new AddSecurityCheckClassAdapter(cw);
        classReader.accept(classAdapter, ClassReader.SKIP_DEBUG);
        byte[] data = cw.toByteArray();
        File file = new File("/Users/gongyan/Documents/home_code/tools/apache/target/classes/" + className.replaceAll("\\.", "/") + ".class");
        if (file.exists()){
            System.out.println("exists");
        }
        FileOutputStream fileOutputStream = new FileOutputStream(file);
        fileOutputStream.write(data);
        fileOutputStream.close();
    }
}

运行结果

安全检查

还是运行原来的main方法,自动加入了检查安全的操作。

对比一下前后的字节码

前:


后:


对比前后,明显可以看到在字节码中增加了我们操作ASM写入的相关字节码。

写在最后,最近生病是在身体乏力,写的文章自己认为也只讲了简单的如何使用后面再好好补补,这个地方会和前面的服务保护有一定的串联!请多多期待!

相关文章

  • ASM(初探使用)

    ASM 背景 写文章特别喜欢写背景,感觉如果不写背景就没法回忆出来当时为什么要搞这个东西。好了,因为之前参与的一个...

  • Asm初探

    最近项目中产品要求接入神策埋点,神策最大的宣传点应该就是所谓无痕全埋点。对于这种"无痕"或者"无感知",大部分An...

  • ASM框架(一)初探

    1. ASM简介 ASM是被设计用于运行时的,离线的类生成和转换,作用于已编译好的Java class,并且被设计...

  • Android Transform + ASM 初探

    背景 随着项目中对 APM (Application Performance Management) 越来越关注,...

  • 05 - 如何编写ASM代码

    在刚开始学习ASM的时候,编写ASM代码是不太容易的。或者,有些人原来对ASM很熟悉,但由于长时间不使用ASM,编...

  • [转]ASM 探秘

    一、ASM 的优势和逆势 使用 ASM 操作字节码的优势与逆势都 比较明显,其分别如下所示。 1、ASM 的优势 ...

  • 安装GRID,使用NETAPP磁盘作ASM磁盘

    在使用NETAPP的存储,制作ASM磁盘,使用多路径的方式,多路径与ASM磁盘分组分别完成之后,安装grid,在执...

  • SpringBoot目录

    一. SpringBoot 使用 SpringBoot 使用(一): 初探 SpringBoot 使用(二): 集...

  • 学习自定义Gradle插件2

    前序步骤参考链接:学习自定义gradle插件 使用 ASM,插入字节码到 Activity 文件 1.在 asm_...

  • Android ASM使用

    ASM ASM是一种基于java字节码层面的代码分析和修改工具,ASM的目标是生成,转换和分析已编译的java c...

网友评论

      本文标题:ASM(初探使用)

      本文链接:https://www.haomeiwen.com/subject/iivbrftx.html