多线程系列之十:Future模式
一,Future模式
假设有一个方法需要花费很长的时间才能获取运行结果。那么,与其一直等待结果,不如先拿一张 提货单。获取提货单并不耗费时间。这里提货单就称为Future角色
获取Future角色的线程会在稍后使用Future角色来获取运行结果
在处理开始时先返回Future角色,等到其他线程出来终止后,在将结果设置到Future角色中。
二,示例程序
Data:访问数据的接口
RealData:实际处理数据的类
FutureData:表示RealData的提货单,
Host:先拿提货单future实例,开启新线程创建RealData实例
public interface Data {
public abstract String getContent();
}
public class RealData implements Data {
private final String content;
public RealData(int count,char c){
System.out.println(" making RealData("+count+" , "+ c +") begin");
char[] buffer = new char[count];
for (int i = 0; i <count ; i++) {
buffer[i] = c ;
try {
Thread.sleep(100);
}catch (InterruptedException e){
}
}
System.out.println(" making RealData("+count+" , "+ c +") end");
this.content = new String(buffer);
}
@Override
public String getContent() {
return content;
}
@Override
public String toString() {
return "real.toString";
}
}
public class FutureData implements Data {
private RealData realData = null;
private boolean ready = false;
/**
* 为Future类中的字段设置值
* @param realData
*/
public synchronized void setRealData(RealData realData) {
if(ready){
return;
}
this.realData = realData;
this.ready = true;
notifyAll();
}
/**
* 还没有成功设置值之前 一直等待。设置成功后就返回值
* @return
*/
@Override
public synchronized String getContent() {
while (!ready){
try {
wait();
}catch (InterruptedException e){
}
}
return realData.getContent();
}
@Override
public String toString() {
return "feature.toString";
}
}
public class Host {
public Data request(final int count, final char c){
System.out.println(" request("+count+" , "+c+") begin");
//1,创建FutureData实例
final FutureData futureData = new FutureData();
//2,启动一个新线程去创建RealData实例,耗时操作
new Thread(){
@Override
public void run() {
RealData realData = new RealData(count,c);
futureData.setRealData(realData);
}
}.start();
System.out.println(" request("+count+" , "+c+") end");
//3,将Future实例作为返回值返回给调用者
return futureData;
}
}
public class Test {
public static void main(String[] args) {
System.out.println(" main begin ..........");
Host host = new Host();
Data data1 = host.request(10,'a');
System.out.println("虚拟数据:data1 = "+data1.toString());
Data data2 = host.request(20,'b');
System.out.println("虚拟数据:data2 = "+data2.toString());
Data data3 = host.request(30,'c');
System.out.println("虚拟数据:data3 = "+data3.toString());
System.out.println(" main OtherJobs begin");
try {
//模拟主线程处理其他业务
Thread.sleep(2000);
}catch (InterruptedException e){
}
System.out.println(" main OtherJobs end");
System.out.println("真实数据:data1 = "+data1.getContent());
System.out.println("真实数据:data2 = "+data2.getContent());
System.out.println("真实数据:data3 = "+data3.getContent());
System.out.println(" main end ..........");
}
}
三,特点
1,提高吞吐量
首先这种模式可以提高程序响应性,但是耗时的操作花费的时间并没有减少啊,当程序在进行磁盘读写时,cpu处于等待状态,这时可以把cpu分配给其他的线程,就可以提高吞吐量了
四,Callable和Future
public class CallableAndFuture {
public static void main(String[] args) {
ExecutorService threadPool= Executors.newSingleThreadExecutor();
/*
threadPool.submit() 返回有结果的
*/
Future<String> future= //Future是用来接收submit的结果的。泛型和Callable的结果一样
threadPool.submit(new Callable<String>() {//这里泛型了,
@Override
public String call() throws Exception {
Thread.sleep(1000);
System.out.println("处理结果中。。。。");
Thread.sleep(1000);
return "hello";
}
});
System.out.println("等待结果");
try {
System.out.println("拿到结果"+future.get());
} catch (InterruptedException e) {
e.printStackTrace();
} catch (ExecutionException e) {
e.printStackTrace();
}
}
}
/**
*
* CompletionService用于提交一组Callable任务,其task方法返回已完成的一个Callable任务对应的Future对象
*/
public class CallableAndFuture2 {
public static void main(String[] args) {
ExecutorService threadPool2= Executors.newFixedThreadPool(10);
CompletionService<Integer> completionService = new ExecutorCompletionService(threadPool2);
for (int i = 0; i < 10; i++) {
final int seq = i;
completionService.submit(new Callable() {
@Override
public Integer call() throws Exception {
Thread.sleep(new Random().nextInt(5000));
return seq;
}
});
}
while (true){
try {
System.out.println(completionService.take().get());
} catch (InterruptedException e) {
e.printStackTrace();
} catch (ExecutionException e) {
e.printStackTrace();
}
}
}
}
多线程系列之十:Future模式的更多相关文章
- java多线程系列13 设计模式 Future 模式
Future 模式 类似于ajax请求 页面异步的进行后台请求 用户无需等待请求的结果 就可以继续浏览或者操作 核心就是:去除了主函数的等待时间,并使得原本需要等待的时间段可以用于处理其他业务逻辑 ...
- 多线程(10) — Future模式
Future模式是多线程开发中常用常见的一种设计模式,它的核心思想是异步调用.在调用一个函数方法时候,如果函数执行很慢,我们就要进行等待,但这时我们可能不着急要结果,因此我们可以让被调者立即返回,让它 ...
- 多线程系列之三:Immutable 模式
一,什么是Immutable模式?immutable就是不变的,不发生改变的.Immutable模式中存在着确保实例状态不发生变化改变的类.这些实例不需要互斥处理.String就是一个Immutabl ...
- 多线程系列之八:Thread-Per-Message模式
一,Thread-Per-Message模式 翻译过来就是 每个消息一个线程.message可以理解为命令,请求.为每一个请求新分配一个线程,由这个线程来执行处理.Thread-Per-Message ...
- 多线程系列之六:Producer-Consumer模式
一,Producer-Consumer模式 Producer:生产者的意思,指的是生成数据的线程.Consumer:消费者的意思,指的是使用数据的线程当生产者和消费者以不同的线程运行时,两者之间的处理 ...
- 多线程系列之五:Balking 模式
一,什么是Balking模式 如果现在不合适执行这个操作,或者没必要执行这个操作,就停止处理,直接返回.在Balking模式中,如果守护条件不成立,就立即中断处理. 二,例子: 定期将当前数据内容写入 ...
- java多线程系列 目录
Java多线程系列1 线程创建以及状态切换 Java多线程系列2 线程常见方法介绍 Java多线程系列3 synchronized 关键词 Java多线程系列4 线程交互(wait和 ...
- Java多线程编程中Future模式的详解
Java多线程编程中,常用的多线程设计模式包括:Future模式.Master-Worker模式.Guarded Suspeionsion模式.不变模式和生产者-消费者模式等.这篇文章主要讲述Futu ...
- Java多线程编程中Future模式的详解<转>
Java多线程编程中,常用的多线程设计模式包括:Future模式.Master-Worker模式.Guarded Suspeionsion模式.不变模式和生产者-消费者模式等.这篇文章主要讲述Futu ...
随机推荐
- human pose estimation
2D Pose estimation主要面临的困难:遮挡.复杂背景.光照.真实世界的复杂姿态.人的尺度不一.拍摄角度不固定等. 单人姿态估计 传统方法:基于Pictorial Structures, ...
- linux 软连接创建 压缩解压缩 linux的dns服务相关
linux软连接创建 注意用绝对路径,语法如下 ln -s 目标文件绝对路径 软连接名字绝对路径 ln -s /小护士.txt /tmp/hs.txt 修改linux的PS1变量,命令提示符变量 PS ...
- linux快速搭建
------------------------------------------ 转载内容 --------------------- Linux升级命令有两个分别是yum upgrade和yum ...
- re库
一.Re库的主要功能: 函数 功能 re.search() 在一个字符串中搜索匹配正则表达式的第一个位置,返回match对象 re.match() 在一个字符串的开始位置匹配正则表达式,返回match ...
- 安装 Kivy
Kivy 是基于Python编程语言的一种用于开发跨平台GUI客户端程序的框架.可以打包成iOS.Android.Windows.OSX等多种平台.目前已经实现一次编码,到处运行的目的. Kivy框架 ...
- [SHOI2015]脑洞治疗仪
嘟嘟嘟 这题其实就是一个线段树维护最大连续和的水题. 别的操作不说,操作1只要二分找区间前\(k\)个0即可. 需要注意的是,因为操作1两区间可能有交,因此要先清空再二分查询-- 复杂度\(O(n l ...
- ubuntu1604配置ss代理
1 安装ss 参考http://www.cnblogs.com/mdzz/p/10140066.html sudo apt install python3-pip sudo pip3 install ...
- RS-485总线通信协议
https://blog.csdn.net/ouyangxin95/article/details/78174145 RS-485总线技术只是规定了接口的电气标准,并没有规定RS-485接口的电缆,插 ...
- (0)HomeAssistant 教程
国外:https://www.home-assistant.io/components/light.mqtt/ 中国:https://www.hachina.io/docs/890.html
- 【Flask】abort和errorhandler、app_errorhandler进行请求中断及自定义异常处理
在view函数中,如果需要中断request,可以使用abort(500)或者直接raise exception.当然我们还需要返回一个出错信息给前端,所以需要定制一下ErrorHandler.一般只 ...