av激情亚洲男人的天堂国语,日韩欧美精品一中文字幕,无码av一区二区三区无码,国产又色又爽又刺激的a片,国产又色又爽又刺激的a片

關(guān)于Java自增操作的原子性

最近在工作中和一個(gè)同事因?yàn)樽栽鍪遣皇窃有圆僮鳡幷摰拿婕t耳赤,那Java的自增操作到底是不是原子性操作呢,答案是否的,即Java的自增操作不是原子性操作。

站在用戶的角度思考問題,與客戶深入溝通,找到邗江網(wǎng)站設(shè)計(jì)與邗江網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗(yàn),讓設(shè)計(jì)與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個(gè)性化、用戶體驗(yàn)好的作品,建站類型包括:做網(wǎng)站、成都做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣、域名申請雅安服務(wù)器托管、企業(yè)郵箱。業(yè)務(wù)覆蓋邗江地區(qū)。

1.首先我們先看看Bruce Eckel是怎么說的:

In the JVM an increment is not atomic and involves both a read and a write. (via the latest Java Performance Tuning Newsletter)

意思很簡單,就是說在jvm中自增不是原子性操作,它包含一個(gè)讀操作和一個(gè)寫操作。

2.以上可能還不能讓你信服,要想讓人心服口服,就必須用代碼說話。正如FaceBook的文化一樣:代碼贏得爭論。那我們就看一段代碼:

以下的代碼是用100個(gè)線程同時(shí)執(zhí)行自增操作,每個(gè)線程自增100次,如果自增操作是原子性操作的話,那么執(zhí)行完amount的值為10,000。運(yùn)行代碼之后,你會發(fā)現(xiàn)amount的值小于10,000,這就說明自增操作不是原子性的

 
 
 
  1. /**  
  2.  *   
  3.  * @author renrun.wu  
  4.  */ 
  5. public class MultiThread implements Runnable {  
  6.     private int count;  
  7.     private int amount = 1;  
  8.       
  9.     public MultiThread() {  
  10.          count = 100;  
  11.     }  
  12.       
  13.     public MultiThread(int count) {  
  14.         this.count = count;  
  15.     }  
  16.       
  17.     @Override 
  18.     public void run() {  
  19.         for (int i = 0; i < count; i++) {  
  20.             amount++;  
  21.         }  
  22.     }  
  23.       
  24.     public static void main(String[] args) {  
  25.         ExecutorService executorService = Executors.newCachedThreadPool();  
  26.         MultiThread multiThread =new MultiThread();  
  27.         for (int i = 0; i < 100; i++) {  
  28.             executorService.execute(multiThread);  
  29.         }  
  30.         executorService.shutdown();  
  31.           
  32.         try {  
  33.             Thread.sleep(60000);  
  34.         } catch (InterruptedException e) {  
  35.             e.printStackTrace();  
  36.         }  
  37.         System.out.println(multiThread.amount);  
  38.     }  

3.如果以上還不能讓你信服的話,也沒關(guān)系。我們就把自增操作反編譯出來,看看java字節(jié)碼是怎么操作的

以下是一個(gè)簡單的自增操作代碼

 
 
 
  1. public class Increment {  
  2.     private int id = 0;  
  3.  
  4.     public void getNext(){  
  5.         id++;  
  6.     }  

我們看看反編譯之后的Java字節(jié)碼,主要關(guān)注getNext()方法內(nèi)部的Java字節(jié)碼。

 
 
 
  1. public class Increment extends java.lang.Object{  
  2.     public Increment();  
  3.       Code:  
  4. :   aload_0  
  5. :   invokespecial   #1; //Method java/lang/Object."":()V  
  6. :   aload_0  
  7. :   iconst_0  
  8. :   putfield        #2; //Field id:I  
  9. :   return 
  10.  
  11.     public void getNext();  
  12.       Code:  
  13. :   aload_0         //加載局部變量表index為0的變量,在這里是this   
  14. :   dup                 //將當(dāng)前棧頂?shù)膶ο笠脧?fù)制一份  
  15. :   getfield        #2; //Field id:I,獲取id的值,并將其值壓入棧頂  
  16. :   iconst_1            //將int型的值1壓入棧頂  
  17. :   iadd                //將棧頂兩個(gè)int類型的元素相加,并將其值壓入棧頂  
  18. :   putfield        #2; //Field id:I,將棧頂?shù)闹蒂x值給id  
  19. :  return 
  20.  
  21.     } 

很明顯,我們能夠看到在getNext()方法內(nèi)部,對于類變量id有一個(gè)先取值后加一再賦值的過程。因此,我們可以很肯定的說Java中的自增操作不是原子性的。

4.也許你會問,那局部變量的自增操作是否是原子性的。好,我們在看看一下代碼:

 
 
 
  1. public class Increment {  
  2.     public void getNext(){  
  3.     int id = 0;  
  4.         id++;  
  5.     }  

我們再看看反編譯之后的Java字節(jié)碼,主要還是關(guān)注getNext()方法內(nèi)部的Java字節(jié)碼。

 
 
 
  1. public class Increment extends java.lang.Object{  
  2. public Increment();  
  3.   Code:  
  4. :   aload_0  
  5. :   invokespecial   #1; //Method java/lang/Object."":()V  
  6. :   return 
  7.  
  8. public void getNext();  
  9.   Code:  
  10. :   iconst_0  
  11. :   istore_1  
  12. :   iinc    1, 1 
  13. :   return 
  14.  

與全局變量的自增操作相比,很明顯局部變量的自增操作少了getfield與putfield操作。而且對于局部變量來說,它無論如何都不會涉及到多線程的操作,因此局部變量的自增操作是否是原子操作也就顯得不那么重要了。


文章名稱:關(guān)于Java自增操作的原子性
文章出自:http://uogjgqi.cn/article/djceesj.html
掃二維碼與項(xiàng)目經(jīng)理溝通

我們在微信上24小時(shí)期待你的聲音

解答本文疑問/技術(shù)咨詢/運(yùn)營咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流