关闭

Java 并发专题 :FutureTask 实现预加载数据 在线看电子书、浏览器浏览网页等

标签: Java并发多线程FutureTask
7971人阅读 评论(6) 收藏 举报
分类:

继续并发专题~

FutureTask 有点类似Runnable,都可以通过Thread来启动,不过FutureTask可以返回执行完毕的数据,并且FutureTask的get方法支持阻塞。

由于:FutureTask可以返回执行完毕的数据,并且FutureTask的get方法支持阻塞这两个特性,我们可以用来预先加载一些可能用到资源,然后要用的时候,调用get方法获取(如果资源加载完,直接返回;否则继续等待其加载完成)。

下面通过两个例子来介绍下:

1、使用FutureTask来预加载稍后要用的的数据。

package com.zhy.concurrency.futuretask;

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

/**
 * 使用FutureTask来提前加载稍后要用到的数据
 * 
 * @author zhy
 * 
 */
public class PreLoaderUseFutureTask
{
	/**
	 * 创建一个FutureTask用来加载资源
	 */
	private final FutureTask<String> futureTask = new FutureTask<String>(
			new Callable<String>()
			{
				@Override
				public String call() throws Exception
				{
					Thread.sleep(3000);
					return "加载资源需要3秒";
				}
			});

	public final Thread thread = new Thread(futureTask);

	public void start()
	{
		thread.start();
	}

	/**
	 * 获取资源
	 * 
	 * @return
	 * @throws ExecutionException 
	 * @throws InterruptedException 
	 */
	public String getRes() throws InterruptedException, ExecutionException
	{
		return futureTask.get();//加载完毕直接返回,否则等待加载完毕

	}

	public static void main(String[] args) throws InterruptedException, ExecutionException
	{

		PreLoaderUseFutureTask task = new PreLoaderUseFutureTask();
		/**
		 * 开启预加载资源
		 */
		task.start();
		// 用户在真正需要加载资源前进行了其他操作了2秒
		Thread.sleep(2000);

		/**
		 * 获取资源
		 */
		System.out.println(System.currentTimeMillis() + ":开始加载资源");
		String res = task.getRes();
		System.out.println(res);
		System.out.println(System.currentTimeMillis() + ":加载资源结束");
	}

}

运行结果:

1400902789275:开始加载资源
加载资源需要3秒
1400902790275:加载资源结束
可以看到,本来加载资源的时间需要3秒,现在只花费了1秒,如果用户其他操作时间更长,则可直接返回,极大增加了用户体验。

2、看下Future的API


可以看到Future的API,还是比简单的,见名知意的感觉,get( long , TimeUnit )还能支持,设置最大等待时间,比如某个操作耗时太长,就可以取消了。

3、FutureTask模拟,用户在线观看电子书的预加载功能

用户观看当前页时,后台预先把下一页加载好,这样可以大幅度提高用户的体验,不需要每一页都等待加载,用户会觉得此电子书软件很流畅,哈哈,用户觉得好,才是真的好。

package com.zhy.concurrency.futuretask;

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;


/**
 * 使用FutureTask模拟预加载下一页图书的内容
 * 
 * @author zhy
 * 
 */
public class BookInstance
{

	/**
	 * 当前的页码
	 */
	private volatile int currentPage = 1;

	/**
	 * 异步的任务获取当前页的内容
	 */
	FutureTask<String> futureTask = new FutureTask<String>(
			new Callable<String>()
			{
				@Override
				public String call() throws Exception
				{
					return loadDataFromNet();
				}
			});

	/**
	 * 实例化一本书,并传入当前读到的页码
	 * 
	 * @param currentPage
	 */
	public BookInstance(int currentPage)
	{
		this.currentPage = currentPage;
		/**
		 * 直接启动线程获取当前页码内容
		 */
		Thread thread = new Thread(futureTask);
		thread.start();
	}

	/**
	 * 获取当前页的内容
	 * 
	 * @return
	 * @throws InterruptedException
	 * @throws ExecutionException
	 */
	public String getCurrentPageContent() throws InterruptedException,
			ExecutionException
	{
		String con = futureTask.get();
		this.currentPage = currentPage + 1;
		Thread thread = new Thread(futureTask = new FutureTask<String>(
				new Callable<String>()
				{
					@Override
					public String call() throws Exception
					{
						return loadDataFromNet();
					}
				}));
		thread.start();
		return con;
	}

	/**
	 * 根据页码从网络抓取数据
	 * 
	 * @return
	 * @throws InterruptedException
	 */
	private String loadDataFromNet() throws InterruptedException
	{
		Thread.sleep(1000);
		return "Page " + this.currentPage + " : the content ....";

	}

	public static void main(String[] args) throws InterruptedException,
			ExecutionException
	{
		BookInstance instance = new BookInstance(1);
		for (int i = 0; i < 10; i++)
		{
			long start = System.currentTimeMillis();
			String content = instance.getCurrentPageContent();
			System.out.println("[1秒阅读时间]read:" + content);
			Thread.sleep(1000);
			System.out.println(System.currentTimeMillis() - start);
		}

	}
}

