Java 學習系列
  • 目錄
  • 新手初入門,切記熟練
    • 參考書籍
    • 安裝 Java
    • 安裝 Eclipse IDE
    • 首次操作 Eclipse
    • 初學常見錯誤 & 提高可讀性
  • 第一章:認識 JAVA
    • 第一支程式:新建檔案
    • 第一支程式:Hello Java
  • 第二章:簡單 Java 程式解析
    • 變數與輸出
    • 重點整理
  • 第三章:變數與資料型態
  • 認識變數與資料型態
  • 溢位
  • 更多資料型態
  • 資料型態的轉換
  • 第四章:運算子、運算式與敘述
    • 運算子、運算式
    • 程式範例
  • 第五章:選擇性敘述與迴圈
    • 選擇性敘述
    • 迴圈
    • 迴圈的跳離
  • 第六章:陣列
    • 一維陣列
    • 二維與多維陣列
  • 第七章:函數
    • 函數與引數
    • 函數傳遞陣列
    • 函數多載
    • 函數 SWAP
  • 第八章:類別
    • 定義類別
    • 使用類別
    • 類別中的函數成員相互呼叫
    • 類別中使用 this
    • 練習:建立圓形類別
    • 傳遞引數與回傳值
    • 多引數
    • 類別中函數的多載
    • 類別成員中的公有與私有
  • 第九章:建構元
    • 認識 建構元
    • 建構元的多載
    • 建構元間的呼叫 - 透過this
    • 建構元中的公有與私有
  • 第十章:實例變數與函數、類別變數與函數
    • 實例變數 與 實例函數
    • 類別變數
    • 類別函數
  • 第11章:類別的繼承
    • 認識類別的繼承
    • 範例:類別的繼承
    • 類別繼承中的建構元呼叫
    • Protected members
    • Overriding
    • super() 與 this()
    • 設定終止繼承
  • 第12章:抽象類別
    • 認識抽象類別
    • 範例:抽象類別
    • 使用抽象類別型態的變數建立物件
  • 第13章:介面Interface
    • 認識介面
    • 範例:介面
    • 介面中的多重繼承
    • 介面的延伸
  • 第14章:例外 Exception
    • 認識例外
    • 例外處理
    • 範例:例外處理
    • 認識例外類別
    • 例外類別捕捉多個例外
    • 於程式中拋出例外
    • 於指定函數拋出例外
    • 自定義例外類別
  • 第15章:執行緒
    • 認識多執行緒
    • 啟動執行緒
    • 實作 RUNNABLE 介面
    • 執行緒生命週期
    • 凍結執行緒 sleep()
    • 凍結執行緒 join()
    • 執行緒的順序
    • 同步處理 Synchronized
  • 習題範例
    • 產生不重複之亂數
  • 後記
  • 致謝
  • 首頁
Powered by GitBook
On this page
  • 範例題目
  • 完整程式碼(未經同步處理)
  • 完整範例程式碼(經同步處理)

Was this helpful?

  1. 第15章:執行緒

同步處理 Synchronized

synchronized 有「同步」的意思,在 Java 中 這個關鍵字可以使各執行緒在時間上做協調, 即 一次只允許一個執行緒進行處理,而其他的執行緒必須等待上個執行緒處理完後才可以進入處理。

當數個執行緒同時啟動,而且還共用同個變數,就會常發生無法發覺的錯誤。

範例題目

例如一個需要計算累加金額的變數 sum,但有兩個執行緒 p1 與 p2 同時共用。

參考以下範例:

慈善捐款接受捐款,而每次會計算出總捐款額(sum)。 今日有兩位善心人士(p1,p2),兩人每天(1次)都會捐100元,連續三天過後 總捐款額若無他人捐款, 那麼總額會是600元。

因應網路塞車等因素:可以加上睡眠,當每接受一次的捐款時,小睡0~1秒。

完整程式碼(未經同步處理)

讓我們先瞧瞧 沒有經過同步處理的程式:

class CDonate
{
   private static int sum=0;
   public static void add(int n)
   {
      int tmp=sum;
      tmp=tmp+n;
      try
      {
         Thread.sleep((int)(1000*Math.random()));
      }
      catch(InterruptedException e){}
      sum=tmp;
      System.out.println("捐款總額= "+sum);
  }
}
class CPerson extends Thread
{
   public void run()
   {
      for(int i=1;i<=3;i++)
         CDonate.add(100);
   }
}
public class synchronized_1
{
   public static void main(String args[])
   {
      CPerson p1=new CPerson();
      CPerson p2=new CPerson();
      p1.start();
      p2.start();
   }
}

當我們執行結果後,會發現執行結果跟預期的不同:

捐款總額竟然只是300,應該要是600呀?

原來在處理p1執行緒還未結束時,p2也開始進入add()當中,因此造成計算錯誤。 但透過加上 synchronized 關鍵字後,又會有怎樣的效果呢?

在 add() 加上synchronized 關鍵字

public synchronized static void add(int n){}

神奇的事情,發生了!! 執行結果正確了!捐款總額等於600

是我們所預期的總捐款金額。

那麼想想看,為什麼不用先前提過的 join() 來先限制p1執行緒結束後 再往下執行p2執行緒呢?

從本篇可以了解同步處理的重要與方便性。 而且我們要注意若共用同個變數,必須特別去注意存取的順序。

完整範例程式碼(經同步處理)

class CDonate
{
   private static int sum=0;
   public synchronized static void add(int n)
   {
      int tmp=sum;
      tmp=tmp+n;
      try
      {
         Thread.sleep((int)(1000*Math.random()));
      }
      catch(InterruptedException e){}
      sum=tmp;
      System.out.println("捐款總額= "+sum);
  }
}
class CPerson extends Thread
{
   public void run()
   {
      for(int i=1;i<=3;i++)
         CDonate.add(100);
   }
}
public class synchronized_2
{
   public static void main(String args[])
   {
      CPerson p1=new CPerson();
      CPerson p2=new CPerson();
      p1.start();
      p2.start();
   }
}
Previous執行緒的順序Next產生不重複之亂數

Last updated 4 years ago

Was this helpful?