GROOVY集成到JAVA中遇到的一个大坑

groovy与java集成的方式

首先说一下我用groovy动态脚本的场景,每次输入一组数据,分别对其中的不同部分做不同运算,而且这些运算会随着业务需求的改变而变化,因此选择在数据库中放置groovy脚本的方式来处理。各大博客中一般在java代码中集成groovy动态脚本的一般有三种方式:

GroovyClassLoader

GroovyShell

GroovyScriptEngine


开始时我采用的是第一种方式,一般代码是这样的:

GroovyClassLoader loader = new GroovyClassLoader();
Class groovyClass = loader.parseClass(script);
GroovyObject groovyObject = (GroovyObject) groovyClass.newInstance();

groovyObject.invokeMethod("run", "helloworld")

在本机运行没有丝毫问题。

但是,在测试服务器中运行,就会遇到问题:

org.springframework.transaction.TransactionSystemException: Could not commit JPA transaction; nested exception is javax.persistence.RollbackException: Transaction marked as rollbackOnly

显示事务回滚。堆栈信息中也是一些事务回滚的一些信息。仔细检查业务逻辑,也没问题,毕竟本地测试是通过的。

通过一行一个log的方式,终于定位到问题之所在:

Class groovyClass = loader.parseClass(script);

之后将整个groovy脚本解析部分放入try catch,发现可以catch住异常,但是无法打印堆栈信息,也就是

e.printStackTrace() 没有效果。为什么没有堆栈信息?!这就让我陷入了深深的苦恼中。

本地可以正常运行,测试服务器不可以。没有什么好的办法,本着试一试的想法,尝试了第二种方案:

GroovyShell shell = new GroovyShell();
Script groovyScript = shell.parse(script);

groovyScript.invokeMethod("compute", object);

这次测试服务器可以正常通过测试。很激动,因为groovy脚本的方案不用大改动了!


回过头来想一想,为何同一个代码,两处反应不一致?JAVA版本都是1.8,细心的我发现了这个

JVM的版本不一样。手动(* ̄︶ ̄)。