# i++和++i的區(qū)別
```java
int i = 1;
int j1 = i++; // 先將i的原始值(1)賦值給變量j1(1),然后i變量的值加1
int j1 = ++i; // 先將i變量的值加1,然后將i的當(dāng)前值(2)賦值給變量j1(2)
```
# 實(shí)現(xiàn)原理
源代碼如下:
```java
public class Test {
public void testIPlus() {
int i = 0;
int j = i++;
}
public void testPlusI() {
int i = 0;
int j = ++i;
}
}
```
經(jīng)`javap`反編譯后查看字節(jié)碼如下:
```java
{
public Test();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
LineNumberTable:
line 1: 0
public void testIPlus();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=1, locals=3, args_size=1
0: iconst_0 // 生成整數(shù)0
1: istore_1 // 將整數(shù)0賦值給1號(hào)存儲(chǔ)單元(即變量i)
2: iload_1 // 將1號(hào)存儲(chǔ)單元的值加載到數(shù)據(jù)棧(此時(shí) i=0,棧頂值為0)
3: iinc 1, 1 // 1號(hào)存儲(chǔ)單元的值+1(此時(shí) i=1)
6: istore_2 // 將數(shù)據(jù)棧頂?shù)闹担?)取出來賦值給2號(hào)存儲(chǔ)單元(即變量j,此時(shí)i=1,j=0)
7: return // 返回時(shí):i=1,j=0
LineNumberTable:
line 4: 0
line 5: 2
line 6: 7
public void testPlusI();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=1, locals=3, args_size=1
0: iconst_0 // 生成整數(shù)0
1: istore_1 // 將整數(shù)0賦值給1號(hào)存儲(chǔ)單元(即變量i)
2: iinc 1, 1 // 1號(hào)存儲(chǔ)單元的值+1(此時(shí) i=1)
5: iload_1 // 將1號(hào)存儲(chǔ)單元的值加載到數(shù)據(jù)棧(此時(shí) i=1,棧頂值為1)
6: istore_2 // 將數(shù)據(jù)棧頂?shù)闹担?)取出來賦值給2號(hào)存儲(chǔ)單元(即變量j,此時(shí)i=1,j=1)
7: return // 返回時(shí):i=1,j=1
LineNumberTable:
line 9: 0
line 10: 2
line 11: 7
}
```
可以看到,區(qū)別在于第三步和第四部的順序問題:
* i++先執(zhí)行`iload_1`,再執(zhí)行`iinc`,也就是先將1號(hào)存儲(chǔ)單元的值加載到數(shù)據(jù)棧,再將1號(hào)存儲(chǔ)單元的值+1
* ++i先執(zhí)行`iinc`,再執(zhí)行`iload_1`,也就先將1號(hào)存儲(chǔ)單元的值+1,再將1號(hào)存儲(chǔ)單元的值加載到數(shù)據(jù)棧
執(zhí)行完這兩步后,將數(shù)據(jù)棧頂?shù)闹等〕鲑x值給2號(hào)存儲(chǔ)單元,也就是賦值給j。
# 實(shí)戰(zhàn)
那么看看接下來的兩個(gè)例子會(huì)輸出什么呢
代碼1:
```java
public static void main(String[] args) {
int a = 0;
for (int i = 0; i < 99; i++) {
a = a++;
}
System.out.print(a);
}
```
代碼2:
```java
public static void main(String[] args) {
int a = 0;
for (int i = 0; i < 99; i++) {
a = ++a;
}
System.out.print(a);
}
```
在代碼`a = a++`處IDE會(huì)提示`The value changed at 'a++' is never used `,也就是說a加1后的值從來沒有被使用到。
那么我們來看看什么兩個(gè)代碼片段的執(zhí)行結(jié)果:
代碼1:
```plain
0
Process finished with exit code 0
```
代碼2:
```plain
99
Process finished with exit code 0
```
*****
接下來我們從字節(jié)碼角度看看為什么會(huì)這樣,編寫一個(gè)簡(jiǎn)單的代碼片段:
```java
int i = 0;
i = i++;
System.out.println("i=" + i); // 輸出 i=0
```
編譯后查看字節(jié)碼如下:
```java
0: iconst_0 // 生成整數(shù)0
1: istore_1 // 將整數(shù)0賦值給1號(hào)存儲(chǔ)單元(即變量i,i=0)
2: iload_1 // 將1號(hào)存儲(chǔ)單元的值加載到數(shù)據(jù)棧(此時(shí) i=0,棧頂值為0)
3: iinc 1, 1 // 1號(hào)存儲(chǔ)單元的值+1(此時(shí) i=1)
6: istore_1 // 將數(shù)據(jù)棧頂?shù)闹担?)取出來賦值給1號(hào)存儲(chǔ)單元(即變量i,此時(shí)i=0)
7: getstatic #16 // 下面是打印到控制臺(tái)指令
10: new #22
13: dup
14: ldc #24
16: invokespecial #26
19: iload_1
20: invokevirtual #29
23: invokevirtual #33
26: invokevirtual #37
29: return
```
可以看到,對(duì)于代碼`i = i++;`,在給1號(hào)存儲(chǔ)單元賦值后,先將1號(hào)存儲(chǔ)單元的值加載到數(shù)據(jù)棧,再將1號(hào)存儲(chǔ)單元的值+1,然后再**將數(shù)據(jù)棧頂?shù)闹担?)取出來賦值給1號(hào)存儲(chǔ)單元**。
因此在上面a++循環(huán)99次后,a的值依舊是0.
# 總結(jié)
* i++會(huì)先使用i的值,也就是將i的值加載到數(shù)據(jù)棧,在給i加1,最后使用數(shù)據(jù)棧中的值。
* ++i會(huì)先將i的值加1,再將增加后的值加載到數(shù)據(jù)棧,再使用數(shù)據(jù)棧中的值。
# 參考
[深入理解Java中的i++、++i語(yǔ)句](https://blog.csdn.net/xialei199023/article/details/76383013)
- 導(dǎo)讀
- Java知識(shí)
- Java基本程序設(shè)計(jì)結(jié)構(gòu)
- 【基礎(chǔ)知識(shí)】Java基礎(chǔ)
- 【源碼分析】Okio
- 【源碼分析】深入理解i++和++i
- 【專題分析】JVM與GC
- 【面試清單】Java基本程序設(shè)計(jì)結(jié)構(gòu)
- 對(duì)象與類
- 【基礎(chǔ)知識(shí)】對(duì)象與類
- 【專題分析】Java類加載過程
- 【面試清單】對(duì)象與類
- 泛型
- 【基礎(chǔ)知識(shí)】泛型
- 【面試清單】泛型
- 集合
- 【基礎(chǔ)知識(shí)】集合
- 【源碼分析】SparseArray
- 【面試清單】集合
- 多線程
- 【基礎(chǔ)知識(shí)】多線程
- 【源碼分析】ThreadPoolExecutor源碼分析
- 【專題分析】volatile關(guān)鍵字
- 【面試清單】多線程
- Java新特性
- 【專題分析】Lambda表達(dá)式
- 【專題分析】注解
- 【面試清單】Java新特性
- Effective Java筆記
- Android知識(shí)
- Activity
- 【基礎(chǔ)知識(shí)】Activity
- 【專題分析】運(yùn)行時(shí)權(quán)限
- 【專題分析】使用Intent打開三方應(yīng)用
- 【源碼分析】Activity的工作過程
- 【面試清單】Activity
- 架構(gòu)組件
- 【專題分析】MVC、MVP與MVVM
- 【專題分析】數(shù)據(jù)綁定
- 【面試清單】架構(gòu)組件
- 界面
- 【專題分析】自定義View
- 【專題分析】ImageView的ScaleType屬性
- 【專題分析】ConstraintLayout 使用
- 【專題分析】搞懂點(diǎn)九圖
- 【專題分析】Adapter
- 【源碼分析】LayoutInflater
- 【源碼分析】ViewStub
- 【源碼分析】View三大流程
- 【源碼分析】觸摸事件分發(fā)機(jī)制
- 【源碼分析】按鍵事件分發(fā)機(jī)制
- 【源碼分析】Android窗口機(jī)制
- 【面試清單】界面
- 動(dòng)畫和過渡
- 【基礎(chǔ)知識(shí)】動(dòng)畫和過渡
- 【面試清單】動(dòng)畫和過渡
- 圖片和圖形
- 【專題分析】圖片加載
- 【面試清單】圖片和圖形
- 后臺(tái)任務(wù)
- 應(yīng)用數(shù)據(jù)和文件
- 基于網(wǎng)絡(luò)的內(nèi)容
- 多線程與多進(jìn)程
- 【基礎(chǔ)知識(shí)】多線程與多進(jìn)程
- 【源碼分析】Handler
- 【源碼分析】AsyncTask
- 【專題分析】Service
- 【源碼分析】Parcelable
- 【專題分析】Binder
- 【源碼分析】Messenger
- 【面試清單】多線程與多進(jìn)程
- 應(yīng)用優(yōu)化
- 【專題分析】布局優(yōu)化
- 【專題分析】繪制優(yōu)化
- 【專題分析】?jī)?nèi)存優(yōu)化
- 【專題分析】啟動(dòng)優(yōu)化
- 【專題分析】電池優(yōu)化
- 【專題分析】包大小優(yōu)化
- 【面試清單】應(yīng)用優(yōu)化
- Android新特性
- 【專題分析】狀態(tài)欄、ActionBar和導(dǎo)航欄
- 【專題分析】應(yīng)用圖標(biāo)、通知欄適配
- 【專題分析】Android新版本重要變更
- 【專題分析】唯一標(biāo)識(shí)符的最佳做法
- 開源庫(kù)源碼分析
- 【源碼分析】BaseRecyclerViewAdapterHelper
- 【源碼分析】ButterKnife
- 【源碼分析】Dagger2
- 【源碼分析】EventBus3(一)
- 【源碼分析】EventBus3(二)
- 【源碼分析】Glide
- 【源碼分析】OkHttp
- 【源碼分析】Retrofit
- 其他知識(shí)
- Flutter
- 原生開發(fā)與跨平臺(tái)開發(fā)
- 整體歸納
- 狀態(tài)及狀態(tài)管理
- 零碎知識(shí)點(diǎn)
- 添加Flutter到現(xiàn)有應(yīng)用
- Git知識(shí)
- Git命令
- .gitignore文件
- 設(shè)計(jì)模式
- 創(chuàng)建型模式
- 結(jié)構(gòu)型模式
- 行為型模式
- RxJava
- 基礎(chǔ)
- Linux知識(shí)
- 環(huán)境變量
- Linux命令
- ADB命令
- 算法
- 常見數(shù)據(jù)結(jié)構(gòu)及實(shí)現(xiàn)
- 數(shù)組
- 排序算法
- 鏈表
- 二叉樹
- 棧和隊(duì)列
- 算法時(shí)間復(fù)雜度
- 常見算法思想
- 其他技術(shù)
- 正則表達(dá)式
- 編碼格式
- HTTP與HTTPS
- 【面試清單】其他知識(shí)
- 開發(fā)歸納
- Android零碎問題
- 其他零碎問題
- 開發(fā)思路
