通過volatile驗證線程之間的可見性
這篇文章主要介紹了通過volatile驗證線程之間的可見性,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
由于JVM運行程序的實體是線程,而每個線程創(chuàng)建時JVM都會為其創(chuàng)建一個工作內存(有些地方稱為??臻g),工作內存是每個線程的私有數據區(qū)域,而Java內存 模型中規(guī)定所有變量都存儲在主內存,主內存是共享內存區(qū)域,所有線程都可以訪問,但線程對變量的操作(讀取賦值等)必須在工作內存中進行,首先要將變量從主內存拷貝的自己的工作內存空間,然后對變量進行操作,操作完成后再將變量寫回主內存,不能直接操作主內存中的變量,各個線程中的工作內存中存儲著主內存中的變量副木拷貝,因此不同的線程間無法訪問對方的工作內存,線程間的通信(傳值)必須通過主內存來完成,其簡要訪問過程如下圖:
代碼驗證如下:
import java.util.concurrent.TimeUnit; class MyData{ //定義初始參數,volatile關鍵字使用為重點 volatile int number = 0; //更改初始參數 public void changeNumber(){ this.number = 60; } } public class VolatileDemo { public static void main(String[] args) { //初始化資源類 MyData myData = new MyData(); new Thread(() -> { System.out.println(Thread.currentThread().getName()+"初始化線程,初始參數為:"+myData.number); //AAA線程暫停三秒后 try{ TimeUnit.SECONDS.sleep(3); }catch(Exception e){ e.printStackTrace(); } //AAA線程更改初始化參數為60 myData.changeNumber(); System.out.println(Thread.currentThread().getName()+"線程更改初始化參數為:"+myData.number); },"AAA").start(); while(myData.number == 0){ //第二個線程是main線程,在number=0時則一直陷入死循環(huán),當number=60時則打印主線程語句。 } System.out.println(Thread.currentThread().getName()+"主線程當前參數:"+myData.number); } }
結果展示:
1.初始參數number未加上volatile:
2.初始參數number加上volatile:
總結:
volatile是JAVA虛擬機提供的輕量級的同步機制,volatile三大特征:保證可見性,不保證原子性,禁止指令重排。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
Spring?BeanFactory?與?FactoryBean?的區(qū)別詳情
這篇文章主要介紹了Spring?BeanFactory?與?FactoryBean?的區(qū)別詳情,BeanFactory?和?FactoryBean?的區(qū)別卻是一個很重要的知識點,在本文中將結合源碼進行分析講解,需要的小伙伴可以參考一下2022-05-05詳解spring boot starter redis配置文件
spring-boot-starter-Redis主要是通過配置RedisConnectionFactory中的相關參數去實現連接redis service。下面通過本文給大家介紹在spring boot的配置文件中redis的基本配置,需要的的朋友參考下2017-07-07