Groovy小结:java调用Groovy方法并传递参数

@(JAVA总结)

1. 场景描述

在网上查了资料发现,java有三种方式调用groovy脚本。但是真正在实际的服务器环境中,嵌入groovy脚本往往需要满足下面的条件:

  1. 可以直接调用groovy脚本中的方法
  2. 能传递对象到groovy方法中,不仅仅是字符串
  3. 提供脚本缓存机制,不用每次调用脚本的时候,都到磁盘读取
  4. 修改groovy后能实时生效

只有满足了上面的这些要求,才能安心的将其嵌入到现有的Java后台服务中。
下面就来具体探讨下具体实现的步骤。

2. 解决方案

其实,GroovyScriptEngine类就已经提供了上面所说的功能。
主要使用GroovyScriptEngine.loadScriptByName来读取脚本,loadScriptByName方法内部提供了缓存功能,在读取groovy脚本的时候,会优先从缓存中读取,如果缓存中没有的话,才去读取脚本,如下:
Alt text

2.1 相关测试类和脚本

在后面的测试后,会用到下面的java类和groovy脚本。

2.1.1 测试类Person.java

该类用于测试传递Java对象到Groovy脚本中

public class Person {
	public String name;
	public String address;
	public Integer age;
	
	public Person(String name, String addr, Integer age){
		this.name = name;
		this.address = addr;
		this.age = age;
	}
	
	public String toString(){
		return String.format("[Person: name:%s, address:%s, age:%s]", name,address, age);
	}
}

2.1.2 测试脚本hello2.groovy

下面脚本中的两个方法用于测试方法的无参调用和带参调用

def helloWithoutParam(){
	println "start to call helloWithoutParam!"
	return "success, helloWithoutParam";
}

def helloWithParam(person, id){
	println "start to call helloWithParam, param{person:" + person + ", id:" + id + "}";
	return "success, helloWithParam";
}

2.2 java调用Groovy脚本方法(无参)

public static void testGroovy2(){
    try {  
    	Class scriptClass = groovyScriptEngine.loadScriptByName("hello2.groovy");
    	GroovyObject scriptInstance = (GroovyObject)scriptClass.newInstance();
    	Object ret = scriptInstance.invokeMethod("helloWithoutParam", null);
    	System.out.println("testGroovy2:" + ret);
    } catch (Exception e) {  
        e.printStackTrace();  
        System.out.println("Exception e="+e.toString());  
    } 
}

执行结果:
start to call helloWithoutParam!
testGroovy2: success, helloWithoutParam

2.3 java调用Groovy脚本方法(带参)

@SuppressWarnings({ "rawtypes" })
public static void testGroovy3(){
    try {  
    	Person person = new Person("wchi", "nanjing", 30);
		Class scriptClass = groovyScriptEngine.loadScriptByName("hello2.groovy");
		GroovyObject scriptInstance = (GroovyObject)scriptClass.newInstance();
		Object ret = scriptInstance.invokeMethod("helloWithParam", new Object[]{person,"lxi"});	
    	System.out.println("testGroovy3:" + ret);
    } catch (Exception e) {  
        e.printStackTrace();  
        System.out.println("Exception e="+e.toString());  
    } 
}

返回结果:
start to call helloWithParam, param{person:[Person: name:wchi, address:nanjing, age:30], id:lxi}
testGroovy3: success, helloWithParam

2.4 封装的公用类

可以将上面的代码封装成公用类,这样就方便很多,如下:

public class GroovyCommonUtil {
    private static final Logger log = LoggerFactory.getLogger(GroovyCommonUtil.class);
    //该变量用于指明groovy脚本所在的父目录
	static String root[]=new String[]{"bin/groovy/"};  
	static GroovyScriptEngine groovyScriptEngine;  
	
    static{
    	try {
			groovyScriptEngine=new GroovyScriptEngine(root);
		} catch (IOException e) {
			e.printStackTrace();
		}  
    }
    
