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

本文主要是介绍Java 并发专题 FutureTask 实现预加载数据 在线看电子书 浏览器浏览网页等,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

分享一下我老师大神的人工智能教程。零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到我们人工智能的队伍中来!https://blog.csdn.net/jiangjunshow

继续并发专题~

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:开始加载资源加载资源需要31400902790275:加载资源结束
可以看到,本来加载资源的时间需要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的应用场景,,,请勿直接在项目中使用。


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




           

分享一下我老师大神的人工智能教程。零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到我们人工智能的队伍中来!https://blog.csdn.net/jiangjunshow

这篇关于Java 并发专题 FutureTask 实现预加载数据 在线看电子书 浏览器浏览网页等的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/365430

相关文章

网页解析 lxml 库--实战

lxml库使用流程 lxml 是 Python 的第三方解析库,完全使用 Python 语言编写,它对 XPath表达式提供了良好的支 持,因此能够了高效地解析 HTML/XML 文档。本节讲解如何通过 lxml 库解析 HTML 文档。 pip install lxml lxm| 库提供了一个 etree 模块,该模块专门用来解析 HTML/XML 文档,下面来介绍一下 lxml 库

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

基于MySQL Binlog的Elasticsearch数据同步实践

一、为什么要做 随着马蜂窝的逐渐发展,我们的业务数据越来越多,单纯使用 MySQL 已经不能满足我们的数据查询需求,例如对于商品、订单等数据的多维度检索。 使用 Elasticsearch 存储业务数据可以很好的解决我们业务中的搜索需求。而数据进行异构存储后,随之而来的就是数据同步的问题。 二、现有方法及问题 对于数据同步,我们目前的解决方案是建立数据中间表。把需要检索的业务数据,统一放到一张M