使用javassist动态生成类

说明:

  • Javassist(JAVA 编程 ASSISTant)使 Java 字节码操作变得简单。 它是一个用 Java 编辑字节码的类库。它使 Java 程序能够在运行时定义新类,并在 JVM 加载时修改类文件。 与其他类似的字节码编辑器不同,Javassist 提供两个级别的 API:源级别和字节码级别。 如果用户使用源级 API,他们可以在不知道 Java 字节码规范的情况下编辑类文件。 整个 API 仅使用 Java 语言的词汇表进行设计。 您甚至可以以源文本的形式指定插入的字节码;Javassist 即时编译它。 另一方面,字节码级 API 允许用户直接编辑类文件作为其他编辑器。

配置环境:

  • javassist依赖:
<dependency>
  <groupId>org.javassist</groupId>
  <artifactId>javassist</artifactId>
  <version>3.29.1-GA</version>
</dependency>
  • 环境配置:
    –add-opens java.base/java.lang=ALL-UNNAMED
    –add-opens java.base/sun.net.util=ALL-UNNAMED
    请添加图片描述

使用javassist生成类:

代码:

package com.user;

import javassist.*;
import java.lang.reflect.Method;

public class JavassistTest {
    public static void main(String[] args) throws Exception{
        //获取类池
        ClassPool pool = ClassPool.getDefault();
        //通过类路径创建类
        CtClass ctClass = pool.makeClass("com.user.Test");
        //为类创建方法(参数:1.方法返回值类型 2.方法名 3.方法形参列表 4.方法所属类)
        CtMethod ctMethod = new CtMethod(CtClass.voidType, "execute", new CtClass[]{}, ctClass);
        //设置方法的修饰符列表
        ctMethod.setModifiers(Modifier.PUBLIC);
        //设置方法体
        ctMethod.setBody("{System.out.println(\"hello world\");}");
        //给类添加方法
        ctClass.addMethod(ctMethod);
        //测试:使用反射机制测试方法调用
        Class<?> aClass = ctClass.toClass();
        Object o = aClass.newInstance();
        Method method = aClass.getDeclaredMethod("execute");
        method.invoke(o);

    }
}

执行结果:
请添加图片描述

使用javassist生成类并实现接口:

接口已定义好:

package com.user;

public interface Dao {
    public  int insert();
    public  int delete();
    public  int update();
    public  void select();
}

代码:

package com.user;

import javassist.*;

import java.lang.reflect.Method;
import java.util.Arrays;

public class JavassistTestImpl {
    public static void main(String[] args) throws Exception {
        //获取类池
        ClassPool pool = ClassPool.getDefault();
        //制造类
        CtClass ctClass = pool.makeClass("com.user.TestImpl");
        //制造接口
        CtClass ctInterface = pool.makeInterface("com.user.Dao");
        //类实现接口
        ctClass.addInterface(ctInterface);
        //获取接口中的所有方法
        Method[] methods = Dao.class.getDeclaredMethods();
        //类实现接口中的所有方法
        Arrays.stream(methods).forEach(method->{
            try {
                //拼接方法头+方法体
                StringBuilder sb = new StringBuilder();
                sb.append("public");//方法修饰符
                sb.append(" ");
                sb.append(method.getReturnType().getName());//方法返回值类型
                sb.append(" ");
                sb.append(method.getName());//方法名
                sb.append("(");
                Class<?>[] parameterTypes = method.getParameterTypes();
                for (int i = 0; i < parameterTypes.length; i++) {
                    sb.append(parameterTypes[i].getName());//参数类型
                    sb.append(" ");
                    sb.append("arg"+i);//参数名,随意指定,但不能重名
                    if(i != parameterTypes.length-1) {
                        sb.append(",");
                    }
                }
                sb.append(")");
                sb.append("{");
                String simpleName = method.getReturnType().getSimpleName();//根据返回值类型写return语句
                if("String".equals(simpleName)){
                    sb.append("return \"hello\";");
                }else if("int".equals(simpleName)){
                    sb.append("return 1;");
                }else if("void".equals(simpleName)){
                    sb.append("System.out.println(\"hello\");");
                }
                sb.append("}");
                //创建方法(参数:方法头+方法体,所属类)
                CtMethod ctMethod = CtNewMethod.make(sb.toString(),ctClass);
                ctClass.addMethod(ctMethod);
            } catch (CannotCompileException e) {
                e.printStackTrace();
            }
        });
        //测试
        Class<?> aClass = ctClass.toClass();
        Dao o = (Dao)aClass.newInstance();
        o.select();
        System.out.println(o.insert());
    }
}

执行结果:
请添加图片描述

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

姓蔡小朋友

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值