文档章节

java ASM 实现新生子类

y
 yan5845hao
发布于 2017/05/15 16:35
字数 502
阅读 16
收藏 0

需要实现效果

package com.uwo.resources.asm.test;

/**
 * Created by yanhao on 2017/5/13.
 */
public class Uwo$Test{

    public Uwo$Test(String name){
        super(name);
    }

    public String hello(String say){
        System.out.println("START");
        System.out.println("hello source!!");
        System.out.println("END");
        return name  + " == " + say;
    }
    
}

原类

package com.uwo.resources.asm.test;

/**
 * Created by yanhao on 2017/5/13.
 */
public class Uwo{

    protected String name;

    public Uwo(String name){
        this.name = name;
    }

    public String hello(String say){

        System.out.println("hello source!!");

        return name  + " == " + say;
    }

    protected Integer sex;

}

实现MethodAdapter

package com.uwo.resources.asm.test;

import org.objectweb.asm.MethodAdapter;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;

/**
 * Created by yanhao on 2017/5/15.
 */
public class UwoMethodAdapter extends MethodAdapter{

    public UwoMethodAdapter(MethodVisitor mv){
        super(mv);
    }


    @Override
    public void visitCode() {
        mv.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;");
        mv.visitLdcInsn("START");
        mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V");
    }

    @Override
    public void visitInsn(int opcode) {
        if ((opcode >= Opcodes.IRETURN && opcode <= Opcodes.RETURN) || opcode == Opcodes.ATHROW) {
            mv.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;");
            mv.visitLdcInsn("END");
            mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V");
        }
        mv.visitInsn(opcode);
    }

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

实现ClassAdapter

package com.uwo.resources.asm.test;

import org.objectweb.asm.*;

/**
 * Created by yanhao on 2017/5/15.
 */
public class UwoClassAdapter extends ClassAdapter{

    private String superName;

    public UwoClassAdapter(ClassVisitor cv){
        super(cv);
    }

    @Override
    public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) {
        System.out.println("visit start");
        this.superName = name;
        super.visit(version, access, name + "$Test", signature, name, interfaces);
        System.out.println("visit end");
    }

    @Override
    public void visitOuterClass(String owner, String name, String desc) {
        System.out.println("visitOuterClass start");
        super.visitOuterClass(owner, name, desc);
        System.out.println("visitOuterClass end");
    }

    @Override
    public void visitInnerClass(String name, String outerName, String innerName, int access) {
        System.out.println("visitInnerClass start");
        super.visitInnerClass(name, outerName, innerName, access);
        System.out.println("visitInnerClass end");
    }

    @Override
    public void visitAttribute(Attribute attr) {
        System.out.println("visitAttribute start");
        super.visitAttribute(attr);
        System.out.println("visitAttribute end");
    }

    @Override
    public void visitSource(String source, String debug) {
        System.out.println("visitSource start");
        super.visitSource(source, debug);
        System.out.println("visitSource end");
    }

    @Override
    public void visitEnd() {
        System.out.println("visitEnd start");
        super.visitEnd();
        System.out.println("visitEnd end");
    }

    @Override
    public FieldVisitor visitField(int access, String name, String desc, String signature, Object value) {
        System.out.println("FieldVisitor");
        return super.visitField(access, name, desc, signature, value);
    }

    @Override
    public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {
        System.out.println("MethodVisitor");
        MethodVisitor mv = cv.visitMethod(access, name, desc, signature, exceptions);
        if(!name.equalsIgnoreCase("<init>")){
            return new UwoMethodAdapter(mv);
        }else{
            mv.visitVarInsn(Opcodes.ALOAD, 0);
            mv.visitVarInsn(Opcodes.ALOAD, 1);
            mv.visitMethodInsn(Opcodes.INVOKESPECIAL, superName, "<init>", "(Ljava/lang/String;)V");
            mv.visitInsn(Opcodes.RETURN);
            return mv;
        }

    }

    @Override
    public AnnotationVisitor visitAnnotation(String desc, boolean visible) {
        System.out.println("AnnotationVisitor");
        return super.visitAnnotation(desc, visible);
    }

}

