用Jetty和redis实现接入服务器adapter
传统的服务器端为若干个客户端提供服务,一般需要开启多个服务器端进程。为了进一步提升服务器端的处理能力,可以如下图所示将服务解耦为两部分(adapter与workers),它们之间通过消息队列传输数据,其中workers处理具体业务,adapter负责接入请求以及反馈结果,具体包含下面两个工作。
1,将所有客户端的请求发送到消息队列(进而传给后台处理)
2,后台处理完毕后将结果返回响应队列,client adapter获取到结果后返回给相应客户端。
流程图如下:

我们选择用Jetty(8),redis以及php简单实现这个场景,主要用到Jetty的continuation机制和redis的list先进先出数据结构
接入服务器:
A.1,先配置一个服务器如下,同时开启一个守护线程阻塞监听response queue(用到json lib库以及jedis库)(使用专用线程处理响应可以避免“惊群”现象,不影响业务线程)
package test; import java.util.HashMap;
import java.util.List; import org.eclipse.jetty.continuation.Continuation;
import org.eclipse.jetty.server.*;
import org.eclipse.jetty.server.nio.SelectChannelConnector;
import org.eclipse.jetty.servlet.ServletContextHandler;
import org.eclipse.jetty.servlet.ServletHolder;
import org.eclipse.jetty.util.thread.QueuedThreadPool; import org.json.simple.*; import redis.clients.jedis.Jedis; public class PJetty{ public static HashMap<String,Continuation>globalMap = new HashMap<String,Continuation>(); // 用一个守护线程阻塞等待结果队列返回数据
public static class DaemonThread extends Thread{ private JSONObject obj = new JSONObject(); private Jedis jedis = new Jedis("127.0.0.1",6379);
private List<String> res; public void run(){ while(true){
// 阻塞等待响应队列
res = jedis.brpop(0, "response_queue"); // 获取结果信息
String response = res.get(1); obj=(JSONObject) JSONValue.parse(response);
String request_sid = obj.get("request_sid").toString();
String result = obj.get("results").toString(); if(request_sid == null){
continue;
} // 通过消息中的连接的sessonid获取到响应的continuation实例,然后设置结果信息再唤醒实例
Continuation con = globalMap.get(request_sid);
if(con == null){continue;}
globalMap.remove(request_sid); //客户端异常断开这里会抛错
try{
con.setAttribute("results", result);
con.resume();
} catch(Exception e){
continue;
}
} }
} public static void main(String[] args) throws Exception { //开启守护线程去阻塞等待响应结果队列,唤醒请求
DaemonThread dt = new DaemonThread();
dt.start(); //设置connectors
SelectChannelConnector connector1 = new SelectChannelConnector();
connector1.setPort(1987);
connector1.setThreadPool(new QueuedThreadPool(5)); Server server = new Server();
server.setConnectors(new Connector[]{connector1}); //使用servlet处理请求
ServletContextHandler context = new ServletContextHandler(ServletContextHandler.SESSIONS);
context.setContextPath("/");
context.addServlet(new ServletHolder(new NonBlockingServlet()), "/fetch");
server.setHandler(context); server.start();
server.join();
}
}
A.2,实现自定义的servlets接受前端client连接,将请求信息传入队列request queue
package test; import java.io.IOException; import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession; import org.eclipse.jetty.continuation.Continuation;
import org.eclipse.jetty.continuation.ContinuationSupport;
import org.json.simple.JSONObject; import redis.clients.jedis.Jedis; public class NonBlockingServlet extends HttpServlet { /**
* generated serialize number
*/
private static final long serialVersionUID = 3313258432391586994L; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException
{
// 用sleeptime来模拟后台工作量
String sleepTime = request.getParameter("st");
if(sleepTime == null){
sleepTime = "0";
} // 查看结果队列是否返回此连接的请求结果
Object results = request.getAttribute("results");
if (results==null) // 如果异步处理尚未返回结果
{
Continuation continuation = ContinuationSupport.getContinuation(request); if(continuation.isInitial()){
// 设置连接超时时间
continuation.setTimeout(10000);
response.setContentType("text/plain");
response.getWriter().flush(); HttpSession session=request.getSession();
String sid = session.getId(); Jedis jedis = new Jedis("127.0.0.1",6379);
//将请求连接sessionid以及请求内容encode后传到处理队列中
JSONObject obj=new JSONObject();
obj.put("request_sid",sid);
obj.put("params",sleepTime); jedis.lpush("request_queue", obj.toJSONString()); //将连接和continuation实例做个映射关系存到全局hashmap中,不确定这里是否应该加锁
PJetty.globalMap.put(sid, continuation);
} // 判断是否超时
if (continuation.isExpired())
{
// 返回超时Response
response.getWriter().println("timeout");
response.getWriter().flush();
return;
} // 挂起HTTP连接
continuation.suspend(); return; // or continuation.undispatch();
} // 连接恢复后返回结果
response.getWriter().println("Got Result:\t" + results);
response.getWriter().flush();
}
}
业务服务器:
B,实现后端worker.php(可以自定义worker进程数,我这里设置为5个php进程,进程数多能获取更好的并发)(用到predis库)
#!/root/bin/php
<?php require_once("lib/Predis/Autoloader.php"); function worker_thread()
{
Predis\Autoloader::register();
$redis = new Predis\Client('tcp://127.0.0.1:6379'); while(true){
try{
$request = $redis->brpop("request_queue", 0);
} catch(Exception $e){
continue;
}
/** demo
array(2) {
[0]=>
string(13) "request_queue"
[1]=>
string(55) "{"request_sid":"q0muxazo8k1h1k3uw85wuayh","params":"4"}"
}
*/
$request = json_decode($request[1], true);
// sleep represents work loads
sleep(intval($request["params"]));
$results = array("request_sid"=>$request["request_sid"], "results"=>$request["params"]);
$response = json_encode($results);
$redis->lpush("response_queue",$response);
}
} //开启多个worker进程提供服务
for ($worker_nbr = 0; $worker_nbr < 5; $worker_nbr++) {
$pid = pcntl_fork();
if ($pid == 0) {
worker_thread(); return;
}
} ?>
运行结果如下:
这只是一个简单的demo,为了防止redis,workers进程挂掉或者客户端异常断开,还需要做些异常处理,比如设置请求超时,捕获一些空指针等,超时需要将continuation从globalMap中剔除,防止内存得不到释放。
root # for((i=10;i>=1;i--)) ; do lynx -dump http://127.0.0.1:1987/fetch?st=$i & done
[] 14112
[] 14113
[] 14114
[] 14115
[] 14116
[] 14117
[] 14118
[] 14119
[] 14120
[] 14121
root # Got Result: 3
Got Result: 4
Got Result: 2
Got Result: 7
Got Result: 1
Got Result: 9
Got Result: 6
timeout
timeout
timeout [] Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
[] Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
[] Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
[] Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
[] Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
[] Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
[] Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
[] Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
[]- Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
[]+ Done lynx -dump http://127.0.0.1:1987/fetch?st=$i
redis数据库中存储的内容如下,可以看出虽然经后台处理后顺序变化了,但是对应关系正确,接入服务器能够根据request_sid把结果返回给相应的用户:
redis 127.0.0.1:6379> lrange request_queue 0 15
1) "{\"request_sid\":\"igiwkwnb715aphw8uvtfa6rj\",\"params\":\"3\"}"
2) "{\"request_sid\":\"wsrglxa3h6ef19ik5i0nbiiys\",\"params\":\"2\"}"
3) "{\"request_sid\":\"tyiqoj6awj5t16ddpqusftwc8\",\"params\":\"6\"}"
4) "{\"request_sid\":\"1052tgkiyy7c31bmxjtbom7ca\",\"params\":\"5\"}"
5) "{\"request_sid\":\"17jo1xwnnkd3h1mhcqcfplrl5k\",\"params\":\"8\"}"
6) "{\"request_sid\":\"1xk521sq6vmmf6enxauwzduj9\",\"params\":\"4\"}"
7) "{\"request_sid\":\"1cxnir1slgjiq1o2n3xwznh0kk\",\"params\":\"9\"}"
8) "{\"request_sid\":\"961vf8hao3stsv4vt1qif3ws\",\"params\":\"7\"}"
9) "{\"request_sid\":\"35pfn5au6p8qdbri17p636si\",\"params\":\"10\"}"
10) "{\"request_sid\":\"1ca4wy8qsfr7av0hwk8xtlqhp\",\"params\":\"1\"}" redis 127.0.0.1:6379> lrange response_queue 0 15
1) "{\"request_sid\":\"tyiqoj6awj5t16ddpqusftwc8\",\"results\":\"6\"}"
2) "{\"request_sid\":\"igiwkwnb715aphw8uvtfa6rj\",\"results\":\"3\"}"
3) "{\"request_sid\":\"wsrglxa3h6ef19ik5i0nbiiys\",\"results\":\"2\"}"
4) "{\"request_sid\":\"35pfn5au6p8qdbri17p636si\",\"results\":\"10\"}"
5) "{\"request_sid\":\"1052tgkiyy7c31bmxjtbom7ca\",\"results\":\"5\"}"
6) "{\"request_sid\":\"1cxnir1slgjiq1o2n3xwznh0kk\",\"results\":\"9\"}"
7) "{\"request_sid\":\"17jo1xwnnkd3h1mhcqcfplrl5k\",\"results\":\"8\"}"
8) "{\"request_sid\":\"961vf8hao3stsv4vt1qif3ws\",\"results\":\"7\"}"
9) "{\"request_sid\":\"1xk521sq6vmmf6enxauwzduj9\",\"results\":\"4\"}"
10) "{\"request_sid\":\"1ca4wy8qsfr7av0hwk8xtlqhp\",\"results\":\"1\"}"
用Jetty和redis实现接入服务器adapter的更多相关文章
- jetty 最后版本类库树, 基本上大多数应用都够了
d:\jetty-distribution-8.1.17.v20150415\lib\annotations\javax.annotation-1.1.0.v201108011116.jarjavax ...
- vue,vuex的后台管理项目架子structure-admin,后端服务nodejs
之前写过一篇vue初始化项目,构建vuex的后台管理项目架子,这个structure-admin-web所拥有的功能 接下来,针对structure-admin-web的不足,进行了补充,开发了具有登 ...
- Spring框架之websocket源码完全解析
Spring框架之websocket源码完全解析 Spring框架从4.0版开始支持WebSocket,先简单介绍WebSocket协议(详细介绍参见"WebSocket协议中文版" ...
- idou老师教你学istio30:Mixer Redis Quota Adapter 实现和机制
1. 配置 1.1参数 1.2 Params.Quota 1.3Params.Override 1.4Params.QuotaAlgorithm 速率限制的算法: Fixed Window 算法每个时 ...
- Codeigniter的Redis使用
1. ./config/redis.php: <?php $config['redis_host'] = '127.0.0.1'; $config['redis_port'] = '6379'; ...
- spring mvc redis消息队列
通常情况下,为了提高系统开发的灵活性和可维护度,我们会采用消息队列队系统进行解耦.下面是一个采用spring redis实现的消息队列实例,但此实例会由于网络延迟和阻塞等情况导致消息处理的延时,因而不 ...
- Jetty使用教程(四:28-30)—Jetty开发指南
二十八.延续机制支持 28.1 延续简介 延续是一种机制用来实现类似于Servlet 3.0异步功能的异步Servlet,但提供了一个简单易操作的接口. 28.1.1 为什么使用异步Servlets ...
- Jetty使用教程(四:21-22)—Jetty开发指南
二十一.嵌入式开发 21.1 Jetty嵌入式开发HelloWorld 本章节将提供一些教程,通过Jetty API快速开发嵌入式代码 21.1.1 下载Jetty的jar包 Jetty目前已经把所有 ...
- springmvc+spring-security+mybatis +redis +solar框架抽取
参考文章:Spring MVC 3 深入总结: 第二章 Spring MVC入门 —— 跟开涛学SpringMVC 参考博客:http://www.cnblogs.com/liukemng/categ ...
随机推荐
- Linux命令之修改主机名
ubuntu永久修改主机名 1.查看主机名 在Ubuntu系统中,快速查看主机名有多种方法: 其一,打开一个GNOME终端窗口,在命令提示符中可以看到主机名,主机名通常位于“@”符号后: 其二,在终端 ...
- C学习之指针强化
char *p = (char *)malloc(100); malloc是用于分配内存的函数,它的参数为int型,表示分配多少个字节长度,其返回类型为void*,在这里用char*就是强制转化,指定 ...
- 星际SC 地图 Big Game Fort 要塞之战 修正了 BIG GAME 地图的平衡性
星际SC 地图 Big Game Fort 要塞之战 修正了 BIG GAME 地图的平衡性 也适合BIG 1V1 对战 此版本目前不开放1打1造功能
- egret命令行编译项目时 版本不对应的问题
egret 命令行编译项目时 如使用 egret build -e 会出现版本不对应的问题 分析原因 A,B项目 A项目使用1.8的egret引擎, B项目使用2.5引擎 但本地引擎升级至2.5.5, ...
- nyist 220 推桌子
题目链接:推桌子 题目意思:给你一些操作,将S出的桌子推到L出,但是这个过道有时会被占用,推一次是10min,不影响的操作可以同时开始,并且只记一次. 思路:贪心,首先按照S从小到大排序,决策:从第一 ...
- ViewPager不能高度自适应?height=wrap_content 无效解决办法
ViewPager用的很多,主要用啦展示广告条.可是高度却不能自适应内容,总是会占满全屏,即使设置android:height="wrap_content"也是没有用的.. 解决办 ...
- CountDownLatch和CyclicBarrier区别及用法的demo
javadoc里面的描述是这样的. CountDownLatch: A synchronization aid that allows one or more threads to wait unti ...
- 10个必备的移动UI设计资源站
http://www.uisdc.com/10-necessary-mobile-ui-design-resources# 交互设计中如何增加趣味性.提升愉悦http://www.uisdc.com/ ...
- 使用python抓取知乎日报的API数据
使用 urllib2 抓取数据时,最简单的方法是: import urllib2, json def getStartImage(): stream = urllib2.urlopen('http:/ ...
- 视频媒体播放,最好的 HTML 解决方法
最好的 HTML 解决方法 HTML 5 + <object> + <embed> <video width="320" height="2 ...