动态编译生成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来做脚本可能更优雅一些



相关文章
|
28天前
|
算法 Java 数据处理
从HashSet到TreeSet,Java集合框架中的Set接口及其实现类以其“不重复性”要求,彻底改变了处理唯一性数据的方式。
从HashSet到TreeSet,Java集合框架中的Set接口及其实现类以其“不重复性”要求,彻底改变了处理唯一性数据的方式。HashSet基于哈希表实现,提供高效的元素操作;TreeSet则通过红黑树实现元素的自然排序,适合需要有序访问的场景。本文通过示例代码详细介绍了两者的特性和应用场景。
37 6
|
16天前
|
存储 安全 Java
java.util的Collections类
Collections 类位于 java.util 包下,提供了许多有用的对象和方法,来简化java中集合的创建、处理和多线程管理。掌握此类将非常有助于提升开发效率和维护代码的简洁性,同时对于程序的稳定性和安全性有大有帮助。
39 17
|
7天前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。
|
12天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
51 4
|
12天前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
29 2
|
17天前
|
存储 安全 Java
如何保证 Java 类文件的安全性?
Java类文件的安全性可以通过多种方式保障,如使用数字签名验证类文件的完整性和来源,利用安全管理器和安全策略限制类文件的权限,以及通过加密技术保护类文件在传输过程中的安全。
|
21天前
|
Java 数据格式 索引
使用 Java 字节码工具检查类文件完整性的原理是什么
Java字节码工具通过解析和分析类文件的字节码,检查其结构和内容是否符合Java虚拟机规范,确保类文件的完整性和合法性,防止恶意代码或损坏的类文件影响程序运行。
|
21天前
|
Java API Maven
如何使用 Java 字节码工具检查类文件的完整性
本文介绍如何利用Java字节码工具来检测类文件的完整性和有效性,确保类文件未被篡改或损坏,适用于开发和维护阶段的代码质量控制。
|
20天前
|
存储 Java 编译器
java wrapper是什么类
【10月更文挑战第16天】
23 3
|
23天前
|
Java 程序员 测试技术
Java|让 JUnit4 测试类自动注入 logger 和被测 Service
本文介绍如何通过自定义 IDEA 的 JUnit4 Test Class 模板,实现生成测试类时自动注入 logger 和被测 Service。
22 5