使用Runnable實(shí)現(xiàn)數(shù)據(jù)共享
使用Runnable實(shí)現(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();
}
}
樓上的代碼很簡(jiǎn)單,模擬一個(gè)售票系統(tǒng)。通過(guò)兩個(gè)Thread對(duì)象開啟兩條線程同時(shí)運(yùn)行一個(gè)MyRunnable實(shí)例。
幾個(gè)注意點(diǎn):
1. 沒(méi)有加上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)的運(yùn)行結(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.
可以看到,缺少同步的程序輸出明顯有問(wèn)題。
2. 在進(jìn)入同步代碼塊之后,還需要對(duì)i的值再進(jìn)行一次判斷,即,如果不加if判斷:
public void run() {
while(i > 0) {
synchronized (this) {
i--;
logger.debug("{} buy one ticket, {} left. ", Thread.currentThread().getName(), i);
}
}
}
程序的運(yùn)行結(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)象, 所以需要在進(jìn)入同步代碼塊中再進(jìn)行一次if判斷。
總結(jié)
synchronised用于互斥訪問(wèn)共享變量, 并在同步代碼塊中使用if判斷更新共享變量。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
spring security認(rèn)證異常后返回中文提示的問(wèn)題
這篇文章主要介紹了spring security認(rèn)證異常后返回中文提示的問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-02-02
springboot項(xiàng)目以jar包運(yùn)行的操作方法
公司一個(gè)springboot項(xiàng)目本來(lái)是打war包的,突然要改為打jar包,不知所措了,糾結(jié)該如何操作呢,折騰半天終于搞定了,下面把解決方案分享給大家,對(duì)springboot打jar包方式感興趣的朋友一起看看吧2021-06-06
Spring boot定時(shí)任務(wù)的原理及動(dòng)態(tài)創(chuàng)建詳解
這篇文章主要給大家介紹了關(guān)于Spring boot定時(shí)任務(wù)的原理及動(dòng)態(tài)創(chuàng)建的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03
SpringBoot基于Redis實(shí)現(xiàn)生成全局唯一ID的方法
在項(xiàng)目中生成全局唯一ID有很多好處,生成全局唯一ID有助于提高系統(tǒng)的可用性、數(shù)據(jù)的完整性和安全性,同時(shí)也方便數(shù)據(jù)的管理和分析,所以本文給大家介紹了SpringBoot基于Redis實(shí)現(xiàn)生成全局唯一ID的方法,文中有詳細(xì)的代碼講解,需要的朋友可以參考下2023-12-12
java中Memcached的使用實(shí)例(包括與Spring整合)
這篇文章主要介紹了java中Memcached的使用實(shí)例(包括與Spring整合),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
SpringBoot使用Sharding-JDBC實(shí)現(xiàn)數(shù)據(jù)分片和讀寫分離的方法
本文主要介紹了SpringBoot使用Sharding-JDBC實(shí)現(xiàn)數(shù)據(jù)分片和讀寫分離,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-10-10
Java數(shù)據(jù)結(jié)構(gòu)之HashMap源碼深入分析
Java HashMap是一種基于哈希表實(shí)現(xiàn)的鍵值對(duì)存儲(chǔ)結(jié)構(gòu),可以實(shí)現(xiàn)快速的數(shù)據(jù)查找和存儲(chǔ)。它是線程不安全的,但在單線程環(huán)境中運(yùn)行效率高,被廣泛應(yīng)用于Java開發(fā)中2023-04-04
一文搞懂并學(xué)會(huì)使用SpringBoot的Actuator運(yùn)行狀態(tài)監(jiān)控組件的詳細(xì)教程
這篇文章主要介紹了一文搞懂并學(xué)會(huì)使用SpringBoot的Actuator運(yùn)行狀態(tài)監(jiān)控組件,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09

