成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

初學Java多線程:使用Synchronized關鍵字同步類方法

開發 后端
本文介紹使用Synchronized關鍵字同步類方法。要達成Java多線程的run方法同步,需要在void和public之間加上synchronized關鍵字。

要想解決“臟數據”的問題,最簡單的方法就是使用synchronized關鍵字來使run方法同步,代碼如下:

  1. public synchronized void run()  
  2. {  
  3.        

從上面的代碼可以看出,只要在void和public之間加上synchronized關鍵字,就可以使run方法同步,也就是說,對于同一個Java類的對象實例,run方法同時只能被一個線程調用,并當前的run執行完后,才能被其他的線程調用。即使當前線程執行到了run方法中的yield方法,也只是暫停了一下。由于其他線程無法執行run方法,因此,最終還是會由當前的線程來繼續執行。先看看下面的代碼:

sychronized關鍵字只和一個對象實例綁定

  1. class Test  
  2. {  
  3.       public synchronized void method()  
  4.      {  
  5.             
  6.      }  
  7. }  
  8.    
  9. public class Sync implements Runnable  
  10. {  
  11.      private Test test;  
  12.      public void run()  
  13.      {  
  14.           test.method();  
  15.      }  
  16.      public Sync(Test test)  
  17.      {  
  18.          this.test = test;  
  19.      }  
  20.      public static void main(String[] args) throws Exception  
  21.      {  
  22.          Test test1 =  new Test();  
  23.          Test test2 =  new Test();  
  24.          Sync sync1 = new Sync(test1);  
  25.          Sync sync2 = new Sync(test2);  
  26.          new Thread(sync1).start();  
  27.          new Thread(sync2).start();   
  28.      }  
  29.  } 

在Test類中的method方法是同步的。但上面的代碼建立了兩個Test類的實例,因此,test1和test2的method方法是分別執行的。要想讓method同步,必須在建立Sync類的實例時向它的構造方法中傳入同一個Test類的實例,如下面的代碼所示:

Sync sync1 = new Sync(test1);     不僅可以使用synchronized來同步非靜態方法,也可以使用synchronized來同步靜態方法。如可以按如下方式來定義method方法:

  1. class Test   
  2. {  
  3.     public static synchronized void method() {   }  

建立Test類的對象實例如下:

  1. Test test = new Test(); 

對于靜態方法來說,只要加上了synchronized關鍵字,這個方法就是同步的,無論是使用test.method(),還是使用Test.method()來調用method方法,method都是同步的,并不存在非靜態方法的多個實例的問題。

在23種設計模式中的單件(Singleton)模式如果按傳統的方法設計,也是線程不安全的,下面的代碼是一個線程不安全的單件模式。

  1. package test;  
  2.  
  3. // 線程安全的Singleton模式  
  4. class Singleton  
  5. {  
  6.     private static Singleton sample;  
  7.  
  8.     private Singleton()  
  9.     {  
  10.     }  
  11.     public static Singleton getInstance()  
  12.     {  
  13.         if (sample == null)  
  14.         {  
  15.             Thread.yield(); // 為了放大Singleton模式的線程不安全性  
  16.             sample = new Singleton();  
  17.         }  
  18.         return sample;  
  19.     }  
  20. }  
  21. public class MyThread extends Thread  
  22. {  
  23.     public void run()  
  24.     {  
  25.         Singleton singleton = Singleton.getInstance();  
  26.         System.out.println(singleton.hashCode());  
  27.     }  
  28.     public static void main(String[] args)  
  29.     {  
  30.         Thread threads[] = new Thread[5];  
  31.         for (int i = 0; i < threads.length; i++)  
  32.             threads[i] = new MyThread();  
  33.         for (int i = 0; i < threads.length; i++)  
  34.             threads[i].start();  
  35.     }  
  36. }  

在上面的代碼調用yield方法是為了使單件模式的線程不安全性表現出來,如果將這行去掉,上面的實現仍然是線程不安全的,只是出現的可能性小得多。

程序的運行結果如下:

25358555
26399554
7051261
29855319
5383406

上面的運行結果可能在不同的運行環境上有所有同,但一般這五行輸出不會完全相同。從這個輸出結果可以看出,通過getInstance方法得到的對象實例是五個,而不是我們期望的一個。這是因為當一個線程執行了Thread.yield()后,就將CPU資源交給了另外一個線程。由于在線程之間切換時并未執行到創建Singleton對象實例的語句,因此,這幾個線程都通過了if判斷,所以,就會產生了建立五個對象實例的情況(可能創建的是四個或三個對象實例,這取決于有多少個線程在創建Singleton對象之前通過了if判斷,每次運行時可能結果會不一樣)。

要想使上面的單件模式變成線程安全的,只要為getInstance加上synchronized關鍵字即可。代碼如下:

  1. public static synchronized Singleton getInstance() {   } 

當然,還有更簡單的方法,就是在定義Singleton變量時就建立Singleton對象,代碼如下:

  1. private static final Singleton sample = new Singleton(); 

然后在getInstance方法中直接將sample返回即可。這種方式雖然簡單,但不知在getInstance方法中創建Singleton對象靈活。讀者可以根據具體的需求選擇使用不同的方法來實現單件模式。

在使用synchronized關鍵字時有以下四點需要注意:

1.  synchronized關鍵字不能繼承。

雖然可以使用synchronized來定義方法,但synchronized并不屬于方法定義的一部分,因此,synchronized關鍵字不能被繼承。如果在父類中的某個方法使用了synchronized關鍵字,而在子類中覆蓋了這個方法,在子類中的這個方法默認情況下并不是同步的,而必須顯式地在子類的這個方法中加上synchronized關鍵字才可以。當然,還可以在子類方法中調用父類中相應的方法,這樣雖然子類中的方法不是同步的,但子類調用了父類的同步方法,因此,子類的方法也就相當于同步了。這兩種方式的例子代碼如下:

在子類方法中加上synchronized關鍵字

  1. class Parent  
  2. {  
  3.     public synchronized void method() {   }  
  4. }  
  5. class Child extends Parent  
  6. {  
  7.     public synchronized void method() {   }  

在子類方法中調用父類的同步方法

  1. class Parent  
  2. {  
  3.     public synchronized void method() {   }  
  4. }  
  5. class Child extends Parent  
  6. {  
  7.     public void method() { super.method();   }  

2.  在定義接口方法時不能使用synchronized關鍵字。

3.  構造方法不能使用synchronized關鍵字,但可以使用下節要討論的synchronized塊來進行同步。

4.  synchronized可以自由放置。

在前面的例子中使用都是將synchronized關鍵字放在方法的返回類型前面。但這并不是synchronized可放置***位置。在非靜態方法中,synchronized還可以放在方法定義的最前面,在靜態方法中,synchronized可以放在static的前面,代碼如下:

  1. public synchronized void method();  
  2. synchronized public void method();  
  3. public static synchronized void method();  
  4. public synchronized static void method();  
  5. synchronized public static void method(); 

但要注意,synchronized不能放在方法返回類型的后面,如下面的代碼是錯誤的:

  1. public void synchronized method();  
  2. public static void synchronized method(); 

synchronized關鍵字只能用來同步方法,不能用來同步類變量,如下面的代碼也是錯誤的。

  1. public synchronized int n = 0;  
  2. public static synchronized int n = 0

雖然使用synchronized關鍵字同步方法是最安全的同步方式,但大量使用synchronized關鍵字會造成不必要的資源消耗以及性能損失。雖然從表面上看synchronized鎖定的是一個方法,但實際上synchronized鎖定的是一個類。也就是說,如果在非靜態方法method1和method2定義時都使用了synchronized,在method1未執行完之前,method2是不能執行的。靜態方法和非靜態方法的情況類似。但靜態和非靜態方法不會互相影響。看看如下的代碼:

  1. package test;  
  2.  
  3. public class MyThread1 extends Thread  
  4. {  
  5.     public String methodName;  
  6.  
  7.     public static void method(String s)  
  8.     {  
  9.         System.out.println(s);  
  10.         while (true)  
  11.             ;  
  12.     }  
  13.     public synchronized void method1()  
  14.     {  
  15.         method("非靜態的method1方法");  
  16.     }  
  17.     public synchronized void method2()  
  18.     {  
  19.         method("非靜態的method2方法");  
  20.     }  
  21.     public static synchronized void method3()  
  22.     {  
  23.         method("靜態的method3方法");  
  24.     }  
  25.     public static synchronized void method4()  
  26.     {  
  27.         method("靜態的method4方法");  
  28.     }  
  29.     public void run()  
  30.     {  
  31.         try 
  32.         {  
  33.             getClass().getMethod(methodName).invoke(this);  
  34.         }  
  35.         catch (Exception e)  
  36.         {  
  37.         }  
  38.     }  
  39.     public static void main(String[] args) throws Exception  
  40.     {  
  41.         MyThread1 myThread1 = new MyThread1();  
  42.         for (int i = 1; i <= 4; i++)  
  43.         {  
  44.             myThread1.methodName = "method" + String.valueOf(i);  
  45.             new Thread(myThread1).start();  
  46.             sleep(100);  
  47.         }  
  48.     }  
  49. }  

運行結果如下:

非靜態的method1方法
靜態的method3方法

從上面的運行結果可以看出,method2和method4在method1和method3未結束之前不能運行。因此,我們可以得出一個結論,如果在類中使用synchronized關鍵字來定義非靜態方法,那將影響這個中的所有使用synchronized關鍵字定義的非靜態方法。如果定義的是靜態方法,那么將影響類中所有使用synchronized關鍵字定義的靜態方法。這有點象數據表中的表鎖,當修改一條記錄時,系統就將整個表都鎖住了,因此,大量使用這種同步方式會使程序的性能大幅度下降。

【編輯推薦】

  1. 初學Java多線程:從線程返回數據的兩種方法
  2. 初學Java多線程:向線程傳遞數據的三種方法
  3. 初學Java多線程:慎重使用volatile關鍵字
  4. 初學Java多線程:join方法的使用
  5. 初學Java多線程:線程的生命周期
責任編輯:yangsai 來源: 真的有外星人嗎
相關推薦

2009-06-29 18:32:52

Java多線程Synchronize

2009-06-29 18:14:23

Java多線程volatile關鍵字

2017-05-27 20:59:30

Java多線程synchronize

2009-06-29 18:44:28

Java多線程Synchronize同步變量

2019-12-20 15:19:41

Synchroinze線程安全

2009-08-12 13:37:01

Java synchr

2011-04-14 13:27:53

Synchronize多線程

2021-03-10 15:59:39

JavaSynchronize并發編程

2024-03-15 15:12:27

關鍵字底層代碼

2020-07-17 20:15:03

架構JMMvolatile

2009-06-29 18:08:51

Java多線程join方法

2024-11-20 15:55:57

線程Java開發

2010-03-15 18:11:38

Java多線程

2022-01-26 00:03:00

關鍵字線程JVM

2023-12-08 18:01:25

Java關鍵字

2021-01-12 09:22:18

Synchronize線程開發技術

2021-08-15 08:11:54

AndroidSynchronize關鍵字

2009-06-25 10:33:53

StaticJava類

2009-06-29 17:54:10

Java多線程Thread類創建線程

2011-03-09 14:36:44

synchronizevolatile
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品欧美一区喷水 | jav成人av免费播放 | 成人在线观看黄 | 国产精品国产三级国产aⅴ中文 | a在线视频 | 国产一区二区av | 久久久av | 精品国产视频 | 国产三级一区二区 | 国产欧美精品一区二区 | 久久福利网站 | 精品久久久久久亚洲综合网站 | 97超碰免费 | 精品久久久久久久久久久久久久久久久 | 欧美黑人国产人伦爽爽爽 | 久久精品国产一区二区电影 | 激情小说综合网 | 久久骚 | 国产在线永久免费 | 国产日韩久久久久69影院 | av黄色片在线观看 | 韩日一区 | 国产精品久久久久无码av | 欧美日韩在线免费观看 | 国产成人综合在线 | 欧美一级片中文字幕 | 911精品国产 | 午夜婷婷激情 | 日韩一区二区三区在线看 | 九九精品在线 | av在线播放免费 | 黄色免费在线观看 | 免费观看a级毛片在线播放 黄网站免费入口 | 亚洲精品一区二区在线观看 | 午夜伦4480yy私人影院 | 免费在线精品视频 | 亚洲一区二区电影在线观看 | 精品国产一区二区三区性色av | 国产精品一卡二卡三卡 | 中文字幕欧美一区 | 成人久久18免费网站麻豆 |