使用Runnable實現(xiàn)數(shù)據(jù)共享
使用Runnable實現(xiàn)數(shù)據(jù)共享,供大家參考,具體內(nèi)容如下
先上代碼:
public class TestThread { private static final Logger logger = LoggerFactory.getLogger(TestThread.class); private final class MyRunnable implements Runnable { private int i; public MyRunnable() { this.i = 10; } public void run() { while(i > 0) { synchronized (this) { if (i > 0) { i--; logger.debug("{} buy one ticket, {} left. ", Thread.currentThread().getName(), i); } } } } } @Test public void testRunable() throws InterruptedException{ MyRunnable myRunnable = new MyRunnable(); Thread th1 = new Thread(myRunnable); Thread th2 = new Thread(myRunnable); th1.start(); th2.start(); th1.join(); th2.join(); } }
樓上的代碼很簡單,模擬一個售票系統(tǒng)。通過兩個Thread對象開啟兩條線程同時運行一個MyRunnable實例。
幾個注意點:
1. 沒有加上synchronised關(guān)鍵詞的話,即
public void run() { while(i > 0) { if (i > 0) { i--; logger.debug("{} buy one ticket, {} left. ", Thread.currentThread().getName(), i); } } }
系統(tǒng)的運行結(jié)果:
Thread-1 buy one ticket, 8 left. Thread-2 buy one ticket, 8 left. Thread-2 buy one ticket, 6 left. Thread-1 buy one ticket, 6 left. Thread-2 buy one ticket, 5 left. Thread-1 buy one ticket, 4 left. Thread-2 buy one ticket, 3 left. Thread-1 buy one ticket, 2 left. Thread-2 buy one ticket, 1 left. Thread-1 buy one ticket, 0 left.
可以看到,缺少同步的程序輸出明顯有問題。
2. 在進入同步代碼塊之后,還需要對i的值再進行一次判斷,即,如果不加if判斷:
public void run() { while(i > 0) { synchronized (this) { i--; logger.debug("{} buy one ticket, {} left. ", Thread.currentThread().getName(), i); } } }
程序的運行結(jié)果為:
Thread-2 buy one ticket, 9 left. Thread-2 buy one ticket, 8 left. Thread-2 buy one ticket, 7 left. Thread-2 buy one ticket, 6 left. Thread-2 buy one ticket, 5 left. Thread-2 buy one ticket, 4 left. Thread-2 buy one ticket, 3 left. Thread-2 buy one ticket, 2 left. Thread-2 buy one ticket, 1 left. Thread-2 buy one ticket, 0 left. Thread-1 buy one ticket, -1 left.
可以看出,出現(xiàn)了“多賣”的現(xiàn)象, 所以需要在進入同步代碼塊中再進行一次if判斷。
總結(jié)
synchronised用于互斥訪問共享變量, 并在同步代碼塊中使用if判斷更新共享變量。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java網(wǎng)絡(luò)編程UDP實現(xiàn)消息發(fā)送及聊天
這篇文章主要為大家詳細(xì)介紹了Java網(wǎng)絡(luò)編程UDP實現(xiàn)消息發(fā)送及聊天,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-07-07SpringMVC JSON數(shù)據(jù)交互實現(xiàn)過程解析
這篇文章主要介紹了SpringMVC JSON數(shù)據(jù)交互實現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-10-10interrupt()和線程終止方式_動力節(jié)點Java學(xué)院整理
線程的thread.interrupt()方法是中斷線程,將會設(shè)置該線程的中斷狀態(tài)位,即設(shè)置為true,中斷的結(jié)果線程是死亡、還是等待新的任務(wù)或是繼續(xù)運行至下一步,就取決于這個程序本身2017-05-05Java 多態(tài)中繼承的轉(zhuǎn)型詳解與用法分析
繼承是java面向?qū)ο缶幊碳夹g(shù)的一塊基石,因為它允許創(chuàng)建分等級層次的類。繼承就是子類繼承父類的特征和行為,使得子類對象(實例)具有父類的實例域和方法,或子類從父類繼承方法,使得子類具有父類相同的行為2021-10-10詳談Java編程之委托代理回調(diào)、內(nèi)部類以及匿名內(nèi)部類回調(diào)(閉包回調(diào))
下面小編就為大家?guī)硪黄斦凧ava編程之委托代理回調(diào)、內(nèi)部類以及匿名內(nèi)部類回調(diào)(閉包回調(diào))。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-05-05mybatis中 if-test 數(shù)字判斷的坑及解決
這篇文章主要介紹了mybatis中 if-test 數(shù)字判斷的坑及解決,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-06-06SpringBoot@DeleteMapping(/xxx/{id})請求報405的解決
這篇文章主要介紹了SpringBoot@DeleteMapping(/xxx/{id})請求報405的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-01-01SpringBoot項目獲取統(tǒng)一前綴配置及獲取非確定名稱配置方法
在SpringBoot項目中,使用@ConfigurationProperties注解可獲取統(tǒng)一前綴的配置,具體做法是創(chuàng)建配置類,使用prefix屬性指定配置的前綴,本文給大家介紹SpringBoot項目獲取統(tǒng)一前綴配置以及獲取非確定名稱配置方法,感興趣的朋友跟隨小編一起看看吧2024-09-09