动态编译生成Java类

简介: 动态创建bean,前面一篇介绍了通过cglib来创建的方式,虽然实现了动态创建java bean,但是有一个问题,java bean中的field name和我们预期的不太一致接下来我们介绍一种直接通过拼接java代码,然后再将其编译成class并加载,从而实现动态类的创建

动态创建bean,前面一篇介绍了通过cglib来创建的方式,虽然实现了动态创建java bean,但是有一个问题,java bean中的field name和我们预期的不太一致


接下来我们介绍一种直接通过拼接java代码,然后再将其编译成class并加载,从而实现动态类的创建


1. java代码动态编译创建



接下来我们主要借助jdk自带的JavaCompiler来实现java源码的编译,生成class,然后交由ClassLoader来加载类


如果我们现在有个java文件,希望在项目运行时,动态编译并加载类,一般的操作流程如下



JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
int compilationResult = compiler.run(null, null, null, '/path/Test.java');
复制代码

相比于存在的java文件,我们更多的场景则是针对String的java源码,基于它来创建java类


测试则需要针对上面的逻辑进行扩展;如果不想关心细节,直接使用开源的三方包来实现


<dependency>
    <groupId>com.itranswarp</groupId>
    <artifactId>compiler</artifactId>
    <version>1.0</version>
</dependency>
复制代码


同样我们来实现一个根据Map,来生成java bean的方式


基于javacode来生成源码

/**
 * 编译java代码,并生成对象
 *
 * @param packageName 包名
 * @param className   类名
 * @param javaCode    源码
 * @return
 * @throws Exception
 */
public Object createBean(String packageName, String className, String javaCode) throws Exception {
    JavaStringCompiler compiler = new JavaStringCompiler();
    Map<String, byte[]> result = compiler.compile(className + ".java", "package " + packageName + ";\n" + javaCode);
    Class clz = compiler.loadClass(packageName + "." + className, result);
    return clz.newInstance();
}
复制代码


接下来我们需要实现的是根据map来生成对应的java code

private String genCode(String clzName, Map<String, Object> map) {
    StringBuilder builder = new StringBuilder("public class ").append(clzName).append("{\n");
    for (Map.Entry<String, Object> obj: map.entrySet()) {
        String fieldType;
        if (obj.getValue() instanceof List) {
            fieldType = "java.util.List";
        } else if (obj.getValue() instanceof Set) {
            fieldType = "java.util.Set";
        } else if (obj.getValue() instanceof  Map) {
            fieldType = "java.util.Map";
        } else {
            fieldType = obj.getValue().getClass().getName().replace("$", ".");
        }
        builder.append("\tpublic ").append(fieldType).append(" ").append(obj.getKey()).append(";\n");
    }
    builder.append("}");
    return builder.toString();
}
复制代码


注意上面的实现,现在只是最基础的实现,需要注意,如果value的类型,是一个内部类,就可能有坑(比如上面针对容器类进行了特殊处理)


最后测试一下

@Test
public void testGenMapBean() throws Exception {
    Map<String, Object> map = new HashMap<>();
    map.put("hello", "world");
    map.put("key", 12);
    map.put("list", Arrays.asList(1, 2, 3));
    String packageName = "com.git.hui.dynamic";
    String clzName = "MBean1";
    String code = genCode(clzName, map);
    Object bean = createBean(packageName, clzName, code);
    // 初始化bean的成员
    for(Map.Entry<String, Object> entry: map.entrySet()) {
        Field field = bean.getClass().getField(entry.getKey());
        field.set(bean, entry.getValue());
    }
    System.out.println("---------------- java code  ---------------\n" + code + "\n------------------");
    System.out.println(JSON.toJSONString(bean));
}
复制代码


上面给出了一个根据Map生成bean对象,并初始化成员值的实例


输出如下

---------------- java code  ---------------
public class MBean1{
  public java.lang.String hello;
  public java.util.List list;
  public java.lang.Integer key;
}
------------------
{"hello":"world","key":12,"list":[1,2,3]}
复制代码


输出和我们预期一致,直接通过组装java代码,然后来生成对象


那么这种方式应用场景在什么地方呢?


  • 比如我之前做过一个动态脚本执行的框架,可以在控制台上写java代码,写完之后直接交给框架来运行;当然这种选择groovy来做脚本可能更优雅一些



相关文章
|
6月前
|
Java 编译器 API
Java 密封类:精细化控制继承关系
Java 密封类:精细化控制继承关系
397 83
|
4月前
|
安全 Java 数据建模
Java记录类:简化数据载体的新选择
Java记录类:简化数据载体的新选择
272 101
|
4月前
|
安全 Java 开发者
Java记录类:简化数据载体的新方式
Java记录类:简化数据载体的新方式
313 100
|
7月前
|
IDE Java 数据挖掘
Java 基础类从入门到精通实操指南
这份指南专注于**Java 17+**的新特性和基础类库的现代化用法,涵盖开发环境配置、数据类型增强(如文本块)、字符串与集合处理进阶、异常改进(如密封类)、IO操作及实战案例。通过具体代码示例,如CSV数据分析工具,帮助开发者掌握高效编程技巧。同时提供性能优化建议和常用第三方库推荐,适合从入门到精通的Java学习者。资源链接:[点此下载](https://pan.quark.cn/s/14fcf913bae6)。
260 36
|
5月前
|
安全 IDE Java
Java记录类型(Record):简化数据载体类
Java记录类型(Record):简化数据载体类
458 143
|
3月前
|
存储 Java 索引
用Java语言实现一个自定义的ArrayList类
自定义MyArrayList类模拟Java ArrayList核心功能,支持泛型、动态扩容(1.5倍)、增删改查及越界检查,底层用Object数组实现,适合学习动态数组原理。
143 4
|
3月前
|
IDE JavaScript Java
在Java 11中,如何处理被弃用的类或接口?
在Java 11中,如何处理被弃用的类或接口?
225 5
|
3月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
219 1
|
3月前
|
Java Go 开发工具
【Java】(8)正则表达式的使用与常用类分享
正则表达式定义了字符串的模式。正则表达式并不仅限于某一种语言,但是在每种语言中有细微的差别。
271 1
|
3月前
|
存储 Java 程序员
【Java】(6)全方面带你了解Java里的日期与时间内容,介绍 Calendar、GregorianCalendar、Date类
java.util 包提供了 Date 类来封装当前的日期和时间。Date 类提供两个构造函数来实例化 Date 对象。第一个构造函数使用当前日期和时间来初始化对象。Date( )第二个构造函数接收一个参数,该参数是从1970年1月1日起的毫秒数。
218 1