一、简介
在项目中,有时候需要将动态调用Java代码,但是代码可能是通过富文本编辑器保存在库里面的,并且使用CLOB数据类型保存。那么在使用的时候,我们必然需要将库里面的代码动态构造成一个Class类,然后通过反射调用类的方法实现功能。下面总结一下如何使用Groovy将一段代码动态构造成一个类。
二、使用方法
【a】pom文件引入groovy依赖包
<dependency>
<groupId>org.codehaus.groovy</groupId>
<artifactId>groovy-all</artifactId>
</dependency>
【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<String, Class<?>> cache = new HashMap<>();
/**
* 缓存代码
*/
private Map<String, Object> 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对象,然后我们就可以利用反射生成类的实例,进而调用其方法实现业务处理。