成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

java.util.concurrent.Future 類基礎(chǔ)

開發(fā) 后端
在第一篇文章中,我們會討論最基本的java.util.concurrent.Future 接口。 接下來我們會跳到其他的框架,庫,甚至是語言。 Future是有限制的,但是有必要去理解future的部分。

我通過《回到Future》這個工作主題開始寫一系列的關(guān)于探討編程語言的Futrure概念的文章。Futrues是非常重要的抽象,甚至隨著異步,事件驅(qū)動,并行和擴(kuò)展系統(tǒng)的發(fā)展顯得更加重要。在第一篇文章中,我們會討論最基本的java.util.concurrent.Future 接口。 接下來我們會跳到其他的框架,庫,甚至是語言。 Future是有限制的,但是有必要去理解future的部分。

在一個單線程應(yīng)用中,當(dāng)你調(diào)用一個方法只有計算結(jié)束才會返回結(jié)果( IOUtils.toString()  comes from Apache Commons IO ):

  1. public String downloadContents(URL url) throws IOException { 
  2.     try(InputStream input = url.openStream()) { 
  3.         return IOUtils.toString(input, StandardCharsets.UTF_8); 
  4.     } 
  5.   
  6. //... 
  7.   
  8. final Future<String> contents = downloadContents(new URL("http://www.example.com")); 

downloadContents() 看上去是無害的, 但是它需要任意長的時間來完成。同時,為了減少延遲,在等待結(jié)果的期間,你可能需要同時獨立的處理其它的工作。以前你可能會啟動一個新的線程 或者等待結(jié)果(共享內(nèi)存,鎖,糟糕的 wait()/notify()對).

通過 Future<T> 模式,它會變得明朗:

  1. public static Future<String> startDownloading(URL url) { 
  2.     //... 
  3.   
  4. final Future<String> contentsFuture = startDownloading(new URL("http://www.example.com")); 
  5. //other computation 
  6. final String contents = contentsFuture.get(); 

我們馬上會實現(xiàn) startDownloading(), startDownloading()不會被阻塞,而是等待外部的站點回應(yīng),你理解這一原則是很重要的。 相反,如果它快速返回了,返回一個輕量級的Future<String> 對象。 這個對象是一個promise那么將來字符串類型就是可用的,雖然我們不知道什么時候,但是會保留這個引用直到它返有結(jié)果返回,你就可以通過Future.get()來獲取它。 換句話說,F(xiàn)uture是一個代理或者一個對象的包裝,不是真實的目標(biāo)對象。一旦異步計算完成,你就可以提取它。 那么Future提供了什么樣的接口呢?

Future.get()是最重要的方法。它阻塞和等待直到承諾的結(jié)果是可用狀態(tài), 因此如果我們確實需要這個字符串,就調(diào)用get() 方法然后等待。 還有一個接受超時參數(shù)的重載版本,如果哪里出現(xiàn)問題你就不用一直等待下去,超過設(shè)定時間就會拋出 TimeoutException。

在某些情況下,你可能想不停地偷偷看看Future是否可用了。這可以通過isDone()來完成。想象一個情景,你的用戶等待某些異步的計算,你想讓他知道這種情況, 同時去做一些其它的計算:

  1. final Future<String> contentsFuture = startDownloading(new URL("http://www.example.com")); 
  2. while (!contentsFuture.isDone()) { 
  3.     askUserToWait(); 
  4.     doSomeComputationInTheMeantime(); 
  5. contentsFuture.get(); 

最后Future.get()調(diào)用的內(nèi)容會保證馬上返回,不會被阻塞,因為Future.isDone() 返回了true。如果你遵循這個模式,就不會忙于每秒百萬次的交替等待和調(diào)用isDone()。 

取消futrues是最后一個我們還沒有覆蓋到的。想象你啟動了異步的工作并且你只能等待一些時間, 如果2秒鐘后,我們放棄,或者把錯誤傳遞出去,或者采用臨時方案解決它。然而,你是一個好市民,你應(yīng)該告訴這個future對象:我不需要你了,你別管了。 那么你可以通過停止過時的任務(wù),來節(jié)約資源。語法很簡單:

  1. contentsFuture.cancel(true);    //meh... 

我們都喜歡隱藏的,布爾類型的參數(shù),對嗎?取消可以通過兩種方式來實現(xiàn):在任務(wù)啟動前通過傳遞false參數(shù)來取消,前提是當(dāng)Future表達(dá)的結(jié)果計算開始之前。一旦Callable.call()已經(jīng)運行到一半,那么我們想讓它結(jié)束,如果我們傳遞true,那么Future.call()就會具有侵入性,試圖打斷正在運行的工作。你覺得這樣好嗎?現(xiàn)象那些拋出InterruptedException這個聲名狼藉的異常的方法,如Thread.sleep(), Object.wait(),Condition.await(),等,甚至包括Future.get(). 如果你被阻塞在這種方法并且有人決定取消你的調(diào)用,他們會毫無疑問的拋出InterruptionException,并發(fā)出有人要打斷當(dāng)前運行的任務(wù)。

因此我們現(xiàn)在明白了Future是什么--- 一個占位符,你可以在未來得到目標(biāo)對象。就像對于一輛車,還沒有制造出來的鑰匙。但是你怎樣才能在應(yīng)用程序中獲得Future的實例? 兩種最普通的資源是線程池和異步方法(線程池支持)。因此, startDownloading()方法可以被重寫為:

  1. private final ExecutorService pool = Executors.newFixedThreadPool(10); 
  2.   
  3. public Future<String> startDownloading(final URL url) throws IOException { 
  4.     return pool.submit(new Callable<String>() { 
  5.         @Override 
  6.         public String call() throws Exception { 
  7.             try (InputStream input = url.openStream()) { 
  8.                 return IOUtils.toString(input, StandardCharsets.UTF_8); 
  9.             } 
  10.         } 
  11.     }); 

雖然有大量的繁瑣的語法問題,但是基本思想是簡單的: 把需要長時間運行的計算包裝到可調(diào)用的<String>,并submit()到線程池,這個線程池包含10個線程。 提交后返回Future<String>的實現(xiàn),就像以某種方式鏈接到你的任務(wù)和線程池。明顯的你的任務(wù)不會被立即執(zhí)行,相反它被放到一個隊列中,稍后會被線程拉出來, 現(xiàn)在需要搞清楚cancel()的兩個特別的意義是什么——你可以取消在隊列中停留的任務(wù),也可以取消早已運行的任務(wù),但這是一件比較復(fù)雜的事情。

你還可以在Spring 和 EJB 碰上Future。比如Spring框架的中你可以為方法加入@Async的注解:

  1. @Async 
  2. public Future<String> startDownloading(final URL url) throws IOException { 
  3.     try (InputStream input = url.openStream()) { 
  4.         return new AsyncResult<>( 
  5.                 IOUtils.toString(input, StandardCharsets.UTF_8) 
  6.         ); 
  7.     } 

注意,我們簡單地通過包裝結(jié)果到AsyncResult來實現(xiàn)Future,但是這個方法本身不會與線程池交互或者異步處理。稍后 Spring會代理所有的調(diào)用來startDownloading()并在線程池中執(zhí)行。 在EJB中,相同的特性通過加@Asynchronousannotation 來完成。

我們學(xué)到了許多java.util.concurrent.Future的知識。 現(xiàn)在我們不得不承認(rèn),接口非常有限,特別是在和其它語言作對比的時候。稍后介紹更多。

你是否不熟悉java 7 中的 try-with-resources Future 模式?那現(xiàn)在最好是遷移到j(luò)ava7上。java 6 在未來2周后將不再維護(hù)更新。 

原文鏈接:http://www.oschina.net/translate/java-util-concurrent-future-basics

責(zé)任編輯:張偉 來源: oschina
相關(guān)推薦

2009-08-14 14:50:41

util.concur

2010-07-12 10:03:50

ibmdwjava

2011-03-15 09:10:48

Concurrent

2022-05-31 07:32:19

JDK8API工具

2024-10-31 09:30:05

線程池工具Java

2023-04-23 08:49:17

Java接口Future

2021-04-12 08:56:00

多線程Future模式

2021-01-20 08:36:15

工具AtomicRefer JDK

2021-04-05 08:11:04

Java基礎(chǔ)Calendar類DateFormat類

2009-06-18 08:51:03

Spring3.0 M

2011-11-18 10:50:25

設(shè)計模式Java線程

2020-08-18 10:20:50

Java 編程開發(fā)

2023-07-13 08:26:49

Java羅漢增強(qiáng)類

2023-10-05 11:12:06

JUCUnsafe安全

2009-07-07 10:08:49

Future Resp

2009-07-08 13:19:25

Future Resp

2023-04-09 16:31:30

Phaser工具Java

2020-11-20 09:36:43

Java對象

2021-03-22 09:56:01

Java基礎(chǔ)System類Static

2022-06-15 07:32:35

Lock線程Java
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 亚洲欧洲成人av每日更新 | 色视频网站 | 国产精品久久av | 国产精品自拍视频 | 久久久久国产精品 | 国产精品99久久久精品免费观看 | 欧美日韩不卡合集视频 | 日韩三区在线观看 | 在线观看黄色电影 | 日韩在线观看中文字幕 | 欧美一级大片免费观看 | 99在线资源| 欧美性一区二区三区 | 亚洲国产一区二区视频 | 亚洲国产成人精品女人久久久 | 日韩毛片免费看 | 成人国产一区二区三区精品麻豆 | 激情六月丁香婷婷 | 精品国产不卡一区二区三区 | 狠狠久久久 | 国内毛片毛片毛片毛片 | 视频在线一区二区 | 欧美视频一区二区三区 | 成人国产网站 | 日本不卡一区二区三区 | 精品欧美一区二区精品久久久 | 97国产精品| 亚洲精彩免费视频 | 国产精品久久久久久久久图文区 | 中文字幕第49页 | 中文字幕日韩三级 | 中文字幕乱码亚洲精品一区 | 欧美成人a| 亚洲视频第一页 | 91精品国产91久久久久青草 | 亚洲精品中文字幕av | 久久麻豆精品 | 中文字幕av一区二区三区 | 国产精品一卡 | 国产高清在线视频 | 亚洲丝袜天堂 |