您好,登錄后才能下訂單哦!
這篇文章給大家分享的是有關Java線程并發訪問的示例分析的內容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。
class ConcurrentThread { /** * 分析線程并發訪問代碼解釋原因 * volatile關鍵字: * 1):保證了不同線程對這個變量進行操作時的可見性,即一個線程修改了某個變量的值,這新值對其他線程來說是立即可見的 * 2):禁止進行指令重排序 * volatile本質是告訴JVM當前變量在寄存器(工作內存)中的值是不確定的,需要從主存中讀取 */ private volatile int count = 0; public void inc() { try { Thread.sleep(3); } catch (InterruptedException e) { e.printStackTrace(); } count++; } @Override public String toString() { return "[count=" + count + "]"; } } //---------------------------------------- public class VolatileTest { public static void main(String[] args) { final ConcurrentThread counter = new ConcurrentThread(); for (int i = 0; i < 1000; i++) { new Thread(new Runnable() { @Override public void run() { counter.inc(); } }).start(); } System.out.println(counter); } }
原因解釋:在Java的內存模型中每一個線程運行時都有一個線程棧,線程棧保存了線程運行時候變量值信息
當線程訪問某一個對象值得時候:
1、通過對象的引用找到對應在堆內存的變量的值
2、把堆內存變量的具體值load到線程本地內存中,建立了一個變量副本,之后線程就不再和對象在堆內存變量值有任何關系,而是直接修改副本變量的值,在修改完之后的某一個時刻(線程退出之前),自動把線程變量副本的值回寫到對象在堆中變量。這樣堆中的對象的值就產生變化了。
結合上例,也就是說上面主函數中開啟了1000個子線程,每個線程都有一個變量副本,每個線程修改變量只是臨時修改了自己的副本,當線程結束時再將修改的值寫入在主內存中,這樣就出現了線程安全問題(拿到了主內存中過時的變量值),因此結果就不可能等于1000了,一般都會小于1000.
感謝各位的閱讀!關于“Java線程并發訪問的示例分析”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。