多线程系列之十: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 ...
随机推荐
- Node中的模块引入机制
1.如果模块在当前目录下,可以通过下面语句将模块引入进来,注意需要使用 "./"表示当前路径 const currency = require('./currency'); ←-- ...
- Win10改AHCI无需重装系统(无需改注册表)的方法
下面就开始:1.开机后按下WIN键 加 R键2.输入 msconfig3.如图中所示进行点击.1 引导界面 2安全引导打钩 .最小打钩 3 下面的确定.4.点击重新启动5.在重启时连续按 F2 进入B ...
- 关于激活Windows10专业版2018长期服务版
之前重装了一次系统,偷懒用了小白一键重装,装好之后显示的是Windows10专业版2018长期服务版,当时也没想太多就放着用了. 然后 ,这几天一直提示 “你的windows许可证即将过期” ,就按 ...
- php函数long2ip与ip2long()
long2ip - Converts an long integer address into a string in (IPv4) Internet standard dotted format s ...
- February 13th, 2018 Week 7th Tuesday
You are your greatest asset. 你就是你自己最大的资本. For most of us, there are few things that we can count on ...
- 洛谷P1904
法一,数字太大,可能通过不了 #include <iostream>#include <algorithm>#include <cstdio>using nam ...
- 转://如何创建ASM磁盘
1 前言 无论是安装单机版的asm还是rac都离不开ASM磁盘组的创建,创建ASM磁盘组的关键是创建好需要的asm磁盘,发现很多网友安装grid软件和grid实例,都在磁盘的创建这里有很大的问题,本 ...
- OmniPlan 3 Pro密钥
密钥用户名都是youliyuan.OmniPlan 3:HOMJ-QOJH-OIBN-TNIH-HWUN-TEEH-WUNNKWO-HVKB-JAZE-UIHH-XAVY-BEEX-AVYBCRW-M ...
- 启动项目显示:非法字符:'\ufeff' 和需要 class ,interface 或者 enum 错误
原来是因为 Windows 记事本在修改 UTF-8 文件时自作聪明地在文件开头添加 BOM 导致的,所以才会导致 IDEA 不能正确读取 .java 文件从而程序出错. 解决: 找到 xxx. ja ...
- Java多线程(二)——常用的实现多线程的两种方式
一.继承Thread类创建线程类 Java使用Thread类代表线程,所有的线程对象都必须是Thread类或其子类的实例.每个线程的作用是完成一定的任务,实际上就是执行一段程序流即一段顺序执行的代码. ...