Java 安装 Java:第一个程序 Hello World Java:建议使用 UTF-8 编写 Java 代码 Java:package 包命名规范 使用 Intellij IDEA 创建 Java 项目 Java 布尔类型 Java 处理日期和时间 Java 正则表达式 Java finalize 方法 Java:空值 null Java 如何触发垃圾回收 Java ThreadLocal Java InheritableThreadLocal Java Integer之间的比较 Java 动态代理 Java 匿名类 Java 枚举 Java 如何静态导入 import static println Java 引用级别:强引用、软引用、弱引用、幽灵引用 Java try finally return 解惑 Java WeakHashMap Java ReferenceQueue 怎么写 Java 示例代码? Java 匿名类双大括号初始化 什么是 Java Bean Java 多行字符串 Java 快速生成 List Java 快速生成 Map Java 将异常堆栈转换为 String JDK SPI 的使用和源码分析 Java Map 中的 key 和 value 能否为 null ? Java List 和 数组的互相转换 Java 获取环境变量 Java 获取和设置系统属性 Java:如何获取当前进程的 PID ? Java 字符串左侧 右侧补充空格或者其他字符 Java 线程 Java:如何获取文本文件内容 Java:读取资源文件内容 Java:使用 JavaFx 构建 GUI Java:Class 类 Java:使用 instanceof 判断对象类型 一个自定义的 Java 工具类 Java:获取当前函数所属类的类名 Java:获取当前执行的函数名 Java:使用 String 的 split 函数拆分字符串 Java:获取字符的 Unicode 编号(代码点) Java:获取当前工作目录 Java:使用 Class 对象的 isArray 方法判断对象是否为数组 使用 Java 生成 CSV 文件 Java Mockito 测试框架快速入门 JUnit 入门 JUnit 单测隔离 Java JOOR 反射库 Java alibaba transmittable-thread-local 库:让 ThreadLocal 跨线程传播 Java 日志组件 slf4j 的使用和源码分析 Java Lombok 库:为你减少样板代码 Java:使用 cglib 实现动态代理 Java Hibernate validator 校验框架 Java 使用 Hessian2 序列化和反序列化 H2 数据库快速入门 Java:使用 Gson 库处理 JSON 数据 Java 集成 groovy 构建规则引擎 Java 13:安装 Java 13 新特性:文本块(多行字符串) 卸载 MacOS 上安装的 Java Java:执行 sql 文件 Java JDK 有哪些发行版 ? java拾遗:String和数组 java拾遗:由反转数组想到System.out的实现机制 java拾遗:如何读取properties文件内容 Java并发概念汇总 java拾遗:System.out.println()是什么? java拾遗:通过示例理解位运算 使用“庖丁解牛”进行中文分词 DBUtils简明教程 试用velocity模板引擎 Java:将字符串哈希为数字 kafka SnappyError no native library is found 问题

Java 集成 groovy 构建规则引擎


#Java


引入 Groovy 依赖

在 build.gralde 中:

dependencies {
    // https://mvnrepository.com/artifact/org.codehaus.groovy/groovy-all
    compile group: 'org.codehaus.groovy', name: 'groovy-all', version: '2.5.9'
}

执行 groovy 脚本的几种方法

方法1

import groovy.lang.Binding;
import groovy.lang.GroovyShell;

public class Demo {

    public static void main(String[] args) {
        Binding binding = new Binding();

        // 绑定变量到 groovy 执行环境
        binding.setVariable("foo", new Integer(2));

        // 构建 groovy shell
        GroovyShell shell = new GroovyShell(binding);

        // 执行脚本
        Object value = shell.evaluate("println 'Hello World!'; x = 123; return foo * 10");

        // 执行结果
        System.out.printf("执行结果:%s, 类型: %s\n", value, value.getClass().getCanonicalName());

        // 获取执行过程中的变量
        System.out.printf("执行过程中的x变量:%s, 类型: %s\n", binding.getVariable("x"), binding.getVariable("x").getClass().getCanonicalName());
    }

}

执行结果:

