Java语言动态代码动态功能

  • Post author:
  • Post category:java


    • 需求

可以让用户动态决定功能,形成类似于插件或动态语言特性的功能。举例:有一个字段,是让用户输入并保存数据的,但是,我可以实现以动态语言特性扩展它的功能:如果用户输入了给字段值乘百分比的代码,这个字段值就会拥有一个系数的特性,如果用户输入了给字段进行格式化的代码,这个字段值就可以进行格式化,如果用户输入了给字段进行其他计算的代码,这个字段就可以由用户决定它的计算逻辑。

    • 技术方案

引入javassist,用它的动态特性,实现功能扩展。

具体做法是:

  1. 修改产品功能,针对目标字段增加个性化功能

  1. 这个字段的个性化功能页中,可以支持输入代码,提交时将代码保存到数据库中,当然,应当做必要的校验,防止代码输入错误无法执行

  1. 在引用字段的地方,从数据库中取出代码,进行动态加载和执行

多少有点低代码平台的性质了,不过,低代码平台通常动态的是JS语言,我们动态Java语言,可以让动态逻辑在后台执行,也是非常不错的。因为安全性会高很多。

    • 具体实现

引入maven依赖

<dependency>
    <groupId>org.javassist</groupId>
    <artifactId>javassist</artifactId>
    <version>3.29.2-GA</version>
</dependency>

示例代码

package com.wasu.t.dynamic_code;

import javassist.*;
import org.junit.Assert;
import org.junit.Test;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.UUID;

/**
 * Java动态代码功能
 *
 * @author lidawei
 * @date 2023/1/5 11:39
 */
public class DynamicCodeTest {
    @Test
    public void yesterday() throws CannotCompileException, ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
        // 创建类
        String className = UUID.randomUUID().toString().replaceAll("-", "");
        ClassPool pool = ClassPool.getDefault();
        CtClass ct = pool.makeClass(className);

        // 动态添加功能
        CtMethod method = CtNewMethod.make("public String yesterday(String fmt) { " +
                        "return java.time.LocalDateTime.now().minusDays(1L).format(java.time.format.DateTimeFormatter.ofPattern(fmt));" +
                        "}",
                ct);
        ct.addMethod(method);

        // 调用动态添加的功能
        Loader loader = new Loader(pool);
        Class<?> clazz = loader.loadClass(className);
        Object instance = clazz.newInstance();
        Method yesterday = clazz.getDeclaredMethod("yesterday", String.class);
        Object result = yesterday.invoke(instance, "yyyy-MM-dd");
        Assert.assertNotNull("动态方法返回了空", result);
        System.out.println(result);
    }
}

将类名、方法名、具体的代码保存到数据库中,然后在使用字段的地方,按上述过程,把代码取出来,执行一遍,就得到结果了。



版权声明:本文为ccpw_cn原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。