栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

javassist对已有的类进行操作

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

javassist对已有的类进行操作

准备工作:

首先创建一个java项目,然后导入javassist.jar包

创建一个注解

package com.chengyu.javassist;

public @interface Auto {
    String name();
    int year();
}

创建一个接口

public interface Earth {
}

创建两个类

public class Pepelo{
}

package com.chengyu.javassist;

@Auto(name="KO",year=1999)
public class Student extends Pepelo implements Earth{
    private int age;
    private String name;

    public int myTest(int a)
    {
        System.out.println("我在学习。。。。" + a);
        return 6666;
    }

    public Student(){}
    public Student(int age,String name)
    {
        this.age = age;
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

Student类继承Pepelo类实现Earth接口

然后我们开始测试

首先我们来对类进行操作,获取类的信息

 
    public static void test01() throws Exception {
        ClassPool pool = ClassPool.getDefault();
        CtClass cc = pool.get("com.chengyu.javassist.Student");
        byte [] bytes = cc.toBytecode(); //获取类的二进制字节码转成byte数组
        String allName = cc.getName();   //获取类的全限定名  com.chengyu.javassist.Student
        String name = cc.getSimpleName(); //获取类的简单名   Student
        CtClass superclass = cc.getSuperclass();    //获取类的父类
        CtClass[] interfaces = cc.getInterfaces();  //获取类的父接口
        System.out.println(allName);
        System.out.println(name);
        System.out.println(superclass.getName());
        System.out.println(interfaces[0].getName());
        System.out.println(Arrays.toString(bytes));
    }

新增一个方法

    public static void test02() throws Exception {
        ClassPool pool = ClassPool.getDefault();
        CtClass cc = pool.get("com.chengyu.javassist.Student");
        //CtMethod make = CtNewMethod.make("public int add(int a,int b){return a+b;}",cc);  创建方法可以使用这种方法 也可以使用下面的方法

        CtMethod ctMethod = new CtMethod(CtClass.intType,"add",new CtClass[]{CtClass.intType,CtClass.intType},cc);
        //new CtMethod 的第一个参数为返回类型,第二个参数是方法名,第三个参数为方法的参数列表(这里没有给参数命名),第四个参数为CtClass类
        ctMethod.setModifiers(Modifier.PUBLIC); //声明方法的访问权限
        ctMethod.setBody("{return $1+$2;}");    //设置方法体 上面没有给参数命名,这里需要使用$符号进行匹配
        
        cc.addMethod(ctMethod); //将方法添加到类中
        //下面使用java反射来进行调用验证
        Class clazz = cc.toClass();
        Object obj = clazz.newInstance();
        Method declaredMethod = clazz.getDeclaredMethod("add",int.class, int.class);
        Object invoke = declaredMethod.invoke(obj, 200, 300);
        System.out.println("result:" + (int)invoke);

    }

执行结果

 对已有的方法进行修改

 
    public static void test03() throws Exception {
        ClassPool pool = ClassPool.getDefault();
        CtClass cc = pool.get("com.chengyu.javassist.Student");
        //根据方法名和参数列表来获取方法
        CtMethod myTest = cc.getDeclaredMethod("myTest", new CtClass[]{CtClass.intType});
        myTest.insertBefore("System.out.println($1 + " 准备打开书");"); //在方法前进行插入代码
        myTest.insertAfter("System.out.println("已经学完了,出去玩");"); //在方法后面进行插入代码
        myTest.insertAt(16,"System.out.println("这是在某一行插入的代码!");"); //在某一行插入代码

        //使用反射进行调用测试
        Class clazz = cc.toClass();
        Object obj = clazz.newInstance();
        Method declaredMethod = clazz.getDeclaredMethod("myTest",int.class);
        Object invoke = declaredMethod.invoke(obj, 200);
        System.out.println("result:" + (int)invoke);
    }

执行的结果

 对属性的操作-新增一个属性并且为其添加set和get方法

 
    public static void test04() throws Exception
    {
        ClassPool pool = ClassPool.getDefault();
        CtClass cc = pool.get("com.chengyu.javassist.Student");
        //CtField make = CtField.make("private String phone;", cc);  可以通过这种方式添加属性,也可以使用下面的方式
        CtField ctField = new CtField(pool.get("java.lang.String"),"phone",cc);
        //new CtField 第一个参数为属性的类型,第二个参数为属性名,第三个参数为CtClass
        ctField.setModifiers(Modifier.PRIVATE);  //设置属性的访问权限
        cc.addField(ctField);                    //将属性添加到方法中

        //CtField phone = cc.getDeclaredField("phone");  这里可以根据属性名来获取属性

        //添加set和get方法 第一个参数为方法名 第二个参数为属性
        cc.addMethod(CtNewMethod.getter("getPhone",ctField));
        cc.addMethod(CtNewMethod.setter("setPhone",ctField));

        //通过反射进行调用测试
        Class clazz = cc.toClass();
        Object obj = clazz.newInstance();
        Method setPhone = clazz.getDeclaredMethod("setPhone",String.class);
        Method getPhone = clazz.getDeclaredMethod("getPhone",null);

        setPhone.invoke(obj,"13088888888");
        Object phone = getPhone.invoke(obj);
        System.out.println(String.valueOf(phone));
    }

执行结果:

 对构造方法的操作

   public static void test05()throws Exception
    {
        ClassPool pool = ClassPool.getDefault();
        CtClass cc = pool.get("com.chengyu.javassist.Student");

        CtConstructor[] constructors = cc.getConstructors();
        for(CtConstructor c : constructors)
        {
            System.out.println(c.getLongName());//打印构造方法的全类限定名
            c.insertBefore("System.out.println("我是构造方法");"); //当然我们也可以对构造方法进行操作
        }

        Class clazz = cc.toClass();
        Object obj = clazz.newInstance();
    }

执行结果:

 对注解的操作

public static void test06()throws Exception
    {
        ClassPool pool = ClassPool.getDefault();
        CtClass cc = pool.get("com.chengyu.javassist.Student");
        Object [] all = cc.getAnnotations();  //获取标注在类上的所有注解
        for(Object a : all)
        {
            if(a instanceof Auto)
            {
                Auto b = (Auto) a;
                System.out.println("name : " + b.name() + " year : " + b.year());
            }
        }
    }

执行结果

 在这里就做这些介绍

当然javassist也是有一定的局限性的:

1、泛型、枚举是不支持的,

2、注解的修改也不支持,虽然可以通过底层的javassist类来解决

3、不支持数组的初始化工作,除非数组的容量为1

4、不支持内部类和匿名类

5、不支持continue和break

6、对于复杂的继承也不支持,支持简单的继承关系

原创不易,不喜勿喷,有更好的建议欢迎大家留言

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/361142.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号