Hello World!
执行结果:20, 类型: java.lang.Integer
执行过程中的x变量:123, 类型: java.lang.Integer

方法2

import groovy.lang.Binding;
import groovy.lang.GroovyShell;
import groovy.lang.Script;

public class Demo {

    public static void main(String[] args) {

        // 绑定变量到 groovy 执行环境
        Binding binding = new Binding();
        binding.setVariable("foo", new Integer(2));

        // 构建 groovy script
        GroovyShell shell = new GroovyShell();
        Script script = shell.parse("println 'Hello World!'; x = 123; return foo * 10");
        script.setBinding(binding);  // 绑定

        // 执行脚本
        Object value = script.run();

        // 执行结果
        System.out.printf("执行结果:%s, 类型: %s\n", value, value.getClass().getCanonicalName());

        // 获取执行过程中的变量
        System.out.printf("执行过程中的x变量:%s, 类型: %s\n", binding.getVariable("x"), binding.getVariable("x").getClass().getCanonicalName());
    }

}

执行结果:

Hello World!
执行结果:20, 类型: java.lang.Integer
执行过程中的x变量:123, 类型: java.lang.Integer

方法3

使用 InvokerHelper.createScript 创建脚本对象。

import groovy.lang.Binding;
import groovy.lang.GroovyClassLoader;
import org.codehaus.groovy.runtime.InvokerHelper;

public class Demo {

    public static void main(String[] args) {

        // 绑定变量到 groovy 执行环境
        Binding binding = new Binding();
        binding.setVariable("foo", new Integer(2));

        GroovyClassLoader groovyClassLoader = new GroovyClassLoader();
        Class scriptClass = groovyClassLoader.parseClass("println 'Hello World!'; x = 123; return foo * 10");

        // 执行脚本
        Object value = InvokerHelper.createScript(scriptClass, binding).run();

        // 执行结果
        System.out.printf("执行结果:%s, 类型: %s\n", value, value.getClass().getCanonicalName());

        // 获取执行过程中的变量
        System.out.printf("执行过程中的x变量:%s, 类型: %s\n", binding.getVariable("x"), binding.getVariable("x").getClass().getCanonicalName());
    }

}

方法4

执行 groovy 文件。

todo

示例1:在 groovy 脚本中执行 java 定义的函数

示例:

import groovy.lang.Binding;
import groovy.lang.GroovyShell;

public class Demo {

    public static class Func {

        public long add(long a, long b) {
            return a+b;
        }

    }

    public static void main(String[] args) {
        Binding binding = new Binding();
        binding.setVariable("func", new Func());
        GroovyShell shell = new GroovyShell(binding);
        Object value = shell.evaluate("return func.add(1, 1)");
        System.out.printf("执行结果:%s, 类型:%s\n", value, value.getClass().getCanonicalName());
    }

}

执行结果:

执行结果:2, 类型:java.lang.Long

示例2:在 groovy 脚本中执行 java 定义的静态函数

也可以执行java 中定义的静态方法:

import groovy.lang.Binding;
import groovy.lang.GroovyShell;

public class Demo {

    public static class Func {

        // 这是一个静态方法
        public static long add(long a, long b) {
            return a+b;
        }
    }

    public static void main(String[] args) {
        Binding binding = new Binding();
        binding.setVariable("func", Func.class);
        GroovyShell shell = new GroovyShell(binding);
        Object value = shell.evaluate("return func.add(1, 1)");
        System.out.printf("执行结果:%s, 类型:%s\n", value, value.getClass().getCanonicalName());

    }

}

示例3:传入上下文

Binding 本身也是一种上下文,但有时需要每一个java函数能感知到上下文,此时需要其他的实现方式。

传入上下文的方法有很多,比如作为函数参数,比如作为类的构造函数参数,比如放入 ThreadLocal 。

示例:

import groovy.lang.Binding;
import groovy.lang.Closure;
import groovy.lang.GroovyShell;

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


public class Demo {

    public static class Func {

        private Map<String, Object> context;

        // 上下文作为构造函数参数传入
        public Func(Map<String, Object> context) {
            this.context = context;
        }

