远程服务调用(RMI)
模块概念的引入,是本框架的一大优势,而跨JVM的远程服务调用则是另一个最有价值的功能。
《本地服务调用》一文中我们讲解了跨模块间的服务调用可以是这样的:
ServiceHelper.invoke("pas","AuthService:auth",new Data("principal",principalInstance,"url","http://localhost:8080/pas/index.shtml"));
如果这个pas不是本地模块,而是远程模块,又该如何调用呢?
正如你期望的那样,还是上面这样调用,这意味着,无论这个pas模块部署在什么地方,你得代码复杂度没有变化。可怜了那些靠代码量体现工作量的程序猿们,处理了一个如此复杂的工作,还是这么简单的两行代码。
远程服务的实现方法
flying中是通过hessian来实现远程调用的,主要是因为hessian简单、功能够用。
服务提供端代码:
1、定义Hessian服务提供接口
public interface RemoteService {
RemoteValue invoke(Principal principal, String moduleId, String serviceId, RemoteValue request) throws Exception;
}
2、定义Hessian的Servlet
@WebServlet(value="/remoting")
public class HessianRemoteService extends HessianServlet implements RemoteService{
private final static Logger logger = Logger.getLogger(HessianRemoteService.class);
public void init(ServletConfig servletConfig) throws ServletException {
super.init(servletConfig);
}
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws IOException, ServletException {
HttpServletRequest req = (HttpServletRequest)servletRequest;
long start = System.currentTimeMillis();
final String requestURI = req.getRequestURI() + (req.getQueryString() == null?"":"?" + req.getQueryString());
try {
super.service(servletRequest, servletResponse);
} finally {
logger.info("Access(" + (System.currentTimeMillis() - start) + ")\tURI:" + requestURI);
}
}
public RemoteValue invoke(Principal principal, String moduleId, String serviceId, RemoteValue remoteData) throws Exception {
long start = System.currentTimeMillis();
try {
Data req = remoteData.getValue();
LocalModule module = Application.getInstance().getModules().getLocalModule(moduleId);
ThreadContext.getContext().reset(module, serviceId, req, principal);
ThreadContext.getContext().setInvokeType(ThreadContext.InvokeType.Remote);
Data result = module.invoke(serviceId, req);
remoteData.setValue(result);
} catch (Exception e){
remoteData.setException(e);
} finally {
logger.info("RemoteInvoker(" + (System.currentTimeMillis() - start) + ")\tModuleId:" + moduleId+";ServiceId:" + serviceId);
}
return remoteData;
}
}
3、实现Hessian调用的客户端
public class HessianRemoteServiceInvoker implements RemoteServiceInvoker {
@Override
public Data invoke(Principal principal, LocalModule localModule, String remoteModuleId, String serviceId, Data request)
throws Exception {
if (localModule == null) {
localModule = ThreadContext.getContext().getModule();
}
if (localModule != null) {
Thread.currentThread().setContextClassLoader(localModule.getClassLoader());
}
String url = Application.getInstance().getModules().getRemoteModule(remoteModuleId).getPath();
RemoteService remoteService = getRemoteService(url);
Map<String, Object> nonSerializable = request.filterValues(new DataFilter(){
@Override
public boolean isValid(String key, Object value) {
return key != null && value != null && (value instanceof Serializable);
}
});
final RemoteValue res = remoteService.invoke(principal, remoteModuleId, serviceId, new RemoteValue(request));
if (res.getException() != null) {
throw (Exception) res.getException();
} else {
return res.getValue().putAll(nonSerializable);
}
}
private static Map<String,RemoteService> remoteServiceMap = new ConcurrentHashMap<String, RemoteService>();
private static Object lock = new Object();
private RemoteService getRemoteService(String url) throws Exception {
RemoteService remoteService = remoteServiceMap.get(url);
if(remoteService==null){
synchronized (lock) {
remoteService = remoteServiceMap.get(url);
if(remoteService==null){
HessianProxyFactory factory = new HessianProxyFactory(Thread.currentThread().getContextClassLoader());
HttpClientHessianConnectionFactory hessianConnectionFactory = HttpClientHessianConnectionFactory.getInstance();
//hessianConnectionFactory.addHeader("authorization", "admin");
factory.setConnectionFactory(hessianConnectionFactory);
final Data config = Application.getInstance().getConfigs("hessian");
if (config != null) {
factory.setConnectTimeout(config.getLong("connectTimeout", 10000l));
factory.setReadTimeout(config.getLong("readTimeout", 10000l));
factory.setHessian2Reply(config.getBoolean("hessian2Reply", true));
factory.setHessian2Request(config.getBoolean("hesian2Request", false));
factory.setChunkedPost(config.getBoolean("chunkedPost", true));
factory.setDebug(config.getBoolean("debug", false));
}
remoteService = (RemoteService) factory.create(RemoteService.class, url);
remoteServiceMap.put(url, remoteService);
}
}
}
return remoteService;
}
}
4、实现RemoteModule中的invoke方法:
public Data invoke(String serviceId, Data request) throws Exception {
return RemoteServiceInvokerHelper.invoke(remoteServiceInvoker, id, serviceId, request);
}
框架源码:https://github.com/hifong/flying
博客空间:http://www.cnblogs.com/hifong/
Demo应用:https://github.com/hifong/pas
技术QQ群:455852142
远程服务调用(RMI)的更多相关文章
- 远程服务调用RMI框架 演示,和底层原理解析
远程服务调用RMI框架: 是纯java写的, 只支持java服务之间的远程调用,很简单, // 接口要继承 Remote接口 public interface IHelloService extend ...
- spring boot / cloud (十四) 微服务间远程服务调用的认证和鉴权的思考和设计,以及restFul风格的url匹配拦截方法
spring boot / cloud (十四) 微服务间远程服务调用的认证和鉴权的思考和设计,以及restFul风格的url匹配拦截方法 前言 本篇接着<spring boot / cloud ...
- dubbo专题」dubbo其实很简单,就是一个远程服务调用的框架(1)
一.dubbo是什么? 1)本质:一个Jar包,一个分布式框架,,一个远程服务调用的分布式框架. 既然是新手教学,肯定很多同学不明白什么是分布式和远程服务调用,为什么要分布式,为什么要远程调用.我简单 ...
- 远程服务调用RPC框架介绍,微服务架构介绍和RPC框架对比,dubbo、SpringClound对比
远程服务调用RPC框架介绍,微服务架构介绍和RPC框架对比,dubbo.SpringClound对比 远程服务调用RPC框架介绍,RPC简单的来说就是像调用本地服务一样调用远程服务. 分布式RPC需要 ...
- 【Dubbo源码阅读系列】之远程服务调用(上)
今天打算来讲一讲 Dubbo 服务远程调用.笔者在开始看 Dubbo 远程服务相关源码的时候,看的有点迷糊.后来慢慢明白 Dubbo 远程服务的调用的本质就是动态代理模式的一种实现.本地消费者无须知道 ...
- 分布式远程服务调用(RPC)框架
分布式远程服务调用(RPC)框架 finagle:一个支持容错,协议无关的RPC系统 热门度(没变化) 10.0 活跃度(没变化) 10.0 Watchers:581 Star:6174 Fork: ...
- maven聚合项目以及使用dubbo远程服务调用debug操作。
1.maven聚合项目以及使用dubbo远程服务调用debug操作. 然后操作如下所示: 然后如下所示: 启动断点所在的包的服务.以debug的形式启动. 断点进来的效果如下所示: 接下来请继续你的表 ...
- RPC远程服务调用
RPC远程服务调用: RPC 的全称是 Remote Procedure Call 是一种进程间通信方式. 它允许程序调用另一个地址空间(通常是共享网络的另一台机器上)的过程或函数,而不用程序员显式编 ...
- 分布式应用开发 | SpringBoot+dubbo+zookeeper实现服务注册发现 | 远程服务调用
前言 通过新建两个独立服务--提供者.消费者,模拟两个独立分布的应用,通过使用dubbo+zookeeper来实现远程服务调用. 目录 项目搭建 provider-server consumer-se ...
随机推荐
- Python基础-*args和**kwargs魔法变量
在学习Python时,总会遇到*args和**kwargs这两个魔法变量,那么它们到底是什么? 首先,并不是必须写成*args和**kwargs.只有变量前面的*(星号)才是必须的,你也可以写成*va ...
- 自学Aruba2.3-Aruba Web UI --Configuration面板介绍
点击返回:自学Aruba之路 自学Aruba2.3-Aruba Web UI --Configuration面板介绍 此文只展示重要面板,大部分通俗易懂就不过多语言介绍, 后期配置实例中再结合理论知识 ...
- JSP动态产生的代码,点击显示确认操作,根据操作跳转并传值。
假如有如下一段代码产生了多个选项我们改如何获得点击删除一项中的值? <%List<Theme> themelist=(List<Theme>)request.getAtt ...
- HTML知识点总结之table
table元素 table用来创建表格,表格也可以用来布局,但是嵌套过于复杂,不利于灵活布局,已经几乎没人用它来布局了. 表格基本上有如下几个标签构成: (1)<table>标签用来创建表 ...
- .net 平台下, Socket通讯协议中间件设计思路(附源码)
.net 平台下,实现通讯处理有很多方法(见下表),各有利弊: 序号 实现方式 特点 1 WCF 优点:封装好,方便.缺点:难学,不跨平台 2 RocketMQ,SuperSocket等中间件 优点: ...
- 详说 Navicat for MySQL 快捷键
详说 Navicat for MySQL 快捷键: Navicat 主窗口 Navicat 主窗口快捷键 常规 Navicat 常规快捷键 表设计器 Navicat 表设计器快捷键 表查看器 Navi ...
- 探讨 java中 接口和对象的关系
接口是对象么?接口可以有对象么?这个问题要跟类比对着,或许更好理解;类是对象的模版.接口不是类,所以:接口肯定不是对象的模版.那接口跟对象有什么样的关系?还是得从类入手;因为类实现了接口,所以可以说, ...
- junit测试模板 unit-test
一个项目能否发布上线,重要的环节就是测试.经过集成测试.性能测试.压力测试等不断循环的测试过后依据测试报告来确定上线.这些由专业的测试人员来完成,因此会导致程序开发者对自身的单元测试的弱化.若在代码中 ...
- [搬运] C# 这些年来受欢迎的特性
原文地址:http://www.dotnetcurry.com/csharp/1411/csharp-favorite-features 在写这篇文章的时候,C# 已经有了 17 年的历史了,可以肯定 ...
- 2017 年“认证杯”数学中国数学建模网络挑战赛 C题思路讲解
之前有小伙伴私信我叫我说说这次比赛C题的思路,怎么写的,我就写篇博客说说吧,仅供参考! 针对C题,该题目比较综合,是一个成熟的数模赛题,与国赛的相似性较高.一般而言,第一问难度较低,题目要求进行数据挖 ...