解析Java編程之Synchronized鎖住的對(duì)象
圖片上傳 密碼修改為 synchronized是java中用于同步的關(guān)鍵字,一般我們通過(guò)Synchronized鎖住一個(gè)對(duì)象,來(lái)進(jìn)行線程同步。我們需要了解在程序執(zhí)行過(guò)程中,synchronized鎖住的到底是哪個(gè)對(duì)象,否則我們?cè)诙嗑€程的程序就有可能出現(xiàn)問(wèn)題。
看下面的代碼,我們定義了一個(gè)靜態(tài)變量n,在run方法中,我們使n增加10,然后在main方法中,我們開(kāi)辟了100個(gè)線程,來(lái)執(zhí)行n增加的操作,如果線程沒(méi)有并發(fā)執(zhí)行,那么n最后的值應(yīng)該為1000,顯然下面的程序執(zhí)行完結(jié)果不是1000,因?yàn)槲覀儧](méi)有進(jìn)行線程同步。
import java.util.concurrent.TimeUnit; public class SynchronizedTest1 extends Thread { public static int n = 0; public void run() { try { //使n自加10次 for (int i = 0; i < 10; i++) { n = n + 1; TimeUnit.MILLISECONDS.sleep(10); } } catch (InterruptedException e) { e.printStackTrace(); } } public static void main(String[] args) throws InterruptedException { Thread[] threads = new Thread[100]; for (int i = 0; i < threads.length; i++) { threads[i] = new SynchronizedTest1(); threads[i].start(); } //使所有其他線程執(zhí)行完,再繼續(xù)執(zhí)行main線程,這樣得出的n是最終的結(jié)果 for (Thread thread : threads) { thread.join(); } System.out.println(n); } }
為了實(shí)現(xiàn)同步,我們修改上面的代碼,增加一個(gè)increase方法,如下。但是當(dāng)我們執(zhí)行下面的代碼時(shí),會(huì)發(fā)現(xiàn)n仍然不是1000.
import java.util.concurrent.TimeUnit; public class SynchronizedTest2 extends Thread { public static int n = 0; public synchronized void increase() { n++; } public void run() { try { //使n自加10次 for (int i = 0; i < 10; i++) { increase(); TimeUnit.MILLISECONDS.sleep(10); } } catch (InterruptedException e) { e.printStackTrace(); } } public static void main(String[] args) throws InterruptedException { Thread[] threads = new Thread[100]; for (int i = 0; i < threads.length; i++) { threads[i] = new SynchronizedTest2(); threads[i].start(); } //使所有其他線程執(zhí)行完,再繼續(xù)執(zhí)行main線程,這樣得出的n是最終的結(jié)果 for (Thread thread : threads) { thread.join(); } System.out.println(n); } }
其實(shí)原因很簡(jiǎn)單,上面的多個(gè)線程在執(zhí)行時(shí)根本就沒(méi)有競(jìng)爭(zhēng)同一個(gè)對(duì)象鎖。當(dāng)我們執(zhí)行用synchronized修飾的非靜態(tài)方法時(shí),線程會(huì)首先獲得調(diào)用這個(gè)方法的對(duì)象的鎖,然后才能繼續(xù)執(zhí)行代碼。那么調(diào)用這個(gè)方法的到底是哪個(gè)對(duì)象,是this對(duì)象。在上面的例子中,thread[i]所代表的線程獲取的鎖對(duì)象是thread[i]對(duì)象,也就是該線程對(duì)象本身。因此上面所開(kāi)辟的100個(gè)線程只要獲得自身對(duì)象就可以執(zhí)行,這樣就使同步失去了作用。
我們?cè)俅涡薷拇a:即將increase方法改為i靜態(tài)的,此時(shí)程序執(zhí)行完后n的值為1000。
import java.util.concurrent.TimeUnit; public class SynchronizedTest3 extends Thread { public static int n = 0; public synchronized static void increase() { n++; } public void run() { try { //使n自加10次 for (int i = 0; i < 10; i++) { increase(); TimeUnit.MILLISECONDS.sleep(10); } } catch (InterruptedException e) { e.printStackTrace(); } } public static void main(String[] args) throws InterruptedException { Thread[] threads = new Thread[100]; for (int i = 0; i < threads.length; i++) { threads[i] = new SynchronizedTest3(); threads[i].start(); } //使所有其他線程執(zhí)行完,再繼續(xù)執(zhí)行main線程,這樣得出的n是最終的結(jié)果 for (Thread thread : threads) { thread.join(); } System.out.println(n); } }
當(dāng)synchronized 修飾static方法,它鎖住的是該類的Class對(duì)象,而不是某一個(gè)具體對(duì)象。在上面的例子中,它鎖住的就是SynchronizedTest3.class對(duì)象。在程序執(zhí)行過(guò)程中,類的Class對(duì)象只有一份,所以上面線程競(jìng)爭(zhēng)的是同一個(gè)對(duì)象鎖。
下面是對(duì)synchronized鎖住對(duì)象的總結(jié):
(1)對(duì)于同步方法,鎖當(dāng)前對(duì)象(this)
(2)對(duì)于靜態(tài)同步方法,鎖當(dāng)前類的Class對(duì)象
(3)對(duì)于同步代碼塊,鎖住的是synchronized括號(hào)中的對(duì)象
總結(jié)
以上就是本文關(guān)于解析Java編程之Synchronized鎖住的對(duì)象的全部?jī)?nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站:Java編程redisson實(shí)現(xiàn)分布式鎖代碼示例、Java并發(fā)編程之重入鎖與讀寫鎖等,有什么問(wèn)題可以直接留言,小編會(huì)及時(shí)回復(fù)大家的。下面推薦本站基本Java編程相關(guān)的書(shū)籍,免費(fèi)下載,供朋友們學(xué)習(xí)參考。
Java初級(jí)開(kāi)發(fā)工程師面試題匯總.PDF
http://www.dbjr.com.cn/books/576989.html
Java經(jīng)典實(shí)例(第三版) 完整版 ([美]達(dá)爾文) 中文pdf掃描版
http://www.dbjr.com.cn/books/577859.html
希望大家能夠喜歡。
相關(guān)文章
IDEA 非常重要的一些設(shè)置項(xiàng)(一連串的問(wèn)題差點(diǎn)讓我重新用回 Eclipse)
這篇文章主要介紹了IDEA 非常重要的一些設(shè)置項(xiàng)(一連串的問(wèn)題差點(diǎn)讓我重新用回 Eclipse),本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-08-08Java自定義注解實(shí)現(xiàn)Redis自動(dòng)緩存的方法
本篇文章主要介紹了Java自定義注解實(shí)現(xiàn)Redis自動(dòng)緩存的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2017-04-04Maven編譯錯(cuò)誤:程序包c(diǎn)om.sun.*包不存在的三種解決方案
J2SE中的類大致可以劃分為以下的各個(gè)包:java.*,javax.*,org.*,sun.*,本文文章主要介紹了maven編譯錯(cuò)誤:程序包c(diǎn)om.sun.xml.internal.ws.spi不存在的解決方案,感興趣的可以了解一下2024-02-02idea中啟動(dòng)項(xiàng)目彈出 IDEA out of memory窗口的解決方案
這篇文章主要介紹了idea中啟動(dòng)項(xiàng)目彈出 IDEA out of memory窗口的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01java 反射 動(dòng)態(tài)調(diào)用不同類的靜態(tài)方法(推薦)
下面小編就為大家?guī)?lái)一篇JAVA 反射 動(dòng)態(tài)調(diào)用不同類的靜態(tài)方法(推薦)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-08-08