执行方法

try {
            ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS);
            ClassReader cr = new ClassReader(Uwo.class.getName());
            UwoClassAdapter adapter = new UwoClassAdapter(cw);
            cr.accept(adapter, 0);
            byte[] bytes = cw.toByteArray();
            TestLoader loader = new TestLoader();
            Class<?> clazz = loader.findClass("com.uwo.resources.asm.test.Uwo$Test", bytes);
            Constructor constructor = clazz.getConstructor(new Class[]{String.class});
            Object o = constructor.newInstance("YANHAO123456");
            Method main = clazz.getMethods()[0];
            Object res = main.invoke(o, new Object[]{"hello"});
            if(res != null){
                System.out.println(res);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

执行结果

START
hello source!!
END
YANHAO123456 == hello

© 著作权归作者所有

共有 人打赏支持
y
粉丝 7
博文 120
码字总数 21375
作品 0
扬州
程序员
AOP 的利器:ASM 3.0 介绍

引言 什么是 ASM ? ASM 是一个 Java 字节码操控框架。它能被用来动态生成类或者增强既有类的功能。ASM 可以直接产生二进制 class 文件,也可以在类被加载入 Java 虚拟机之前动态改变类行为。...

goto-array
2013/09/21
0
0
枚举类型可以实现接口,以及枚举类型原理

今天在看 《漫谈设计模式》的时候,看到状态模式的时候,居然发现,枚举类型原来也是可以 实现接口的。 真是涨见识了。 而且 枚举类型的静态属性也是可以 实现方法的, 只是如果没有 实现接口...

之渊
07/24
0
0
CGLIB子类代理和JDK动态代理示例和比较

Cglib代理,也叫做子类代理。在内存中构建一个子类对象从而实现对目标对象功能的扩展。 比较: JDK的动态代理有一个限制,就是使用动态代理的对象必须实现一个或多个接口。如果想代理没有实现...

AI9o後
2017/11/06
0
0
动态代理机制详解(JDK 和CGLIB,Javassist,ASM)

在运行时期可以按照Java虚拟机规范对class文件的组织规则生成对应的二进制字节码。当前有很多开源框架可以完成这些功能,如ASM,Javassist。 动态代理机制详解(JDK 和CGLIB,Javassist,ASM...

素雷
2017/10/19
0
0
cglib代理和java自带代理的区别

一 动态代理主要有java自带的代理和cglib方式实现的代理 首先讲下java自带的代理 ------------------然后是cglib自带的代理------------------------- 测试类: java动态代理是利用反射机制生...

sen_ye
07/15
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

Java IO类库之PrintStreamWriter

* A <code>PrintStream</code> adds functionality to another output stream, * namely the ability to print representations of various data values * conveniently. Two other fea......

老韭菜
今天
0
0
qduoj~前端~二次开发~笔记

青岛大学qdu的onlinejudge是js的写的前端,框架是vue.js,在nodejs上部署运行,其实整体运行还是建立在docker的容器虚拟环境里,这里暂时不需要docker。安装环境是Ubuntu14-64bit 1.安装一大...

虚拟世界的懒猫
今天
6
0
ConcurrentHashMap源码解读

部分内容转自:http://jiabinyuan.xyz/#/app/archive/detail/25 内部结构 内部采用了segment结构,每一个segment相当于一个hashtable。看下面的结构图: 从图的结构我们可以了解到,Concurr...

edwardGe
今天
1
0
Ubuntu终端Tab键自动补全

打开 /etc/bash.bashrc,找到下列代码,取消注释。 #enable bash completion in interactive shells#if ! shopt -oq posix; then# if [-f /usr/share/bash-completion/bash_compl......

大熊猫
今天
0
0
polipo socks5代理转http代理

天朝的网络,哎~ 装个 yarn 都时而会卡 假设在SSlocal 已经装好运行的前提下,来安装设置 polipo sudo apt-get install polipo sudo vim /etc/polipo/config 追加下列配置内容,并保存 socksP...

纯洁徐
今天
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部