亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

Java中關鍵字synchronized的使用方法詳解

瀏覽:96日期:2023-02-11 11:48:26

synchronized是Java里的一個關鍵字,起到的一個效果是“監視器鎖”~~,它的功能就是保證操作的原子性,同時禁止指令重排序和保證內存的可見性!

public class TestDemo { static class Counter{public int count = 0;public void add(){ count++;} } public static void main(String[] args) throws InterruptedException {Counter counter = new Counter();Thread t1 = new Thread(){ @Override public void run() {for (int i = 0; i < 50000; i++) { counter.add();} }};Thread t2 = new Thread(){ @Override public void run() {for (int i = 0; i < 50000; i++) { counter.add();} }};//啟動兩個線程t1.start();t2.start();//等待兩個線程結束t1.join();t2.join();System.out.println(counter.count); }}

Java中關鍵字synchronized的使用方法詳解

此時的線程就是不安全的,如何解決呢?

給我們的Counter對象里的add方法加上synchronized關鍵字,針對這個方法進行了加鎖操作。進入代碼塊(調用方法)自動加鎖,出了代碼塊(方法結束),自動解鎖。

public class TestDemo { static class Counter{public int count = 0; //修飾方法 synchronized public void add{ count++;} } public static void main(String[] args) throws InterruptedException {Counter counter = new Counter();Thread t1 = new Thread(){ @Override public void run() {for (int i = 0; i < 50000; i++) { counter.add();} }};Thread t2 = new Thread(){ @Override public void run() {for (int i = 0; i < 50000; i++) { counter.add();} }};//啟動兩個線程t1.start();t2.start();//等待兩個線程結束t1.join();t2.join();System.out.println(counter.count); }}

Java中關鍵字synchronized的使用方法詳解

那么這里的代碼是如何保證正確的呢?

使用synchronized 就相當于在我們執行的指令里又加入了2條新指令。

LOCK (加鎖)

UNLOCK (解鎖)

LOCK操作特性:只有一個線程能執行成功!如果第一個線程執行成功,第二個線程也嘗試LOCK,就會阻塞等待,直到第一個線程執行UNLOCK 釋放鎖~

Java中關鍵字synchronized的使用方法詳解

通過LOCK和UNLOCK 就把 LOAD ADD SAVE 這三個指令,給打包成了一個原子的操作(中間不能被打斷,也不能被其他線程穿插)。

這里的加鎖也是保證原子性的核心操作,所以線程里的沒組指令就會順序執行,不在穿插執行,就保證了線程1執行完之后再去執行線程2。

舉個例子:

就好比張三和李四去ATM里去取錢,當張三進去取錢時,進去后就會鎖門,李四就會在外面等待,直到張三取完錢出來后,李四在進去取錢。

synchronized 也會禁止編譯器進行“內存可見性”和“指令重排序”的優化~ 同時程序運行的效率就會降低,也會導致線程之間相互去等待,就涉及到系統的一些調度,也會引入一些時間成本。

synchronized修飾的對象有以下幾種:

修飾一個代碼塊,被修飾的代碼塊稱為同步語句塊,其作用的范圍是大括號{}括起來的代碼,作用的對象是調用這個代碼塊的對象;

public class TestDemo{ public void methond() {// 進入代碼塊會鎖 this 指向對象中的鎖;// 出代碼塊會釋放 this 指向的對象中的鎖synchronized (this) {} } public static void main(String[] args) {TestDemo demo = new TestDemo();demo.methond(); }}

修飾一個方法,被修飾的方法稱為同步方法,其作用的范圍是整個方法,作用的對象是調用這個方法的對象;

public class TestDemo{ public synchronized void methond() { } public static void main(String[] args) {TestDemo demo = new TestDemo();demo.methond();// 進入方法會鎖 demo 指向對象中的鎖;// 出方法會釋放 demo 指向的對象中的鎖 }}

修改一個靜態的方法,其作用的范圍是整個靜態方法,作用的對象是這個類的所有對象;

public class TestDemo{ public synchronized static void methond() { } public static void main(String[] args) {methond();// 進入方法會鎖 TestDemo.class 指向對象中的鎖;//出方法會釋放 TestDemo.class 指向的對象中的鎖 }}

修改一個類,其作用的范圍是synchronized后面括號括起來的部分,作用主的對象是這個類的所有對象。

public class TestDemo{ public static void methond() {// 進入代碼塊會鎖 TestDemo.class 指向對象中的鎖;//出代碼塊會釋放 TestDemo.class 指向的對象中的鎖synchronized (TestDemo.class) {} } public static void main(String[] args) {TestDemo demo = new TestDemo();demo.methond(); }}

總結:

無論synchronized關鍵字加在方法上還是對象上,如果它作用的對象是非靜態的,則它取得的鎖是對象; 如果synchronized作用的對象是一個靜態方法或一個類,則它取得的鎖是對類,該類所有的對象同一把鎖。 每個對象只有一個鎖(lock)與之相關聯,誰拿到這個鎖誰就可以運行它所控制的那段代碼。 實現同步是要很大的系統開銷作為代價的,甚至可能造成死鎖,所以盡量避免無謂的同步控制。

拓展:

public class TestDemo { static class Counter{public int count = 0; public void add(){ synchronized (this){ count++; }} } public static void main(String[] args) throws InterruptedException {Counter counter = new Counter();Thread t1 = new Thread(){ @Override public void run() {for (int i = 0; i < 50000; i++) { synchronized (counter){counter.add(); }} }};Thread t2 = new Thread(){ @Override public void run() {for (int i = 0; i < 50000; i++) { synchronized (counter){counter.add(); }} }};//啟動兩個線程t1.start();t2.start();//等待兩個線程結束t1.join();t2.join();System.out.println(counter.count); }}

此時可以看出上述代碼,加了兩次鎖,會發生什么呢?

Java中關鍵字synchronized的使用方法詳解

Java中關鍵字synchronized的使用方法詳解

但是運行代碼發現程序依然正確運行?? 為什么

但是上述分析死鎖的思路是對的

只是因為synchronized內部使用特殊手段來處理了這種情況 。

這樣的操作特性我們叫做 可重入鎖

synchronized 內部記錄了當前這把鎖是哪個線程持有的~

如果當前加鎖線程和持有鎖的線程是同一個線程~

此時就并不是真的進行“加鎖操作”,而是把一個計數器加一;

如果后續該線程繼續嘗試獲取鎖,繼續判定加鎖線程和持有鎖線程是不是同一個線程,只要是同一個線程,就不真的加鎖,而是計數器+1;

如果該線程調用解鎖操作,也不是立刻就解鎖,而是計數器減1

直到計數器減成0了,才認為真的要“釋放鎖”,才允許其他線程來獲取鎖~~

總結

到此這篇關于Java中synchronized使用的文章就介紹到這了,更多相關Java中synchronized用法內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
主站蜘蛛池模板: 国产精选在线观看 | 日韩成人国产精品视频 | 日本九九视频 | 亚洲高清一区二区三区久久 | 日本日韩欧美 | 免看一级a毛片一片成人不卡 | 欧美一区二区手机在线观看视频 | 亚洲经典激情春色另类 | 日本美女a级片 | 国产啪精品| 欧美日韩精品国产一区在线 | 91大神大战酒店翘臀美女 | 亚洲爱爱图片 | 高清国产在线播放成人 | 久久无码精品一区二区三区 | 伊人中文字幕在线 | 狠狠干影视| 久久se精品一区二区影院 | 国产11一12周岁女毛片 | 91天堂亚洲精品一区 | 一级毛片真人免费观看 | 国产最强大片免费视频 | 国产91色综合久久免费分享 | 香蕉视频在线观看网站 | 国产成人综合一区人人 | 成人综合国产乱在线 | 国产人做人爱视频精品 | 黄色网址网站 | 国产午夜亚洲精品不卡 | 极品美女一级毛片免费 | 五月婷婷六月丁香综合 | 黄色激情视频网站 | 亚洲精品乱码电影在线观看 | 伊人丁香婷婷综合一区二区 | 精品视频午夜一区二区 | 国产精品亚洲片在线观看不卡 | 欧美一级片网 | 看全色黄大色黄大片色黄看的 | 一区二区视频在线播放 | 久久国产首页 | 成人区在线观看免费视频 |