传统的服务器端为若干个客户端提供服务,一般需要开启多个服务器端进程。为了进一步提升服务器端的处理能力,可以如下图所示将服务解耦为两部分(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的更多相关文章

  1. jetty 最后版本类库树, 基本上大多数应用都够了

    d:\jetty-distribution-8.1.17.v20150415\lib\annotations\javax.annotation-1.1.0.v201108011116.jarjavax ...

  2. vue,vuex的后台管理项目架子structure-admin,后端服务nodejs

    之前写过一篇vue初始化项目,构建vuex的后台管理项目架子,这个structure-admin-web所拥有的功能 接下来,针对structure-admin-web的不足,进行了补充,开发了具有登 ...

  3. Spring框架之websocket源码完全解析

    Spring框架之websocket源码完全解析 Spring框架从4.0版开始支持WebSocket,先简单介绍WebSocket协议(详细介绍参见"WebSocket协议中文版" ...

  4. idou老师教你学istio30:Mixer Redis Quota Adapter 实现和机制

    1. 配置 1.1参数 1.2 Params.Quota 1.3Params.Override 1.4Params.QuotaAlgorithm 速率限制的算法: Fixed Window 算法每个时 ...

  5. Codeigniter的Redis使用

    1. ./config/redis.php: <?php $config['redis_host'] = '127.0.0.1'; $config['redis_port'] = '6379'; ...

  6. spring mvc redis消息队列

    通常情况下,为了提高系统开发的灵活性和可维护度,我们会采用消息队列队系统进行解耦.下面是一个采用spring redis实现的消息队列实例,但此实例会由于网络延迟和阻塞等情况导致消息处理的延时,因而不 ...

  7. Jetty使用教程(四:28-30)—Jetty开发指南

    二十八.延续机制支持 28.1 延续简介 延续是一种机制用来实现类似于Servlet 3.0异步功能的异步Servlet,但提供了一个简单易操作的接口. 28.1.1 为什么使用异步Servlets ...

  8. Jetty使用教程(四:21-22)—Jetty开发指南

    二十一.嵌入式开发 21.1 Jetty嵌入式开发HelloWorld 本章节将提供一些教程,通过Jetty API快速开发嵌入式代码 21.1.1 下载Jetty的jar包 Jetty目前已经把所有 ...

  9. springmvc+spring-security+mybatis +redis +solar框架抽取

    参考文章:Spring MVC 3 深入总结: 第二章 Spring MVC入门 —— 跟开涛学SpringMVC 参考博客:http://www.cnblogs.com/liukemng/categ ...

随机推荐

  1. Linux命令之修改主机名

    ubuntu永久修改主机名 1.查看主机名 在Ubuntu系统中,快速查看主机名有多种方法: 其一,打开一个GNOME终端窗口,在命令提示符中可以看到主机名,主机名通常位于“@”符号后: 其二,在终端 ...

  2. C学习之指针强化

    char *p = (char *)malloc(100); malloc是用于分配内存的函数,它的参数为int型,表示分配多少个字节长度,其返回类型为void*,在这里用char*就是强制转化,指定 ...

  3. 星际SC 地图 Big Game Fort 要塞之战 修正了 BIG GAME 地图的平衡性

    星际SC 地图 Big Game Fort 要塞之战 修正了 BIG GAME 地图的平衡性 也适合BIG 1V1 对战 此版本目前不开放1打1造功能

  4. egret命令行编译项目时 版本不对应的问题

    egret 命令行编译项目时 如使用 egret build -e 会出现版本不对应的问题 分析原因 A,B项目 A项目使用1.8的egret引擎, B项目使用2.5引擎 但本地引擎升级至2.5.5, ...

  5. nyist 220 推桌子

    题目链接:推桌子 题目意思:给你一些操作,将S出的桌子推到L出,但是这个过道有时会被占用,推一次是10min,不影响的操作可以同时开始,并且只记一次. 思路:贪心,首先按照S从小到大排序,决策:从第一 ...

  6. ViewPager不能高度自适应?height=wrap_content 无效解决办法

    ViewPager用的很多,主要用啦展示广告条.可是高度却不能自适应内容,总是会占满全屏,即使设置android:height="wrap_content"也是没有用的.. 解决办 ...

  7. CountDownLatch和CyclicBarrier区别及用法的demo

    javadoc里面的描述是这样的. CountDownLatch: A synchronization aid that allows one or more threads to wait unti ...

  8. 10个必备的移动UI设计资源站

    http://www.uisdc.com/10-necessary-mobile-ui-design-resources# 交互设计中如何增加趣味性.提升愉悦http://www.uisdc.com/ ...

  9. 使用python抓取知乎日报的API数据

    使用 urllib2 抓取数据时,最简单的方法是: import urllib2, json def getStartImage(): stream = urllib2.urlopen('http:/ ...

  10. 视频媒体播放,最好的 HTML 解决方法

    最好的 HTML 解决方法 HTML 5 + <object> + <embed> <video width="320" height="2 ...