使用javascript让项目支持热插拔 原

    突然想起之前做过的一个小项目,项目虽小,需求却不小,要求解析特定格式的字符串,并且特定格式并非一成不变,想要一套系统能够支持解析多变的规则且更改规则时不能起停服务。

    java先天因素决定了他不能支持热加载,所以就着手从其它语言结合使用,于是javascript就被揪出来了,编写容易,与java易集成。下面是对其简单做了封装,使其通用易迁移。

IJSEngine.java 核心接口

package com.voole.p2pauth.util.js.service;

import java.util.Map;

public interface IJSEngine {

	
	String getStringReturn(String jsFile, String functionname);

	
	String getStringReturn(String jsFile, String functionname, Object... args);

	String getStringReturnByRealPath(String jsFile, String functionname);

	
	String getStringReturnByRealPath(String jsFile, String functionname, Object... args);

	
	Boolean addGlobleJavaObj(Map<String, Object> globleJavaObjMap);
}

JSEngineImpl.java 接口实现

package com.voole.p2pauth.util.js.center;

import java.io.BufferedReader;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.HashMap;
import java.util.Map;
import java.util.Map.Entry;

import javax.script.Invocable;
import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
import javax.script.ScriptException;

import org.apache.log4j.Logger;

import com.voole.p2pauth.util.js.service.IJSEngine;

public class JSEngineImpl implements IJSEngine {
	private static final Logger LOGGER = Logger.getLogger(JSEngineImpl.class);
	private static final ScriptEngineManager manager = new ScriptEngineManager();
	private static final ScriptEngine engine = manager.getEngineByName("javascript");
	private Map<String, Object> globleJavaObj = new HashMap<>();

	private boolean evalJSByClassPathFile(String jsfileName) {
		try {
			ClassLoader classLoader = JSEngineImpl.class.getClassLoader();
			InputStream reader = classLoader.getResourceAsStream("rule/" + jsfileName + ".js");
			if (null == reader) {
				return false;
			}
			if (null != this.globleJavaObj && this.globleJavaObj.size() > 0) {
				for (Entry<String, Object> obj : this.globleJavaObj.entrySet()) {
					engine.put(obj.getKey(), obj.getValue());
				}
			}
			engine.eval(new BufferedReader(new InputStreamReader(reader)));
			return true;
		} catch (ScriptException e) {
			LOGGER.error(e.getMessage(), e);
			// e.printStackTrace();
			return false;
		}
	}

	private boolean evalJSByRealPathFile(String jsfileName) {
		try {
			if (null != this.globleJavaObj && this.globleJavaObj.size() > 0) {
				for (Entry<String, Object> obj : this.globleJavaObj.entrySet()) {
					engine.put(obj.getKey(), obj.getValue());
				}
			}
			engine.eval(new BufferedReader(new FileReader(jsfileName)));
			return true;
		} catch (ScriptException | FileNotFoundException e) {
			LOGGER.error(e.getMessage(), e);
			// e.printStackTrace();
			return false;
		}
	}

	@Override
	public String getStringReturn(String jsFile, String functionname) {
		if (this.evalJSByClassPathFile(jsFile)) {
			if (engine instanceof Invocable) {
				return this.invoke(functionname);
			}
		}
		return null;
	}

	
	@Override
	public String getStringReturn(String jsFile, String functionname, Object... args) {
		if (this.evalJSByClassPathFile(jsFile)) {
			if (engine instanceof Invocable) {
				return this.invoke(functionname, args);
			}
		}
		return null;
	}

	@Override
	public String getStringReturnByRealPath(String jsFile, String functionname) {
		if (this.evalJSByRealPathFile(jsFile)) {
			if (engine instanceof Invocable) {
				return this.invoke(functionname);
			}
		}
		return null;
	}

	@Override
	public String getStringReturnByRealPath(String jsFile, String functionname, Object... args) {
		if (this.evalJSByRealPathFile(jsFile)) {
			if (engine instanceof Invocable) {
				return this.invoke(functionname, args);
			}
		}
		return null;
	}

	private <T> T invoke(String funcname, Object... args) {
		Invocable invocable = (Invocable) engine;
		try {
			return (T) invocable.invokeFunction(funcname, args);
		} catch (NoSuchMethodException | ScriptException e) {
			LOGGER.error(e.getMessage(), e);
			// e.printStackTrace();
			return null;
		}
	}

	
	@Override
	public Boolean addGlobleJavaObj(Map<String, Object> globleJavaObjMap) {
		this.globleJavaObj = globleJavaObjMap;
		return true;
	}

}

调用

public static void main(String[] args) {
	IJSEngine ENGINE = SccUtilFactory.instanceJSEngine();
	// 设置js全局变量
	Map<String, Object> globleJavaObj = getGlobleObj();
	ENGINE.addGlobleJavaObj(globleJavaObj);
	// 执行js规则解析,并返回解析后的playurl
	String play_url = ENGINE.getStringReturn(test.jd,……);
}

js文件

function auth_validate_1_0(data){
	LOGGER.info("data:"+data);
	var jdata=eval('(' + data + ')');
	var urlObj=eval('(' + JsUtil.urlToObject(jdata.url) + ')');
	
}

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏菩提树下的杨过

Flash/Flex学习笔记(8):ActionScript3.0中的面对对象

首先要习惯AS3.0的几个BT约定: 1.一个.as文件中,只能定义一个类 2.类名称必须与.as的文件名相同 3.类定义中必须要有package包声明 4.一...

18690
来自专栏java、Spring、技术分享

fastjson详解

  fastjson用于将Java Bean序列化为JSON字符串,也可以从JSON字符串反序列化到JavaBean。

94210
来自专栏WindCoder

MyBatis传入参数为集合 list 数组 map写法

这几天需要or和拼接in的特定查询条件来做查询,想看看mybatis是否可以通过传递list集合实现,于是找到了他的foreach标签。

4.9K20
来自专栏androidBlog

归并排序 递归版和非递归版的实现(java)

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/gdutxiaoxu/article/details/...

29810
来自专栏10km的专栏

fastjson:javabean按字段(field)序列化存储为Map并反序列化

大部分json工具对java对象整体序列化都提供了简单的调用方式,以fastjson为例: Model model = new Model(); String ...

40050
来自专栏小白的技术客栈

Python之递归函数

Python之递归函数 好久没有更新内容了,也好久没有给大家打个招呼了,小白想死你们了。今天跟大家说说Python中的递归函数。 Python是支持递归函数的...

42960
来自专栏cs

串匹配算法

问题:给定二个字符串S和T,在主串S中查找子串T的过程称之为字符串匹配问题(string matching,也称之为模式匹配)。在文本处理系统,操作系统,编译系...

365100
来自专栏JadePeng的技术博客

\x 开头编码的数据解码成中文

在python里,直接decode('utf-8')即可 >>> "\xE5\x85\x84\xE5\xBC\x9F\xE9\x9A\xBE\xE5\xBD\x...

1.6K120
来自专栏java初学

java异常处理及自定义异常的使用

15570
来自专栏xingoo, 一个梦想做发明家的程序员

Oozie分布式工作流——EL表达式

oozie支持使用EL(expression language)表达式。 基本的EL常量 KB MB GB TB PB 基本EL函数 string fir...

27780

扫码关注云+社区

领取腾讯云代金券