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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...