Future 示例
public static Object send(RequestClient request)
future.channel().writeAndFlush(JSONObject.toJSONString(request));
future.channel().writeAndFlush("\r\n");
DefaultFuture defaultFuture = new DefaultFuture(request);//请求未来的响应
Response response = defaultFuture.get(10);//阻塞获取响应
return response;
return null;
}
public class ClientHandler extends ChannelInboundHandlerAdapter {
@Override
public void channelRead(ChannelHandlerContext ctx, Object msg)
Response res = JSONObject.parseObject(msg.toString(), Response.class);
DefaultFuture.recive(res);//拿到结果,通知阻塞解除
}
}
public class DefaultFuture { private long id;
public final static Map<Long,DefaultFuture> FUTURES= new ConcurrentHashMap<Long,DefaultFuture>();
private long timeout;
private final long start=System.currentTimeMillis(); //get方法和recive方法是不同线程调用的同一个对象,要volatile。
private volatile Response response;
private volatile Lock lock = new ReentrantLock();
private volatile Condition condition = lock.newCondition(); public DefaultFuture(){} public DefaultFuture(RequestClient request){
id=request.getId();//通过id进行异步判断,
FUTURES.put(id, this);//请求的id和响应对应。
} public Response get(){
lock.lock();
while(!hasDone()){
try {
condition.await();
} catch (InterruptedException e) {
e.printStackTrace();
}finally{
lock.unlock();
}
}
return response;
} //超时是防止服务器卡死了。
public Response get(long timeout){
long start = System.currentTimeMillis();
lock.lock();
while(!hasDone()){
try {//condition是依赖ReentrantLock
//此时当前线程释放lock锁,进入[等待状态],等待其他线程执行aCondition.signal()时才有可能执行
condition.await(timeout, TimeUnit.SECONDS);
if(System.currentTimeMillis()-start>=timeout){
break;
}
} catch (InterruptedException e) {
e.printStackTrace();
}finally{
lock.unlock();
}
}
return response;
} //收到服务器响应
public static void recive(Response res){
//找到res相对应的DefaultFuture
DefaultFuture future = FUTURES.remove(res.getId());
if(future==null){
return ;
}
Lock lock= future.getLock();
lock.lock();
try{
future.setResponse(res);
Condition condition = future.getCondition();
if(condition!=null){
condition.signal();
} }catch(Exception e){
e.printStackTrace();
}finally{
lock.unlock();
}
} //开一条线程处理超时
static class FutureTimeOutThread extends Thread{
@Override
public void run() {
while(true){
for(long futureId : FUTURES.keySet()){
DefaultFuture f = FUTURES.get(futureId);
if(f==null){//等于空直接移除
FUTURES.remove(futureId);
continue;
}
if(f.getTimeout()>0){//不等于空判断是否超时
if((System.currentTimeMillis()-f.getStart())>f.getTimeout()){
Response res = new Response();
res.setContent(null);
res.setMsg("请求超时!");
res.setStatus(1);//响应异常处理
res.setId(f.getId());
DefaultFuture.recive(res);
}
}
}
}
}
} static{
FutureTimeOutThread timeOutThread = new FutureTimeOutThread();
timeOutThread.setDaemon(true);//守护线程,主线程在就在,主线程挂掉就挂掉,
timeOutThread.start();
} private boolean hasDone() {return response !=null? true:false;}
public long getId() {return id;}
public Response getResponse() {return response;}
public void setResponse(Response response) {this.response = response;}
public Lock getLock() {return lock;}
public void setLock(Lock lock) {this.lock = lock;}
public Condition getCondition() {return condition;}
public void setCondition(Condition condition) {this.condition = condition;}
public void setId(long id) {this.id = id;}
public long getTimeout() {return timeout;}
public void setTimeout(long timeout) {this.timeout = timeout;}
public long getStart() {return start;}
}
Future 示例的更多相关文章
- java多线程获取返回结果--Callable和Future示例
package test.guyezhai.thread; import java.util.ArrayList; import java.util.Date; import java.util.Li ...
- java并发:获取线程执行结果(Callable、Future、FutureTask)
初识Callable and Future 在编码时,我们可以通过继承Thread或是实现Runnable接口来创建线程,但是这两种方式都存在一个缺陷:在执行完任务之后无法获取执行结果.如果需要获取执 ...
- Akka系列(五):Java和Scala中的Future
前言....... 随着CPU的核数的增加,异步编程模型在并发领域中的得到了越来越多的应用,由于Scala是一门函数式语言,天然的支持异步编程模型,今天主要来看一下Java和Scala中的Futrue ...
- HowToDoInJava Java 教程·翻译完成
原文:HowToDoInJava 协议:CC BY-NC-SA 4.0 欢迎任何人参与和完善:一个人可以走的很快,但是一群人却可以走的更远. ApacheCN 学习资源 目录 核心 Java 教程 什 ...
- 一文带你了解J.U.C的FutureTask、Fork/Join框架和BlockingQueue
摘要: J.U.C是Java并发编程中非常重要的工具包,今天,我们就来着重讲讲J.U.C里面的FutureTask.Fork/Join框架和BlockingQueue. 本文分享自华为云社区<[ ...
- 混合使用ForkJoin+Actor+Future实现一千万个不重复整数的排序(Scala示例)
目标 实现一千万个不重复整数的排序,可以一次性加载到 2G 的内存里. 本文适合于想要了解新语言 Scala 并发异步编程框架 Akka, Future 的筒鞋. 读完本文后,将了解如何综 ...
- Java中的Runnable、Callable、Future、FutureTask的区别与示例
Java中存在Runnable.Callable.Future.FutureTask这几个与线程相关的类或者接口,在Java中也是比较重要的几个概念,我们通过下面的简单示例来了解一下它们的作用于区别. ...
- 【腾讯Bugly干货分享】WebVR如此近-three.js的WebVR示例解析
本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/57c7ff1689a6c9121b1adb16 作者:苏晏烨 关于WebVR 最 ...
- java Future 接口介绍
(转自:http://blog.csdn.net/yangyan19870319/article/details/6093481) 在Java中,如果需要设定代码执行的最长时间,即超时,可以用Java ...
随机推荐
- FatFs文件系统的移植
FatFs 的底层可以写一次命令,读写多个扇区.FatFs的设计的读写的思想就很好,小块的数据,我就经过Buffer来存储,大块的数据,我就直接进行存取,那样速度,效率高了很多,看图: FatFs文件 ...
- 关于ping地址的几个意义
1.ping 127.0.0.1 判断电脑的tcp/ip协议栈有没有问题 2.ping 本机ip 判断电脑的网卡驱动有没有问题 3.ping 网关 判断数据是否可以到达路由 4.ping www.ba ...
- Twig---for循环
如何使用twig做for循环. Twig中文文档: https://www.kancloud.cn/yunye/twig-cn/159620 {% for item in list %} <li ...
- NW.js 桌面应用程序
nw.js官网 https://nwjs.io/downloads/ 中文网:https://nwjs.org.cn/ 参考文档 https://www.cnblogs.com/xuanhun/ ...
- 怎么才能使服务器Nginx(或者Apache)支持字体文件
为了在前端正确地显示字体,浏览器必须使用正确的http header来接受字体文件.如果服务器没有设置要求的头信息,那么有些浏览器就会在控制台报错或者直接不能显示. 可能你的服务器已经配置好了,你无须 ...
- php 判断手机号 和 手机号中间四位以**** 代替
代码: //自定义函数手机号隐藏中间四位 function get_phone($str){ $str=$str; $resstr=substr_replace($str,'****',3,4); r ...
- Python学习之旅(十)
Python基础知识(9):函数(Ⅰ) Python中函数的定义:是逻辑结构和过程化的一种编程方法 定义方法: def test(x): #def:定义函数的关键字 test:函数名 x:形参,也可以 ...
- python学习之旅(五)
Python基础知识(4):基础数据类型之字符串(Ⅰ) 字符串是 Python 中最常用的数据类型.可以使用引号“ ”来创建字符串,只要为变量分配一个值即可.例如: name=“Alice” 注:字符 ...
- LeetCode 12 - 整数转罗马数字 - [简单模拟]
题目链接:https://leetcode-cn.com/problems/integer-to-roman/ 题解: 把 $1,4,5,9,10,40,50, \cdots, 900, 1000$ ...
- arcpy加载mxd文件时,无效的MXD路径,提示assert (os.path.isfile(mxd) or (mxd.lower() == "current")), gp.getIDMessage(89004, "Invalid MXD filename")
无效的MXD路径,将路径前加‘u’,改为这种: mxdPath = u"C:\\1331\\DB\\Original Files\\dd.mxd" 参考: https://gis. ...