線程安全

線程安全

計算機程序代碼概念
線程安全是多線程編程時的計算機程序代碼中的一個概念。在擁有共享數據的多條線程并行執行的程序中,線程安全的代碼會通過同步機制保證各個線程都可以正常且正确的執行,不會出現數據污染等意外情況。
    中文名:線程安全 外文名: 适用領域: 所屬學科: 包含:增大 Size 的值 屬于:線程不安全 例如:Size=1

概述

如果你的代碼所在的進程中有多個線程在同時運行,而這些線程可能會同時運行這段代碼。如果每次運行結果和 單線程運行的結果是一樣的,而且其他的 變量的值也和預期的是一樣的,就是線程安全的。

或者說:一個類或者程序所提供的接口對于線程來說是 原子操作或者多個線程之間的切換不會導緻該接口的執行結果存在二義性,也就是說我們不用考慮同步的問題。

線程安全問題都是由 全局變量及 靜态變量引起的。

若每個線程中對 全局變量、 靜态變量隻有讀操作,而無寫操作,一般來說,這個 全局變量是線程安全的;若有多個線程同時執行寫操作,一般都需要考慮 線程同步,否則的話就可能影響線程安全。

安全性

類要成為線程安全的,首先必須在 單線程環境中有正确的行為。如果一個類實現正确(這是說它符合規格說明的另一種方式),那麼沒有一種對這個類的對象的操作序列(讀或者寫公共字段以及調用公共方法)可以讓對象處于無效狀态,觀察到對象處于無效狀态、或者違反類的任何不可 變量、前置條件或者後置條件的情況。

此外,一個類要成為線程安全的,在被多個線程訪問時,不管運行時環境執行這些線程有什麼樣的時序安排或者交錯,它必須仍然有如上所述的正确行為,并且在調用的代碼中沒有任何額外的同步。其效果就是,在所有線程看來,對于線程安全對象的操作是以固定的、全局一緻的順序發生的。

正确性與 線程安全性之間的關系非常類似于在描述 ACID(原子性、一緻性、獨立性和持久性) 事務時使用的一緻性與獨立性之間的關系:從特定線程的角度看,由不同線程所執行的對象操作是先後(雖然順序不定)而不是 并行執行的。

安全程度

線程安全性不是一個非真即假的命題。 Vector 的方法都是同步的,并且 Vector 明确地設計為在 多線程環境中工作。但是它的 線程安全性是有限制的,即在某些方法之間有狀态依賴(類似地,如果在叠代過程中 Vector 被其他線程修改,那麼由 Vector.iterator() 返回的 iterator會抛出ConcurrentModifiicationException)。

對于 Java 類中常見的 線程安全性級别,沒有一種 分類系統可被廣泛接受,不過重要的是在編寫類時盡量記錄下它們的線程安全行為。

Bloch 給出了描述五類線程安全性的分類方法:不可變、線程安全、有條件線程安全、線程兼容和線程對立。隻要明确地記錄下線程安全特性,那麼您是否使用這種系統都沒關系。這種系統有其局限性 -- 各類之間的界線不是百分之百地明确,而且有些情況它沒照顧到 -- 但是這套系統是一個很好的起點。這種 分類系統的核心是調用者是否可以或者必須用外部同步包圍操作(或者一系列操作)。下面幾節分别描述了 線程安全性的這五種類别。

不可變

不可變的對象一定是線程安全的,并且永遠也不需要額外的同步 。因為一個不可變的對象隻要構建正确,其外部可見狀态永遠也不會改變,永遠也不會看到它處于不一緻的狀态。Java 類庫中大多數基本數值類如 Integer 、 String 和 BigInteger 都是不可變的。

需要注意的是,對于Integer,該類不提供add方法,加法是使用+來直接操作。而+操作是不具線程安全的。這是提供原子操作類AtomicInteger的原因。

線程安全

線程安全的對象具有在上面“線程安全”一節中描述的屬性 -- 由類的規格說明所規定的約束在對象被多個線程訪問時仍然有效,不管運行時環境如何排列,線程都不需要任何額外的同步。這種 線程安全性保證是很嚴格的 -- 許多類,如 Hashtable 或者 Vector 都不能滿足這種嚴格的定義。

有條件的線程安全

有條件的線程安全類對于單獨的操作可以是線程安全的,但是某些操作序列可能需要外部同步。條件線程安全的最常見的例子是遍曆由 Hashtable 或者 Vector 或者返回的 叠代器 -- 由這些類返回的 fail-fast 叠代器假定在叠代器進行遍曆的時候底層集合不會有變化。為了保證其他線程不會在遍曆的時候改變集合,進行叠代的線程應該确保它是獨占性地訪問集合以實現遍曆的完整性。通常,獨占性的訪問是由對鎖的同步保證的 -- 并且類的文檔應該說明是哪個鎖(通常是對象的内部 監視器(intrinsic monitor))。

如果對一個有條件線程安全類進行記錄,那麼您應該不僅要記錄它是有條件線程安全的,而且還要記錄必須防止哪些操作序列的并發訪問。用戶可以合理地假設其他操作序列不需要任何額外的同步。

線程兼容

線程兼容類不是線程安全的,但是可以通過正确使用同步而在并發環境中安全地使用。這可能意味着用一個 synchronized 塊包圍每一個方法調用,或者創建一個包裝器對象,其中每一個方法都是同步的(就像 Collections.synchronizedList() 一樣)。也可能意味着用 synchronized 塊包圍某些操作序列。為了最大程度地利用線程兼容類,如果所有調用都使用同一個塊,那麼就不應該要求調用者對該塊同步。這樣做會使線程兼容的對象作為 變量實例包含在其他線程安全的對象中,從而可以利用其所有者對象的同步。

許多常見的類是線程兼容的,如集合類 ArrayList 和 HashMap 、 java.text.SimpleDateFormat 、或者 JDBC 類 Connection 和 ResultSet 。

線程對立

線程對立類是那些不管是否調用了外部同步都不能在并發使用時安全地呈現的類。線程對立很少見,當類修改靜态數據,而靜态數據會影響在其他線程中執行的其他類的行為,這時通常會出現線程對立。線程對立類的一個例子是調用 System.setOut() 的類。

相關詞條

相關搜索

其它詞條