scala-Future和Promise

来源:http://www.cnblogs.com/hangscer/archive/2017/12/24/8097901.html
-Advertisement-
Play Games

首先說明同步與非同步,阻塞與非阻塞的問題: Asynchronous vs. Synchronous A method call is considered synchronous if the caller cannot make progress until the method returns ...


首先說明同步與非同步,阻塞與非阻塞的問題:
Asynchronous vs. Synchronous

A method call is considered synchronous if the caller cannot make progress until the method returns a value or throws an exception. On the other hand, an asynchronous call allows the caller to progress after a finite number of steps, and the completion of the method may be signalled via some additional mechanism (it might be a registered callback, a Future, or a message).
A synchronous API may use blocking to implement synchrony, but this is not a necessity. A very CPU intensive task might give a similar behavior as blocking. In general, it is preferred to use asynchronous APIs, as they guarantee that the system is able to progress.

Non-blocking vs. Blocking

We talk about blocking if the delay of one thread can indefinitely delay some of the other threads. A good example is a resource which can be used exclusively by one thread using mutual exclusion. If a thread holds on to the resource indefinitely (for example accidentally running an infinite loop) other threads waiting on the resource can not progress. In contrast, non-blocking means that no thread is able to indefinitely delay others.
Non-blocking operations are preferred to blocking ones, as the overall progress of the system is not trivially guaranteed when it contains blocking operations.

以上文獻摘自akka文檔,一個方法之所以被稱為同步方法,是因為直到該方法返回某值或者拋出異常,該方法的調用者才能得到結果(make progress)。如果一個非同步調用需要通過額外的機制(比如callback,Future,message)。如果一個線程的延遲導致了另一個(一些)線程的延遲,那麼久出現了阻塞(blocking)。一個例子就是一個資源被一個線程所獨占,那麼其他線程需要等待這個線程釋放資源才能繼續執行。

scala中的FuturePromise都是非阻塞的執行,既可以通過回調函數獲取結果,但是也可以通過阻塞的方法串列獲取結果。

Future

一個Future會持有一個值,雖然這個值在將來某個時間點才可用。

  1. 如果計算未完成,那麼這個Future就未完成。
  2. 如果計算完成(得到結果或者異常),這個Future就完成了。

Future只能被指派一次,一旦Future給定了一個值或異常,它的結果不能修改。

object Main extends App {
  import scala.concurrent.ExecutionContext.Implicits.global
  import scala.concurrent.Future
  val f:Future[Int]=Future{
    Thread.sleep(100)//模擬某個耗時操作 比如網路請求
    println("haha")
    10
  }
  Thread.sleep(1000)
}
  //haha

非同步方法獲取結果

目的獲取結果,而不是控制執行過程。
scala提供了onSuccess等回調函數。其簽名為:def onComplete[U](f: Try[T] => U)(implicit executor: ExecutionContext): Unit

f.onComplete({
    case Success(i) => println(i)
    case Failure(e) => e.printStackTrace()
  })

以上代碼採用偏函數形式,或者:

  f.onComplete(result=>result match {
    case Success(i)=>println(i)
    case Failure(e)=>e.printStackTrace()
  })

還可以註冊多個回調:

f.onComplete(result=>result match {
    case Success(i)=>println(i)
  })
f.onComplete(result=>result match {
    case Success(i)=>println(i+20)
  })

註:多個回調函數之間並不保證執行順序

同步方法獲取結果

通過Await.result可以同步獲取結果,或者超時或者異常。Await.ready等待計算完成,不返回結果。

val r=Await.result(f,Duration.Inf) //Await.result(f,1 seconds)

Promise

除了通過Future.apply創建Future對象,還可以使用Promise.future。如果說Future是一個只讀的,值還沒計算的占位符。那麼Promise就是一個可寫的,單次指派的容器。Promise可以通過調用success代表Future成功完成,failure方法拋出異常。或者更抽象的complete

object Main extends App {
  import scala.concurrent.ExecutionContext.Implicits.global
  import scala.concurrent.{Future,Promise}
//  import scala.concurrent.duration._
  val p=Promise[Int]
  val f=p.future
  val producer=Future{
    p complete Try{
      100
    }
  }
  val consumer=Future{
    f onComplete{
      case Success(i)=>println(i)
      case Failure(e)=>e.printStackTrace()
    }
  }
  Thread.sleep(1000)
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 閉包是指在創建時封裝周圍狀態的函數。即使閉包所在的環境不存在了,閉包中封裝的狀態依然存在。匿名函數就是沒有名稱的函數。匿名函數可以賦值給變數,還能像其他任何PHP對象那樣傳遞。不過匿名函數仍是函數,因此可以調用,還可以傳入參數。匿名函數特別適合作為函數或方法的回調。 ...
  • [1]修改hosts文件 [2]啟動apache虛擬主機功能 [3]修改vhosts配置文件 ...
  • 可選的<generator>子元素是 一個Java類的名稱,用來生成該持久化類實例的唯一標識符。如果這個生成器實例需要某些配置值或者初始化參數,可以使用<param>元素來傳遞這些參數。 所有的生成器都實現了org.hibernate.id.IdentifierGenerator介面。這是一個非常簡 ...
  • 一、簡介 Swagger的目標是為REST API定義一個與語言無關的標準介面,允許用戶發現和理解電腦服務的功能,而無需訪問源代碼。當通過Swagger正確定義時,用戶可以用最少量的實現邏輯理解遠程服務並與之交互。類似於低級編程所做的介面。 二、實現步驟 1、添加 Maven 依賴 2、Swagg ...
  • 緩存一致性協議給緩存行(通常為64位元組)定義了個狀態:獨占(exclusive)、共用(share)、修改(modified)、失效(invalid),用來描述該緩存行是否被多處理器共用、是否修改。所以緩存一致性協議也稱MESI。 ...
  • 相對於版本1.0,多了很多方法, 比如,獲取文件的尾碼名,或修改尾碼名和一些文件的簡單操作。 文件複製到文件,文件複製到路徑,路徑下文件複製到新路徑下, 代碼如下,請享用: 組合併且封裝了File的一些方法,文件路徑就是File對象。 上邊都是字元流,至於線程開啟字元流的的運算,加油,你可以的。 ...
  • 五分鐘輕鬆學會管理項目開發環境。 在開發Python應用程式的時候,系統安裝的Python3只有一個版本:3.x。所有第三方的包都會被pip安裝到Python3的site-packages目錄下。 pycharm安裝可以在設置里進行管理。 如果我們要同時開發多個應用程式,每個應用可能需要各自擁有一套 ...
  • 最近忽然要用到在VerilogHDL中調用VHDL的模塊,從網上找了常式,把自己會忘掉的東西記在這裡,。 2選1多路復用器的VHDL描述:entity mux2_1 is port( dina : in bit; dinb : in bit; sel : in bit; dout : out bit ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...