        public long add(long a, long b) {
            System.out.println("add 方法上下文: " + context);
            return a+b;
        }
    }

    public static void main(String[] args) {

        // 构造上下文
        Map<String, Object> context = new HashMap<>();
        context.put("name", "lt");

        Binding binding = new Binding();
        binding.setVariable("func", new Func(context));

        GroovyShell shell = new GroovyShell(binding);

        Object value = shell.evaluate("return func.add(1, 1)");

        System.out.printf("执行结果:%s, 类型:%s\n", value, value.getClass().getCanonicalName());
    }
}

执行结果:

add 方法上下文: {name=lt}
执行结果:2, 类型:java.lang.Long

示例4:闭包

groovy 有一个闭包的概念,类似 java 中的 lambda 。

示例:

import groovy.lang.Binding;
import groovy.lang.Closure;
import groovy.lang.GroovyShell;

import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;


public class Demo {

    public static class Func {

        // 过滤函数,第2个参数是闭包
        public List<Integer> filter(List<Integer> list, Closure<Boolean> closure) {
            return list.stream().filter(x -> {
                return closure.call(x);
            }).collect(Collectors.toList());
        }

    }

    public static void main(String[] args) {
        Binding binding = new Binding();
        binding.setVariable("func", new Func());

        GroovyShell shell = new GroovyShell(binding);

        Object value = shell.evaluate("return func.filter([-1,2,3,4], {x -> x > 0} )");

        System.out.printf("执行结果:%s, 类型:%s\n", value, value.getClass().getCanonicalName());
    }
}

执行结果:

执行结果:[2, 3, 4], 类型:java.util.ArrayList

示例5:在 Java 中定义一个 Groovy 闭包类

import groovy.lang.Binding;
import groovy.lang.Closure;
import groovy.lang.GroovyShell;

public class Demo {


    // 这是一个 groovy 闭包
    public static class Add extends Closure<Long> {

        public Add() {
            super(null, null);
        }

        @Override
        public Long call(Object... args) {
            long result = 0;
            for (Object obj : args) {
                if (obj instanceof Integer) {
                    result = result + (Integer) obj;
                    continue;
                }

                if (obj instanceof Long) {
                    result = result + (Long) obj;
                    continue;
                }
            }
            return result;
        }



    }

    public static void main(String[] args) {
        Binding binding = new Binding();

        binding.setVariable("add", new Add());

        GroovyShell shell = new GroovyShell(binding);

        Object value = shell.evaluate("return add(1,2,3)");

        System.out.printf("执行结果:%s, 类型:%s\n", value, value.getClass().getCanonicalName());
    }

}

执行结果:

执行结果:6, 类型:java.lang.Long

示例6:在 groovy 中修改 java 变量

import groovy.lang.Binding;
import groovy.lang.GroovyShell;

import java.util.Arrays;
import java.util.List;


public class Demo {

    public static class Person {
        public String name;
        public int age;
        public List<String> books;

        public String getName() {
            return name;
        }

        @Override
        public String toString() {
            return "Person{" +
                    "name='" + name + '\'' +
                    ", age=" + age +
                    ", books=" + books +
                    '}';
        }
    }

    public static void main(String[] args) {
        Binding binding = new Binding();

        Person person = new Person();
        person.name = "lt";
        person.age = 10;
        person.books = Arrays.asList("book0", "book1");

        System.out.println("修改前: " + person);

        binding.setVariable("person", person);

        GroovyShell shell = new GroovyShell(binding);

        Object value = shell.evaluate("println 'in groovy books : ' + person.books; person.books[1] = 'new book1';return person.name");

        System.out.printf("执行结果:%s, 类型:%s\n", value, value.getClass().getCanonicalName());

        System.out.println("修改后: " + person);
    }
}

执行结果:

修改前: Person{name='lt', age=10, books=[book0, book1]}
in groovy books : [book0, book1]
执行结果:lt, 类型:java.lang.String
修改后: Person{name='lt', age=10, books=[book0, new book1]}

参考

  • http://ifeve.com/embedding-groovy/


( 本文完 )