当前位置:   article > 正文

Groovy编写规则引擎学习_groovy规则引擎

groovy规则引擎

一、Groovy与Java集成
Groovy脚本引擎的执行本质只是接受context对象,然后基于context对象中的关键信息进行逻辑判断,输出结果。
参考文章:
文章1
文章2
文章3
文章4

Java中运行Groovy
在java中运行Groovy脚本,有三种比较常用的类支持:GroovyShell、GroovyClassLoader 以及 Java-Script引擎(JSR-223).

GroovyShell
GroovyShell允许在Java类中(甚至Groovy类)求任意Groovy表达式的值。您可使用Binding对象输入参数给表达式,并最终通过GroovyShell返回Groovy表达式的计算结果。
通常用来运行"script片段"或者一些零散的表达式(Expression)

GroovyClassLoader
用 Groovy 的 GroovyClassLoader ,它会动态地加载一个脚本并执行它。GroovyClassLoader是一个Groovy定制的类装载器,负责解析加载Java类中用到的Groovy类。
如果脚本是一个完整的文件,特别是有API类型的时候,比如有类似于JAVA的接口,面向对象设计时,通常使用GroovyClassLoader.

GroovyScriptEngine
GroovyShell多用于推求对立的脚本或表达式,如果换成相互关联的多个脚本,使用GroovyScriptEngine会更好些。GroovyScriptEngine从您指定的位置(文件系统,URL,数据库,等等)加载Groovy脚本,并且随着脚本变化而重新加载它们。如同GroovyShell一样,GroovyScriptEngine也允许您传入参数值,并能返回脚本的值。

首先导入Groovy包

<properties>
    <groovy.version>2.5.6</groovy.version>
</properties>
<dependency>
    <groupId>org.codehaus.groovy</groupId>
    <artifactId>groovy-all</artifactId>
    <version>${groovy.version}</version>
</dependency>

注意:我的是这样导入包的,否则编译运行时会报异常:java.lang.ClassNotFoundException:org.codehaus.groovy.ast.MethodCallTransformation
参考解答:解决1 和 解决2
说白了就是springboot2.x版本和groovy2.5.x不能完全兼容。

二、SpringBoot动态运行groovy脚本
该小节是在参考这篇文章后,加入一些自己的理解完成的。

简介
在SpringBoot项目中,通过容器和依赖注入,可以很方便的实现开发功能。那么如何通过加载实例来动态编译脚本呢。
groovy支持通过GroovyShell预设对象,在groovy动态脚本中直接调用预设对象的方法。因此我们可以通过将spring的bean预设到GroovyShell运行环境中,在groovy动态脚本中直接调用spring容器中bean来调用其方法。

项目目录


项目解释
1、首先,自定义注解@GroovyFunction。用来标识用于绑定到GroovyShell的类。

import java.lang.annotation.*;

/**
 * @author Caocs
 * @date 2020/3/12
 */
@Retention(RetentionPolicy.RUNTIME)
@Target(value = {ElementType.TYPE})
@Documented
public @interface GroovyFunction {
}

2、service层用来编写自定义的方法,用@GroovyFunction注解标识该方法可以被绑定到GroovyShell中。

import org.springframework.stereotype.Service;

/**
 * @author Caocs
 * @date 2020/3/12
 */
@Service
@GroovyFunction
public class TestService {
    public String testQuery(long id) {
        return "Test query success, id is " + id;
    }
}

3、然后,利用SpringBoot中的Configuration类来设置Binding
首先配置类实现org.springframework.context.ApplicationContextAware接口,用来获取应用上下文。然后在配置从上下文获取到的指定Bean实例,并注入到groovy的Binding中。
这里,我是利用上文提到的@GroovyFunction注解来过滤需要的实例。i

  1. mport groovy.lang.Binding;
  2. import org.springframework.beans.BeansException;
  3. import org.springframework.context.ApplicationContext;
  4. import org.springframework.context.ApplicationContextAware;
  5. import org.springframework.context.annotation.Bean;
  6. import org.springframework.context.annotation.Configuration;
  7. import java.util.Map;
  8. /**
  9.  * @author Caocs
  10.  * @date 2020/3/12
  11.  */
  12. @Configuration
  13. public class GroovyBindingConfig implements ApplicationContextAware {
  14.     // 实现ApplicationContextAware接口后的方法类,可以获取Spring中已经实例化的bean
  15.     private ApplicationContext applicationContext;
  16.     
  17.     @Override
  18.     public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
  19.         this.applicationContext = applicationContext;
  20.     }
  21.     
  22.     /**
  23.      * 将标注@GroovyFunction注解的类对象绑定到Binding中,并在spring容器中实例化出一个对象
  24.      * @return
  25.      */
  26.     @Bean("groovyBinding")
  27.     public Binding groovyBinding() {
  28.         Binding groovyBinding = new Binding();
  29.         // 根据注解过滤掉不需要的实例
  30.         Map<String, Object> beanMap = applicationContext.getBeansWithAnnotation(GroovyFunction.class);
  31.         for (String beanName : beanMap.keySet()) {
  32.             groovyBinding.setVariable(beanName, beanMap.get(beanName));
  33.         }
  34.         return groovyBinding;
  35.     }
  36. }


