使用Groovy将一段代码动态构造成一个类的方法

一、简介

在项目中,有时候需要将动态调用Java代码,但是代码可能是通过富文本编辑器保存在库里面的,并且使用CLOB数据类型保存。那么在使用的时候,我们必然需要将库里面的代码动态构造成一个Class类,然后通过反射调用类的方法实现功能。下面总结一下如何使用Groovy将一段代码动态构造成一个类。

二、使用方法

【a】pom文件引入groovy依赖包


    org.codehaus.groovy
    groovy-all

【b】定义一个Java接口

public interface GroovyTest {

    /**
     * 求和计算
     */
    int sum(int a, int b);

}

定义一个接口实现类:

public class GroovyTestImpl implements GroovyTest {
    @Override
    public int sum(int a, int b) {
        return a + b;
    }
}

 【c】定义Groovy处理工具类:将一段代码动态构造成一个类

import groovy.lang.GroovyClassLoader;

import java.util.HashMap;
import java.util.Map;

/**
 * 使用groovy将一段代码动态构造成一个类
 */
public class GroovyAutoBuildClassUtils {
    /**
     * 缓存Class类对象
     */
    private Map> cache = new HashMap<>();
    /**
     * 缓存代码
     */
    private Map codeCache = new HashMap<>();

    private GroovyClassLoader classLoader;

    public GroovyAutoBuildClassUtils() {
        //初始化GroovyClassLoader类加载器
        classLoader = new GroovyClassLoader();
    }

    /**
     * 使用groovy将一段代码动态构造成一个类,构造类时请指定一个classUuid,这个字段用来减少类的重复创建问题
     *
     * @param classUuid 接口的唯一标识
     * @param code      接口对应代码
     * @return
     */
    public Class buildClass(String classUuid, String code) {
        //如果代码有改变过或者没有加载过,都必须重新加载
        if (code == null || "".equals(code) || NHStringUtils.isEmpty(classUuid)) {
            return null;
        }
        Class clazz = cache.get(classUuid);
        if (clazz == null || !code.equals(codeCache.get(classUuid))) {
            clazz = classLoader.parseClass(code);
            cache.put(classUuid, clazz);
            codeCache.put(classUuid, code);
        }
        return clazz;
    }

}

 【d】使用示例

public class TestGroovy {

    public static void main(String[] args) {
        GroovyAutoBuildClassUtils groovy = new GroovyAutoBuildClassUtils();
        try {
            //classUUid: 需要保证唯一
            //code:通常从数据库查询获取,为了做测试直接将代码放进去
            Class clazz = groovy.buildClass("test", "package com.ly.cloud.utils;\n" +
                    "\n" +
                    "public class GroovyTestImpl implements GroovyTest {\n" +
                    "    @Override\n" +
                    "    public int sum(int a, int b) {\n" +
                    "        return a + b;\n" +
                    "    }\n" +
                    "}");
            GroovyTest groovyTest = (GroovyTest) clazz.newInstance();
            int sum = groovyTest.sum(1, 2);
            System.out.println("运算结果: " + sum);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

 运行结果:

可见,成功调用接口返回数据,这样就实现了通过Groovy动态将String类型的java代码编译成一个Class对象,然后我们就可以利用反射生成类的实例,进而调用其方法实现业务处理。

你可能感兴趣的:(Java后端)