Java多線(xiàn)程 volatile關(guān)鍵字詳解
volatile
volatile是一種輕量同步機(jī)制。請(qǐng)看例子
MyThread25類(lèi)
public class MyThread25 extends Thread{
private boolean isRunning = true;
public boolean isRunning()
{
return isRunning;
}
public void setRunning(boolean isRunning)
{
this.isRunning = isRunning;
}
public void run()
{
System.out.println("進(jìn)入run了");
while (isRunning == true){}
System.out.println("線(xiàn)程被停止了");
}
public static void main(String[] args) throws InterruptedException {
MyThread25 mt = new MyThread25();
mt.start();
Thread.sleep(1000);
mt.setRunning(false);
System.out.println("已設(shè)置為false");
}
}
輸出結(jié)果如下
進(jìn)入run了 已設(shè)置為false
為什么程序始終不結(jié)束?說(shuō)明mt.setRunning(false);沒(méi)有起作用。
這里我們說(shuō)下Java內(nèi)存模型(JMM)
java虛擬機(jī)有自己的內(nèi)存模型(Java Memory Model,JMM),JMM可以屏蔽掉各種硬件和操作系統(tǒng)的內(nèi)存訪(fǎng)問(wèn)差異,以實(shí)現(xiàn)讓java程序在各種平臺(tái)下都能達(dá)到一致的內(nèi)存訪(fǎng)問(wèn)效果。
JMM定義了線(xiàn)程和主內(nèi)存之間的抽象關(guān)系:共享變量存儲(chǔ)在主內(nèi)存(Main Memory)中,每個(gè)線(xiàn)程都有一個(gè)私有的本地內(nèi)存(Local Memory),本地內(nèi)存保存了被該線(xiàn)程使用到的主內(nèi)存的副本,線(xiàn)程對(duì)變量的所有操作都必須在本地內(nèi)存中進(jìn)行,而不能直接讀寫(xiě)主內(nèi)存中的變量。這三者之間的交互關(guān)系如下

