偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Java 多線程同步問題的探究(二)

開發(fā) 后端
本文系列文章主要介紹詳細的討論Java多線程同步機制,同步機制是線程的一個非常重要的問題,希望對你有幫助。

上一篇中,我們講到了JAVA多線程是如何處理共享資源的,以及保證他們對資源進行互斥訪問所依賴的重要機制:對象鎖。

 

本篇中,我們來看一看傳統(tǒng)的同步實現(xiàn)方式以及這背后的原理。

二、給我一把鎖,我能創(chuàng)造一個規(guī)矩

 

很多人都知道,在Java多線程編程中,有一個重要的關鍵字,synchronized。但是很多人看到這個東西會感到困惑:“都說同步機制是通過對象鎖來實現(xiàn)的,但是這么一個關鍵字,我也看不出來Java程序鎖住了哪個對象阿?“

 

沒錯,我一開始也是對這個問題感到困惑和不解。不過還好,我們有下面的這個例程:

 

 

  1. public class ThreadTest extends Thread {  
  2. private int threadNo;  
  3. public ThreadTest(int threadNo) {  
  4. this.threadNo = threadNo;  
  5. }  
  6. public static void main(String[] args) throws Exception {  
  7. for (int i = 1; i < 10; i++) {  
  8. new ThreadTest(i).start();  
  9. Thread.sleep(1);  
  10. }  
  11. }  
  12. @Override 
  13. public synchronized void run() {  
  14. for (int i = 1; i < 10000; i++) {  
  15. System.out.println("No." + threadNo + ":" + i);  
  16. }  
  17. }  

 

 

這個程序其實就是讓10個線程在控制臺上數(shù)數(shù),從1數(shù)到9999。理想情況下,我們希望看到一個線程數(shù)完,然后才是另一個線程開始數(shù)數(shù)。但是這個程序的執(zhí)行過程告訴我們,這些線程還是亂糟糟的在那里搶著報數(shù),絲毫沒有任何規(guī)矩可言。

 

但是細心的讀者注意到:run方法還是加了一個synchronized關鍵字的,按道理說,這些線程應該可以一個接一個的執(zhí)行這個run方法才對阿。

 

但是通過上一篇中,我們提到的,對于一個成員方法加synchronized關鍵字,這實際上是以這個成員方法所在的對象本身作為對象鎖。在本例中,就是以ThreadTest類的一個具體對象,也就是該線程自身作為對象鎖的。一共十個線程,每個線程持有自己 線程對象的那個對象鎖。這必然不能產(chǎn)生同步的效果。換句話說,如果要對這些線程進行同步,那么這些線程所持有的對象鎖應當是共享且***的!

我們來看下面的例程:

 

 

  1. public class ThreadTest2 extends Thread {  
  2. private int threadNo;  
  3. private String lock;  
  4. public ThreadTest2(int threadNo, String lock) {  
  5. this.threadNo = threadNo;  
  6. this.lock = lock;  
  7. }  
  8. public static void main(String[] args) throws Exception {  
  9. String lock = new String("lock");  
  10. for (int i = 1; i < 10; i++) {  
  11. new ThreadTest2(i, lock).start();  
  12. Thread.sleep(1);  
  13. }  
  14. }  
  15. public void run() {  
  16. synchronized (lock) {  
  17. for (int i = 1; i < 10000; i++) {  
  18. System.out.println("No." + threadNo + ":" + i);  
  19. }  
  20. }  
  21. }  

 

 

我們注意到,該程序通過在main方法啟動10個線程之前,創(chuàng)建了一個String類型的對象。并通過ThreadTest2的構造函數(shù),將這個對象賦值給每一個ThreadTest2線程對象中的私有變量lock。根據(jù)Java方法的傳值特點,我們知道,這些線程的lock變量實際上指向的是堆內(nèi)存中的同一個區(qū)域,即存放main函數(shù)中的lock變量的區(qū)域。

程序?qū)⒃瓉韗un方法前的synchronized關鍵字去掉,換用了run方法中的一個synchronized塊來實現(xiàn)。這個同步塊的對象鎖,就是 main方法中創(chuàng)建的那個String對象。換句話說,他們指向的是同一個String類型的對象,對象鎖是共享且***的!

于是,我們看到了預期的效果:10個線程不再是爭先恐后的報數(shù)了,而是一個接一個的報數(shù)。

再來看下面的例程:

 

 

  1. public class ThreadTest3 extends Thread {  
  2. private int threadNo;  
  3. private String lock;  
  4. public ThreadTest3(int threadNo) {  
  5. this.threadNo = threadNo;  
  6. }  
  7. public static void main(String[] args) throws Exception {  
  8. //String lock = new String("lock");  
  9. for (int i = 1; i < 20; i++) {  
  10. new ThreadTest3(i).start();  
  11. Thread.sleep(1);  
  12. }  
  13. }  
  14. public static synchronized void abc(int threadNo) {  
  15. for (int i = 1; i < 10000; i++) {  
  16. System.out.println("No." + threadNo + ":" + i);  
  17. }  
  18. }  
  19. public void run() {  
  20. abc(threadNo);  
  21. }  

 

 

細心的讀者發(fā)現(xiàn)了:這段代碼沒有使用main方法中創(chuàng)建的String對象作為這10個線程的線程鎖。而是通過在run方法中調(diào)用本線程中一個靜態(tài)的同步方法abc而實現(xiàn)了線程的同步。我想看到這里,你們應該很困惑:這里synchronized靜態(tài)方法是用什么來做對象鎖的呢?

我們知道,對于同步靜態(tài)方法,對象鎖就是該靜態(tài)放發(fā)所在的類的Class實例,由于在JVM中,所有被加載的類都有***的類對象,具體到本例,就是***的 ThreadTest3.class對象。不管我們創(chuàng)建了該類的多少實例,但是它的類實例仍然是一個!

這樣我們就知道了:

 

1、對于同步的方法或者代碼塊來說,必須獲得對象鎖才能夠進入同步方法或者代碼塊進行操作;

2、如果采用method級別的同步,則對象鎖即為method所在的對象,如果是靜態(tài)方法,對象鎖即指method所在的
Class對象(***);

3、對于代碼塊,對象鎖即指synchronized(abc)中的abc;

4、因為***種情況,對象鎖即為每一個線程對象,因此有多個,所以同步失效,第二種共用同一個對象鎖lock,因此同步生效,第三個因為是static因此對象鎖為ThreadTest3的class 對象,因此同步生效。

如上述正確,則同步有兩種方式,同步塊和同步方法(為什么沒有wait和notify?這個我會在補充章節(jié)中做出闡述)

如果是同步代碼塊,則對象鎖需要編程人員自己指定,一般有些代碼為synchronized(this)只有在單態(tài)模式才生效;
(本類的實例有且只有一個)

如果是同步方法,則分靜態(tài)和非靜態(tài)兩種。

靜態(tài)方法則一定會同步,非靜態(tài)方法需在單例模式才生效,推薦用靜態(tài)方法(不用擔心是否單例)。

所以說,在Java多線程編程中,最常見的synchronized關鍵字實際上是依靠對象鎖的機制來實現(xiàn)線程同步的。
我們似乎可以聽到synchronized在向我們說:“給我一把鎖,我能創(chuàng)造一個規(guī)矩”。

下一篇中,我們將看到JDK 5提供的新的同步機制,也就是大名鼎鼎的Doug Lee提供的Java Concurrency框架。

【編輯推薦】

  1. WordPress的JavaScript本地化
  2. Javascript的this用法
  3. 在Java中>、>>、>>>三者的區(qū)別
  4. JAVA虛擬機內(nèi)存分配與回收機制
  5. Java中靜態(tài)變量的適用場景
責任編輯:于鐵 來源: 互聯(lián)網(wǎng)
相關推薦

2011-06-22 13:47:16

Java多線程

2012-06-05 02:12:55

Java多線程

2009-07-01 17:34:03

Servlet和JSP

2019-07-31 09:06:35

Java跳槽那些事兒文章

2010-01-21 11:27:30

linux多線程機制線程同步

2010-03-15 19:37:00

Java多線程同步

2011-04-14 13:27:53

Synchronize多線程

2009-11-12 14:32:00

BGP路由協(xié)議

2015-07-22 09:39:38

IOS多線程同步

2015-07-22 09:51:51

iOS開發(fā)線程

2009-03-24 08:56:23

數(shù)據(jù)同步多線程Java

2015-09-10 09:30:54

Java多線程同步

2009-09-14 19:39:14

批量線程同步

2011-08-30 15:44:57

C#

2009-06-11 10:48:53

Java多線程

2010-03-15 16:47:30

Java多線程同步

2010-03-15 16:31:34

Java多線程

2013-07-16 12:13:27

iOS多線程多線程概念GCD

2013-06-08 13:07:23

Java線程池調(diào)度器

2024-08-28 08:00:00

點贊
收藏

51CTO技術棧公眾號