自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

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

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

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

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中自增不是原子性操作,它包含一個讀操作和一個寫操作。

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

以下的代碼是用100個線程同時(shí)執(zhí)行自增操作,每個線程自增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é)碼是怎么操作的

以下是一個簡單的自增操作代碼

  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."<init>":()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                //將棧頂兩個int類型的元素相加,并將其值壓入棧頂  
  18. :   putfield        #2//Field id:I,將棧頂?shù)闹蒂x值給id  
  19. :  return 
  20.  
  21.     } 

很明顯,我們能夠看到在getNext()方法內(nèi)部,對于類變量id有一個先取值后加一再賦值的過程。因此,我們可以很肯定的說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."<init>":()V  
  6. :   return 
  7.  
  8. public void getNext();  
  9.   Code:  
  10. :   iconst_0  
  11. :   istore_1  
  12. :   iinc    11 
  13. :   return 
  14.  

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

原文鏈接:http://www.cnblogs.com/kevinwu/archive/2012/05/02/2479464.html

責(zé)任編輯:林師授 來源: wurr707的博客
相關(guān)推薦

2023-01-05 12:30:32

Redis

2022-08-21 17:35:31

原子多線程

2020-08-24 07:19:13

主鍵自增數(shù)據(jù)庫

2023-12-26 01:09:28

MySQL存儲釋放鎖

2021-09-22 12:56:19

編程技能Golang

2014-01-09 09:45:41

原子飛原子

2010-04-30 12:15:42

Oracle自增ID

2010-07-09 12:49:41

SQL Server自

2011-03-21 12:58:26

Oracle自增字段

2021-06-03 14:00:35

PolarDB

2021-06-02 16:30:33

PolarDB原子性數(shù)據(jù)庫

2010-04-26 11:55:48

Oracle自增字段

2010-09-28 16:14:14

SQL Server自

2010-04-09 09:28:30

Oracle自增字段

2009-09-24 13:49:31

Hibernate自增

2021-11-01 15:35:31

C++自增自減

2023-11-07 08:04:19

Go并發(fā)程序原子操作

2010-10-20 10:04:36

sql server自

2010-03-29 14:09:12

Oracle ID 自

2021-06-29 10:07:24

Javalong原子操作
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號