出現(xiàn)上述運(yùn)行結(jié)果的原因是,主內(nèi)存isRunning = true, mt.setRunning(false)設(shè)置主內(nèi)存isRunning = false,本地內(nèi)存中isRunning仍然是true,線(xiàn)程用的是本地內(nèi)存,所以進(jìn)入了死循環(huán)。
在isRunning前加上volatile
private volatile boolean isRunning = true;
輸出結(jié)果如下
進(jìn)入run了 已設(shè)置為false 線(xiàn)程被停止了
volatile不能保證原子類(lèi)線(xiàn)程安全
先看例子
MyThread26_0類(lèi),用volatile修飾num
public class MyThread26_0 extends Thread {
public static volatile int num = 0;
//使用CountDownLatch來(lái)等待計(jì)算線(xiàn)程執(zhí)行完
static CountDownLatch countDownLatch = new CountDownLatch(30);
@Override
public void run() {
for(int j=0;j<1000;j++){
num++;//自加操作
}
countDownLatch.countDown();
}
public static void main(String[] args) throws InterruptedException {
MyThread26_0[] mt = new MyThread26_0[30];
//開(kāi)啟30個(gè)線(xiàn)程進(jìn)行累加操作
for(int i=0;i<mt.length;i++){
mt[i] = new MyThread26_0();
}
for(int i=0;i<mt.length;i++){
mt[i].start();
}
//等待計(jì)算線(xiàn)程執(zhí)行完
countDownLatch.await();
System.out.println(num);
}
}
輸出結(jié)果如下
25886
理論上,應(yīng)該輸出30000。原子操作表示一段操作是不可分割的,因?yàn)閚um++不是原子操作,這樣會(huì)出現(xiàn)線(xiàn)程對(duì)過(guò)期的num進(jìn)行自增,此時(shí)其他線(xiàn)程已經(jīng)對(duì)num進(jìn)行了自增。
num++分三步:讀取、加一、賦值。
結(jié)論:
volatile只會(huì)對(duì)單個(gè)的的變量讀寫(xiě)具有原子性,像num++這種復(fù)合操作volatile是無(wú)法保證其原子性的
解決方法:
用原子類(lèi)AtomicInteger的incrementAndGet方法自增
public class MyThread26_1 extends Thread {
//使用原子操作類(lèi)
public static AtomicInteger num = new AtomicInteger(0);
//使用CountDownLatch來(lái)等待計(jì)算線(xiàn)程執(zhí)行完
static CountDownLatch countDownLatch = new CountDownLatch(30);
@Override
public void run() {
for(int j=0;j<1000;j++){
num.incrementAndGet();//原子性的num++,通過(guò)循環(huán)CAS方式
}
countDownLatch.countDown();
}
public static void main(String []args) throws InterruptedException {
MyThread26_1[] mt = new MyThread26_1[30];
//開(kāi)啟30個(gè)線(xiàn)程進(jìn)行累加操作
for(int i=0;i<mt.length;i++){
mt[i] = new MyThread26_1();
}
for(int i=0;i<mt.length;i++){
mt[i].start();
}
//等待計(jì)算線(xiàn)程執(zhí)行完
countDownLatch.await();
System.out.println(num);
}
}
輸出結(jié)果如下
30000
原子類(lèi)方法組合使用線(xiàn)程不安全
例子如下
ThreadDomain27類(lèi)
public class ThreadDomain27 {
public static AtomicInteger aiRef = new AtomicInteger();
public void addNum()
{
System.out.println(Thread.currentThread().getName() + "加了100之后的結(jié)果:" + aiRef.addAndGet(100));
aiRef.getAndAdd(1);
}
}
MyThread27類(lèi)
public class MyThread27 extends Thread{
private ThreadDomain27 td;
public MyThread27(ThreadDomain27 td)
{
this.td = td;
}
public void run()
{
td.addNum();
}
public static void main(String[] args)
{
try
{
ThreadDomain27 td = new ThreadDomain27();
MyThread27[] mt = new MyThread27[5];
for (int i = 0; i < mt.length; i++)
{
mt[i] = new MyThread27(td);
}
for (int i = 0; i < mt.length; i++)
{
mt[i].start();
}
Thread.sleep(1000);
System.out.println(ThreadDomain27.aiRef.get());
}
catch (InterruptedException e)
{
e.printStackTrace();
}
}
}
輸出結(jié)果如下
Thread-2加了100之后的結(jié)果:100 Thread-3加了100之后的結(jié)果:200 Thread-0加了100之后的結(jié)果:302 Thread-1加了100之后的結(jié)果:403 Thread-4加了100之后的結(jié)果:504 505
理想的輸出結(jié)果是100,201,302...,因?yàn)閍ddAndGet方法和getAndAdd方法構(gòu)成的addNum不是原子操作。
解決該問(wèn)題只需要在addNum加上synchronized關(guān)鍵字。
輸出結(jié)果如下
Thread-1加了100之后的結(jié)果:100 Thread-0加了100之后的結(jié)果:201 Thread-2加了100之后的結(jié)果:302 Thread-3加了100之后的結(jié)果:403 Thread-4加了100之后的結(jié)果:504 505
結(jié)論:
volatile解決的是變量在多個(gè)線(xiàn)程之間的可見(jiàn)性,但是無(wú)法保證原子性。
synchronized不僅保障了原子性外,也保障了可見(jiàn)性。
volatile和synchronized比較
先看實(shí)例,使用volatile是什么效果
CountDownLatch保證10個(gè)線(xiàn)程都能執(zhí)行完成,當(dāng)然你也可以在System.out.println(test.inc);之前使用Thread.sleep(xxx)
public class MyThread28 {
//使用CountDownLatch來(lái)等待計(jì)算線(xiàn)程執(zhí)行完
static CountDownLatch countDownLatch = new CountDownLatch(10);
public volatile int inc = 0;
public void increase() {
inc++;
}
public static synchronized void main(String[] args) throws InterruptedException {
final MyThread28 test = new MyThread28();
for(int i=0;i<10;i++){
new Thread(){
public void run() {
for(int j=0;j<1000;j++)
test.increase();
countDownLatch.countDown();
}
}.start();
}
countDownLatch.await();
System.out.println(test.inc);
}
}
運(yùn)行結(jié)果如下
9677
每次運(yùn)行結(jié)果都不一致。剛才我已經(jīng)解釋過(guò),這里我再解釋一遍。
使用volatile修飾int型變量i,多個(gè)線(xiàn)程同時(shí)進(jìn)行i++操作。比如有兩個(gè)線(xiàn)程A和B對(duì)volatile修飾的i進(jìn)行i++操作,i的初始值是0,A線(xiàn)程執(zhí)行i++時(shí)從本地內(nèi)存剛讀取了i的值0(i++不是原子操作),就切換到B線(xiàn)程了,B線(xiàn)程從本地內(nèi)存中讀取i的值也為0,然后就切換到A線(xiàn)程繼續(xù)執(zhí)行i++操作,完成后i就為1了,接著切換到B線(xiàn)程,因?yàn)橹耙呀?jīng)讀取過(guò)了,所以繼續(xù)執(zhí)行i++操作,最后的結(jié)果i就為1了。同理可以解釋為什么每次運(yùn)行結(jié)果都是小于10000的數(shù)字。
解決方法:
使用synchronized關(guān)鍵字
public class MyThread28 {
//使用CountDownLatch來(lái)等待計(jì)算線(xiàn)程執(zhí)行完
static CountDownLatch countDownLatch = new CountDownLatch(10);
public int inc = 0;
public synchronized void increase() {
inc++;
}
public static synchronized void main(String[] args) throws InterruptedException {
final MyThread28 test = new MyThread28();
for(int i=0;i<10;i++){
new Thread(){
public void run() {
for(int j=0;j<1000;j++)
test.increase();
countDownLatch.countDown();
}
}.start();
}
countDownLatch.await();
System.out.println(test.inc);
}
}
輸出結(jié)果如下
10000
synchronized不管是否是原子操作,它能保證同一時(shí)刻只有一個(gè)線(xiàn)程獲取鎖執(zhí)行同步代碼,會(huì)阻塞其他線(xiàn)程。
結(jié)論:
- volatile只能用在變量,synchronized可以在變量、方法上使用。
- volatile不會(huì)造成線(xiàn)程阻塞,synchronized會(huì)造成線(xiàn)程阻塞。
- volatile效率比synchronized高。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot參數(shù)校驗(yàn)之@Validated的使用詳解
這篇文章主要通過(guò)示例為大家詳細(xì)介紹一下介紹了SpringBoot參數(shù)校驗(yàn)中@Validated的使用方法,文中的示例代碼講解詳細(xì),需要的可以參考一下2022-06-06
SpringBoot淺析安全管理之基于數(shù)據(jù)庫(kù)認(rèn)證
在真實(shí)的項(xiàng)目中,用戶(hù)的基本信息以及角色等都存儲(chǔ)在數(shù)據(jù)庫(kù)中,因此需要從數(shù)據(jù)庫(kù)中獲取數(shù)據(jù)進(jìn)行認(rèn)證和授權(quán)2022-08-08
親測(cè)SpringBoot參數(shù)傳遞及@RequestBody注解---踩過(guò)的坑及解決
這篇文章主要介紹了親測(cè)SpringBoot參數(shù)傳遞及@RequestBody注解---踩過(guò)的坑及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
解析本地方法映射Java層的數(shù)據(jù)類(lèi)型
這篇文章給大家介紹了本地方法映射Java層的數(shù)據(jù)類(lèi)型,包括基礎(chǔ)類(lèi)型映射,引用類(lèi)型映射等等,對(duì)java層數(shù)據(jù)類(lèi)型映射相關(guān)知識(shí),感興趣的朋友跟隨腳本之家小編一起看看吧2018-03-03
springboot3請(qǐng)求參數(shù)種類(lèi)及接口測(cè)試案例小結(jié)
這篇文章主要介紹了springboot3請(qǐng)求參數(shù)種類(lèi)及接口測(cè)試案例小結(jié),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2023-10-10
聊聊springboot2.2.3升級(jí)到2.4.0單元測(cè)試的區(qū)別
這篇文章主要介紹了springboot 2.2.3 升級(jí)到 2.4.0單元測(cè)試的區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
java實(shí)現(xiàn)簡(jiǎn)單美女拼圖游戲
這篇文章主要介紹了java實(shí)現(xiàn)簡(jiǎn)單美女拼圖游戲的相關(guān)資料,需要的朋友可以參考下2015-03-03
Java日常練習(xí)題,每天進(jìn)步一點(diǎn)點(diǎn)(61)
下面小編就為大家?guī)?lái)一篇Java基礎(chǔ)的幾道練習(xí)題(分享)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧,希望可以幫到你2021-08-08