4、然后,定义一个请求类
这个其实没啥好说的,因为我不会再controller层请求两个参数。

import java.util.Map;

/**
 * @author Caocs
 * @date 2020/3/12
 */
public class ScriptRequest {
    private String expression;
    private Map<String, Object> paramMap;

    @Override
    public String toString() {
        return "ScriptRequest{" +
                "expression='" + expression + '\'' +
                ", paramMap=" + paramMap +
                '}';
    }

    public String getExpression() {
        return expression;
    }

    public void setExpression(String expression) {
        this.expression = expression;
    }

    public Map<String, Object> getParamMap() {
        return paramMap;
    }

    public void setParamMap(Map<String, Object> paramMap) {
        this.paramMap = paramMap;
    }
}

5、最后,在controller层中,实现动态脚本运行
注意:下面是我的一些想法和实现
(1)采用单例模式,将GroovyShell在初始化时就实例化出来。以后每次都直接调用该实例。
(2)通过依赖注入,将已经绑定自定义方法的Binding实例注入进来
(3)维护一个HashTable,用于存放~~<expression,Script>~~ 的映射关系,这样就可以重复利用已经实例化过的Script的实例。
(4)多个请求同时操作HashTable,所以需要注意线程安全问题。

  1. import groovy.lang.Binding;
  2. import groovy.lang.GroovyShell;
  3. import groovy.lang.Script;
  4. import org.codehaus.groovy.control.CompilerConfiguration;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.stereotype.Controller;
  7. import org.springframework.web.bind.annotation.RequestBody;
  8. import org.springframework.web.bind.annotation.RequestMapping;
  9. import org.springframework.web.bind.annotation.RequestMethod;
  10. import org.springframework.web.bind.annotation.ResponseBody;
  11. import java.util.Hashtable;
  12. import java.util.Map;
  13. /**
  14.  * @author Caocs
  15.  * @date 2020/3/12
  16.  */
  17. // @RestController
  18. @Controller
  19. @RequestMapping("/groovy/single/script")
  20. public class SingleScriptController {
  21.     private static final Object lock = new Object();
  22.     private static final GroovyShell groovyShell;
  23.     private static Hashtable<String, Script> scriptCache = new Hashtable<>();
  24.     @Autowired
  25.     private Binding groovyBinding; // 默认绑定已有方法的实例
  26.     static {
  27.         CompilerConfiguration cfg = new CompilerConfiguration();
  28.         groovyShell = new GroovyShell(cfg);
  29.     }
  30.     /**
  31.      * 在客户端本地只实例化单例RuleExecutor
  32.      * 然后多个线程同时操作scriptCache,需要保证线程安全。
  33.      *
  34.      * @param expression
  35.      * @return
  36.      */
  37.     private Script getScriptFromCache(String expression) {
  38.         if (scriptCache.containsKey(expression)) {
  39.             return scriptCache.get(expression);
  40.         }
  41.         synchronized (lock) {
  42.             if (scriptCache.containsKey(expression)) {
  43.                 return scriptCache.get(expression);
  44.             }
  45.             Script script = groovyShell.parse(expression);
  46.             scriptCache.put(expression, script);
  47.             return script;
  48.         }
  49.     }
  50.     public Object ruleParse(String expression) {
  51.         Script script = getScriptFromCache(expression);
  52.         script.setBinding(groovyBinding);
  53.         return script.run();
  54.     }
  55.     public Object ruleParse(String expression, Map<String, Object> paramMap) {
  56.         Binding binding = groovyBinding;
  57.         paramMap.forEach((key,value)->binding.setProperty(key,value));
  58.         Script script = getScriptFromCache(expression);
  59.         script.setBinding(binding);
  60.         return script.run();
  61.     }
  62.     @WatchAspect
  63.     @RequestMapping(value = "/execute", method = RequestMethod.POST)
  64.     public @ResponseBody
  65.     Object ruleExecutor(@RequestBody ScriptRequest request) {
  66.         if (request.getParamMap() == null) {
  67.             return ruleParse(request.getExpression());
  68.         } else {
  69.             return ruleParse(request.getExpression(), request.getParamMap());
  70.         }
  71.     }
  72. }


使用postman进行测试,定义一个Post请求,执行后结果如下。
测试1:

测试2:

最后,因为个人能力问题,欢迎指正。

2020-03-26
突然发现代码会有问题(GC和线程安全问题)
详细内容参考:

JVM执行Groovy脚本导致堆外内存溢出问题排查
https://www.liangzl.com/get-article-detail-161916.html
————————————————
版权声明:本文为CSDN博主「愚愚是个大笨蛋」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_29698805/article/details/104639140

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/2023面试高手/article/detail/386584
推荐阅读
相关标签
  

闽ICP备14008679号