	/**
	 *	用于调用指定Groovy脚本中的指定方法 
	 * @param scriptName	脚本名称
	 * @param methodName	方法名称
	 * @param params		方法参数
	 * @return
	 */
@SuppressWarnings({ "rawtypes"})
public Object invokeMethod(String scriptName, String methodName, Object... params) throws Exception{
	Object ret = null;
	Class scriptClass = null;
	GroovyObject scriptInstance = null;
	
	try {
		scriptClass = groovyScriptEngine.loadScriptByName(scriptName);
		scriptInstance = (GroovyObject)scriptClass.newInstance();
	} catch (ResourceException | ScriptException | InstantiationException | IllegalAccessException e1) {
		log.warn("加载脚本["+scriptName+"]出现异常", e1);
		throw new Exception("加载脚本"+scriptName+"失败");
	}

	try {
		ret = (String)scriptInstance.invokeMethod(methodName, params);
	} catch (IllegalArgumentException e) {
		log.warn("执行方法" + methodName + "参数出现异常, 参数为" + params, e);
		throw new Exception("调用方法[" + methodName + "]失败,因参数不合法");
	} catch(Exception e){
		log.warn("执行方法" + methodName + "出现异常", e);
		throw new Exception("调用方法[" + methodName + "]失败");
	}

	return ret;
}

使用上面的公用类,改写的测试代码如下:

/**
 * 测试没有参数的方法调用
 */
public static void testGroovyWithoutParam(){
	String result = (String)GroovyCommonUtil.invokeMethod("hello2.groovy", "helloWithoutParam");
	System.out.println("testGroovy4: " + result + "\n");
}

/**
 * 测试携带参数的方法调用
 */
public static void testGroovyWithParam(){
	Person person = new Person("wchi", "nanjing", 30);
	String result = (String)GroovyCommonUtil.invokeMethod("hello2.groovy", "helloWithParam", person, "testGroovy4");
	System.out.println("testGroovy4: " + result + "\n");
}
posted @ 2015-12-07 22:20 <默言> 阅读(17223) 评论(1) 推荐(0) 编辑
摘要: 网上能查到的mobicents的资料都是基于比较老的版本,而官网现在已经更新到3.0的版本,很多资料都已经无效,所以把自己的摸索过程记录下来,以便后来者能少走点弯路,快速上手。闲言少叙,开始正文。。。##安装文件1、基于jboss7.2的sip-servlets>mss-3.0.564-jboss-... 阅读全文
posted @ 2015-04-08 15:16 <默言> 阅读(1038) 评论(0) 推荐(0) 编辑
摘要: 问题描述:重装过一次系统,在重装之前git+tortoisegit配合很好,提交的时候都能自动加载ppk,但是重装系统后,也重新生成pulic key上传到了服务器,但是每次提交的时候都提示key加载失败,必须手工的将key加载到pageant中才可以正常提交问题分析:每个git项目的config文... 阅读全文
posted @ 2015-03-28 11:07 <默言> 阅读(1438) 评论(0) 推荐(0) 编辑
摘要: 笔记本:Levono E440问题描述:在控制面板中,点击鼠标时,弹出“无法连接synaptics定点装置驱动程序”错误,如何解决?即使在安装联想的驱动后,也没办法解决解决步骤:1、查看驱动文件安装位置:"开始“--”运行“--regedit(打开注册表)--依次打开HKEY_LOCAL_MACHI... 阅读全文
posted @ 2015-03-05 11:27 <默言> 阅读(1984) 评论(0) 推荐(0) 编辑
摘要: FreeSWITCH技巧:notify与message-waiting-----------@(Freeswitch经验点滴)###现象描述在客户端登陆抓包时,发现了FreeSWITCH发来的包:>NOTIFY sip:9988002@192.168.168.106:63984;transport=... 阅读全文
posted @ 2015-02-13 10:29 <默言> 阅读(2889) 评论(0) 推荐(1) 编辑
摘要: ##1、关于SIP的UDP与MTU的关系如果sip消息的大小超过了MTU,则有可能被网络中的某一节点分片,而UDP处理分片会有很大的问题,从而导致sip消息传输失败。要解决该问题的话,两种方案:1)减少sip消息的体积,比如减少codecs、x headers等2)使用tcp来替代udp传输sip消... 阅读全文
posted @ 2015-02-13 10:25 <默言> 阅读(7435) 评论(0) 推荐(0) 编辑
摘要: 主要讲解export的原因,以及根据原理如何在api中应用 阅读全文
posted @ 2014-11-10 21:04 <默言> 阅读(2838) 评论(0) 推荐(1) 编辑
摘要: 虽然现有的FreeSWITCH功能已经很强大,但是很多情况下,为了配合业务上的功能,还需要做一些定制开发。 有一个基本需求是:如何控制fs外呼,并跟踪外呼后的一系列状态。 阅读全文
posted @ 2014-11-09 16:55 <默言> 阅读(8772) 评论(1) 推荐(1) 编辑
摘要: 来自为知笔记(Wiz) 阅读全文
posted @ 2014-07-17 17:57 <默言> 阅读(1193) 评论(0) 推荐(0) 编辑
摘要: 在实际的应用中,经常有这样的需求,比如一个号码拨打外线,需要送dtmf出去(如拨打10086,根据提示按1按2等),在这种情况下,FreeSWITCH如果处理呢? 阅读全文
posted @ 2013-11-24 21:29 <默言> 阅读(3381) 评论(2) 推荐(2) 编辑
点击右上角即可分享
微信分享提示