输出结果:

[1秒阅读时间]read:Page 1 : the content ....
2001
[1秒阅读时间]read:Page 2 : the content ....
1000
[1秒阅读时间]read:Page 3 : the content ....
1001
[1秒阅读时间]read:Page 4 : the content ....
1000
[1秒阅读时间]read:Page 5 : the content ....
1001

可以看到,除了第一次观看当前页需要等待网络加载数据的过程(输出的:2001,1000是加载耗时,1000是用户阅读时间),接下来的页面都是瞬间返回(输出的1000是用户阅读时间),完全不需要等待。

代码都是为了讲解FutureTask的应用场景,,,请勿直接在项目中使用。


好了,就到这里,欢迎各位留言。




15
0
查看评论
发表评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场

Web:预加载与延时加载

最近做了个某珠宝的微信活动页面。由于每个活动页面都有一张背景图,在移动端网速不佳的时候,每翻一页,都会有较长时间的空白页出现。这种体验当然是不能够接受的。为了避免这种情况,使用图片预加载,当用户进入首...
  • u010703975
  • u010703975
  • 2015-12-20 21:29
  • 2223

Java 并发专题 :FutureTask 实现预加载数据 在线看电子书、浏览器浏览网页等

继续并发专题~ FutureTask 有点类似Runnable,都可以通过Thread来启动,不过FutureTask可以返回执行完毕的数据,并且FutureTask的get方法支持阻塞。 由于:Fu...
  • lmj623565791
  • lmj623565791
  • 2014-05-24 11:51
  • 7971

Servlet流操作——在浏览器中打开或者下载PDF文件

在Servlet与客户的请求应答的过程中,底层是通过输入/输出流来实现的。Servlet支持两种格式的输入/输出流。一个是字符输入/输出流。ServletResponse的getWriter()方法返...
  • yanglun1
  • yanglun1
  • 2015-06-14 13:12
  • 5979

使用FlexPaper实现PDF文件在线预览

PDF在线预览,使用OpenOffice、JODConverter、FlexPaper、swfTools实现PDF文件在线预览。 PPT转换为pdf、Word转换为pdf、Excel转换为pdf、Te...
  • xinxin19881112
  • xinxin19881112
  • 2015-09-23 16:31
  • 14057

Java 并发专题 :FutureTask 实现预加载数据 在线看电子书、浏览器浏览网页等

版权声明:本文为博主原创文章,未经博主允许不得转载。 继续并发专题~ FutureTask 有点类似Runnable,都可以通过Thread来启动,不过FutureTask可以返回...
  • rabbit_in_android
  • rabbit_in_android
  • 2015-12-18 11:51
  • 284

Java 并发专题 :FutureTask 实现预加载数据 在线看电子书、浏览器浏览网页等

Java 并发专题 :FutureTask 实现预加载数据 在线看电子书、浏览器浏览网页等
  • u013651026
  • u013651026
  • 2017-08-02 22:44
  • 145

Java 并发专题 :FutureTask 实现预加载数据 在线看电子书、浏览器浏览网页等

转自:http://blog.csdn.net/lmj623565791/article/details/26817403  继续并发专题~ FutureTask 有点类似Ru...
  • linhaosheng123456
  • linhaosheng123456
  • 2015-12-05 12:54
  • 378

Java 并发专题 :FutureTask 实现预加载数据 在线看电子书、浏览器浏览网页等

继续并发专题~FutureTask 有点类似Runnable,都可以通过Thread来启动,不过FutureTask可以返回执行完毕的数据,并且FutureTask的get方法支持阻塞。由于:Futu...
  • m0_37651226
  • m0_37651226
  • 2017-06-22 09:03
  • 348

Java 并发专题 :FutureTask 实现预加载数据 在线看电子书、浏览器浏览网页等

继续并发专题~FutureTask 有点类似Runnable,都可以通过Thread来启动,不过FutureTask可以返回执行完毕的数据,并且FutureTask的get方法支持阻塞。由于:Futu...
  • liuxinyang666
  • liuxinyang666
  • 2017-12-15 15:29
  • 15

FutureTask 实现预加载数据 在线看电子书、浏览器浏览网页等

FutureTask 有点类似Runnable,都可以通过Thread来启动,不过FutureTask可以返回执行完毕的数据,并且FutureTask的get方法支持阻塞。 由于:FutureT...
  • michael1112
  • michael1112
  • 2017-11-29 14:40
  • 21
    个人资料
    • 访问:17922293次
    • 积分:51373
    • 等级:
    • 排名:第66名
    • 原创:205篇
    • 转载:0篇
    • 译文:6篇
    • 评论:15630条
    我的微信公众号
    联系方式


    QQ群:
    • 55032675
    • 423372824
    • 497438697
    • 请勿重复加群,Thx
    博客专栏
    统计