
Java 教程
- Java - 首頁
- Java - 概述
- Java - 歷史
- Java - 特性
- Java 與 C++
- JVM - Java 虛擬機器
- Java - JDK 與 JRE 與 JVM
- Java - Hello World 程式
- Java - 環境搭建
- Java - 基本語法
- Java - 變數型別
- Java - 資料型別
- Java - 型別轉換
- Java - Unicode 系統
- Java - 基本運算子
- Java - 註釋
- Java - 使用者輸入
- Java - 日期和時間
Java 控制語句
- Java - 迴圈控制
- Java - 決策制定
- Java - If-else
- Java - Switch
- Java - For 迴圈
- Java - For-Each 迴圈
- Java - While 迴圈
- Java - do-while 迴圈
- Java - Break
- Java - Continue
面向物件程式設計
- Java - OOPs 概念
- Java - 物件和類
- Java - 類屬性
- Java - 類方法
- Java - 方法
- Java - 變數作用域
- Java - 建構函式
- Java - 訪問修飾符
- Java - 繼承
- Java - 聚合
- Java - 多型
- Java - 重寫
- Java - 方法過載
- Java - 動態繫結
- Java - 靜態繫結
- Java - 例項初始化塊
- Java - 抽象
- Java - 封裝
- Java - 介面
- Java - 包
- Java - 內部類
- Java - 靜態類
- Java - 匿名類
- Java - 單例類
- Java - 包裝類
- Java - 列舉
- Java - 列舉建構函式
- Java - 列舉字串
Java 內建類
Java 檔案處理
Java 錯誤和異常
- Java - 異常
- Java - try-catch 塊
- Java - try-with-resources
- Java - 多重捕獲塊
- Java - 巢狀 try 塊
- Java - Finally 塊
- Java - throw 異常
- Java - 異常傳播
- Java - 內建異常
- Java - 自定義異常
Java 多執行緒
- Java - 多執行緒
- Java - 執行緒生命週期
- Java - 建立執行緒
- Java - 啟動執行緒
- Java - 合併執行緒
- Java - 執行緒命名
- Java - 執行緒排程器
- Java - 執行緒池
- Java - 主執行緒
- Java - 執行緒優先順序
- Java - 守護執行緒
- Java - 執行緒組
- Java - 關閉鉤子
Java 同步
Java 網路
- Java - 網路
- Java - 套接字程式設計
- Java - URL 處理
- Java - URL 類
- Java - URLConnection 類
- Java - HttpURLConnection 類
- Java - Socket 類
- Java - 泛型
Java 集合
Java 介面
Java 資料結構
Java 集合演算法
高階 Java
- Java - 命令列引數
- Java - Lambda 表示式
- Java - 傳送郵件
- Java - Applet 基礎
- Java - Javadoc 註釋
- Java - 自動裝箱和拆箱
- Java - 檔案不匹配方法
- Java - REPL (JShell)
- Java - 多版本 Jar 檔案
- Java - 私有介面方法
- Java - 內部類菱形運算子
- Java - 多解析度影像 API
- Java - 集合工廠方法
- Java - 模組系統
- Java - Nashorn JavaScript
- Java - Optional 類
- Java - 方法引用
- Java - 函式式介面
- Java - 預設方法
- Java - Base64 編碼解碼
- Java - Switch 表示式
- Java - Teeing 收集器
- Java - 微基準測試
- Java - 文字塊
- Java - 動態 CDS 歸檔
- Java - Z 垃圾收集器 (ZGC)
- Java - 空指標異常
- Java - 打包工具
- Java - 密封類
- Java - 記錄類
- Java - 隱藏類
- Java - 模式匹配
- Java - 緊湊數字格式化
- Java - 垃圾回收
- Java - JIT 編譯器
Java 雜項
- Java - 遞迴
- Java - 正則表示式
- Java - 序列化
- Java - 字串
- Java - 程序 API 改進
- Java - 流 API 改進
- Java - 增強的 @Deprecated 註解
- Java - CompletableFuture API 改進
- Java - 流
- Java - 日期時間 Api
- Java 8 - 新特性
- Java 9 - 新特性
- Java 10 - 新特性
- Java 11 - 新特性
- Java 12 - 新特性
- Java 13 - 新特性
- Java 14 - 新特性
- Java 15 - 新特性
- Java 16 - 新特性
Java API 和框架
Java 類引用
- Java - Scanner
- Java - 陣列
- Java - 字串
- Java - Date
- Java - ArrayList
- Java - Vector
- Java - Stack
- Java - PriorityQueue
- Java - LinkedList
- Java - ArrayDeque
- Java - HashMap
- Java - LinkedHashMap
- Java - WeakHashMap
- Java - EnumMap
- Java - TreeMap
- Java - IdentityHashMap
- Java - HashSet
- Java - EnumSet
- Java - LinkedHashSet
- Java - TreeSet
- Java - BitSet
- Java - Dictionary
- Java - Hashtable
- Java - Properties
- Java - Collection
- Java - Array
Java 有用資源
Java - 塊同步
當我們在 Java 程式中啟動 兩個或多個執行緒 時,可能會出現多個執行緒嘗試訪問相同資源的情況,最終由於併發問題而產生不可預見的結果。例如,如果多個執行緒嘗試寫入同一個檔案,則它們可能會損壞資料,因為其中一個執行緒可能會覆蓋資料,或者當一個執行緒正在開啟同一個檔案時,另一個執行緒可能正在關閉同一個檔案。
因此,需要同步多個執行緒的操作,並確保在給定時間點只有一個執行緒可以訪問資源。這是使用稱為監視器的概念實現的。Java 中的每個物件都與一個監視器關聯,執行緒可以鎖定或解鎖該監視器。一次只有一個執行緒可以持有監視器的鎖。
Java 中的塊同步
Java 程式語言提供了一種非常方便的方法來建立執行緒並使用同步塊 同步 它們的作業。您將共享資源保留在此塊中。
語法
以下是 synchronized 語句的一般形式:
synchronized(objectidentifier) { // Access shared variables and other shared resources }
這裡,objectidentifier 是對一個物件的引用,其鎖與 synchronized 語句表示的監視器關聯。現在我們將看兩個示例,我們將在其中使用兩個不同的執行緒列印計數器。當執行緒不同步時,它們列印的計數器值不是按順序的,但是當我們將計數器放在 synchronized() 塊內列印時,它會為兩個執行緒按順序列印計數器。
無同步的多執行緒示例
這是一個簡單的示例,它可能列印或不列印按順序的計數器值,並且每次我們執行它時,它都會根據 CPU 對執行緒的可用性產生不同的結果。
示例
package com.tutorialspoint; class PrintDemo { public void printCount() { try { for(int i = 5; i > 0; i--) { Thread.sleep(50); System.out.println("Counter --- " + i ); } } catch (Exception e) { System.out.println("Thread interrupted."); } } } class ThreadDemo extends Thread { private Thread t; private String threadName; PrintDemo printDemo; ThreadDemo( String name, PrintDemo pd) { threadName = name; printDemo = pd; } public void run() { printDemo.printCount(); System.out.println("Thread " + threadName + " exiting."); } public void start () { System.out.println("Starting " + threadName ); if (t == null) { t = new Thread (this, threadName); t.start (); } } } public class TestThread { public static void main(String args[]) { PrintDemo printDemo = new PrintDemo(); ThreadDemo t1 = new ThreadDemo( "Thread - 1 ", printDemo ); ThreadDemo t2 = new ThreadDemo( "Thread - 2 ", printDemo ); t1.start(); t2.start(); // wait for threads to end try { t1.join(); t2.join(); } catch ( Exception e) { System.out.println("Interrupted"); } } }
輸出
每次執行此程式都會產生不同的結果:
Starting Thread - 1 Starting Thread - 2 Counter --- 5 Counter --- 5 Counter --- 4 Counter --- 4 Counter --- 3 Counter --- 3 Counter --- 2 Counter --- 2 Counter --- 1 Counter --- 1 Thread Thread - 1 exiting. Thread Thread - 2 exiting.
塊級同步的多執行緒示例
這是同一個示例,它按順序列印計數器值,並且每次我們執行它時,它都會產生相同的結果。我們在塊上放置了 synchronized 關鍵字,以便計數器增量程式碼現在在方法執行期間根據物件被鎖定。我們使用當前物件作為鎖,將其作為引數傳遞給 synchronized 塊。
示例
package com.tutorialspoint; class PrintDemo { public void printCount() { try { for(int i = 5; i > 0; i--) { Thread.sleep(50); System.out.println("Counter --- " + i ); } } catch (Exception e) { System.out.println("Thread interrupted."); } } } class ThreadDemo extends Thread { private Thread t; private String threadName; PrintDemo printDemo; ThreadDemo( String name, PrintDemo pd) { threadName = name; printDemo = pd; } public void run() { synchronized(printDemo) { printDemo.printCount(); } System.out.println("Thread " + threadName + " exiting."); } public void start () { System.out.println("Starting " + threadName ); if (t == null) { t = new Thread (this, threadName); t.start (); } } } public class TestThread { public static void main(String args[]) { PrintDemo printDemo = new PrintDemo(); ThreadDemo t1 = new ThreadDemo( "Thread - 1 ", printDemo ); ThreadDemo t2 = new ThreadDemo( "Thread - 2 ", printDemo ); t1.start(); t2.start(); // wait for threads to end try { t1.join(); t2.join(); } catch ( Exception e) { System.out.println("Interrupted"); } } }
輸出
每次執行此程式都會產生相同的結果:
Starting Thread - 1 Starting Thread - 2 Counter --- 5 Counter --- 4 Counter --- 3 Counter --- 2 Counter --- 1 Thread Thread - 1 exiting. Counter --- 5 Counter --- 4 Counter --- 3 Counter --- 2 Counter --- 1 Thread Thread - 2 exiting.
方法級同步的多執行緒示例
這是同一個示例,它按順序列印計數器值,並且每次我們執行它時,它都會產生相同的結果。這次我們在方法上放置了 synchronized 關鍵字,以便整個方法在方法執行期間根據物件被鎖定。
示例
package com.tutorialspoint; class PrintDemo extends Thread { public void printCount() { try { for(int i = 5; i > 0; i--) { Thread.sleep(50); System.out.println("Counter --- " + i ); } } catch (Exception e) { System.out.println("Thread " + Thread.currentThread().getName()+" interrupted."); } } public synchronized void run() { printCount(); System.out.println("Thread " + Thread.currentThread().getName() + " exiting."); } } public class TestThread { public static void main(String args[]) { PrintDemo printDemo = new PrintDemo(); Thread t1 = new Thread(printDemo ); Thread t2 = new Thread(printDemo ); t1.start(); t2.start(); // wait for threads to end try { t1.join(); t2.join(); } catch ( Exception e) { System.out.println("Interrupted"); } } }
輸出
每次執行此程式都會產生相同的結果:
Counter --- 5 Counter --- 4 Counter --- 3 Counter --- 2 Counter --- 1 Thread Thread-1 exiting. Counter --- 5 Counter --- 4 Counter --- 3 Counter --- 2 Counter --- 1 Thread Thread-2 exiting.