最近有小伙伴告訴我,在循環(huán)的判斷條件只會計算一次,本金魚不相信,于是就做了測試,本文記錄我做的測試。
先來寫一個簡單的代碼, 就一個循環(huán),循環(huán)的判斷是從一個函數(shù)獲取值
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
class program { static void main( string [] args) { var meepemorcear = new meepemorcear(); meepemorcear.birmerulerrayjairbay(); } } class meepemorcear { public void birmerulerrayjairbay() { for ( int i = 0; i < daydrearnenawerlai(); i++) { console.writeline( "第" +i.tostring()+ "個逗比" ); } } public int daydrearnenawerlai() { console.writeline( "進入" ); return 10; } } |
通過 main 調(diào)用 birmerulerrayjairbay ,這個函數(shù)里面的 for 判斷是 daydrearnenawerlai
拿到一個值,我嘗試在 release 運行,結(jié)果每個判斷都需要進入 daydrearnenawerlai
函數(shù),請看輸出
進入
第0個逗比
進入
第1個逗比
進入
第2個逗比
進入
第3個逗比
進入
第4個逗比
進入
第5個逗比
進入
第6個逗比
進入
第7個逗比
進入
第8個逗比
進入
第9個逗比
也就是在 debug 或 release 下,for 里面的判斷都是需要執(zhí)行,所以在 for 里的判斷如果寫了很長的計算,那么就會在每次循環(huán)都需要重新計算。即使每次計算出來的值都是一樣,也需要重新計算。
所以這樣看起來性能不如這樣寫,使用一個臨時的變量獲取判斷的值
1
2
3
4
5
6
7
8
|
public void birmerulerrayjairbay() { var mowraitepalor = daydrearnenawerlai(); for ( int i = 0; i < mowraitepalor; i++) { console.writeline( "第" +i.tostring()+ "個逗比" ); } } |
但是很快,另一個小伙伴告訴我,你把輸出去掉,然后使用斷點,你再看看
我添加了斷點,在斷點輸出 123 然后運行
這時我發(fā)現(xiàn)運行沒有輸出 123 也就是函數(shù)沒有進來,我再次添加斷點,跟著函數(shù)也沒有訪問
所以這時的 daydrearnenawerlai
函數(shù)就被優(yōu)化掉了
我和一個小伙伴說了這個問題,他說是被 il 優(yōu)化了,我一點不相信,所以我就去看 il 代碼
從下面的代碼
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public void birmerulerrayjairbay() { for ( int i = 0; i < daydrearnenawerlai(); i++) { console.writeline( "第" +i.tostring()+ "個逗比" ); } } /// <summary> /// 進入lindexi.github.io可以看到更多博客 /// </summary> /// <returns></returns> public static int daydrearnenawerlai() { return 10; } |
轉(zhuǎn) il 可以看到下面代碼,我會在 il 添加很多注釋,所以很容易看懂。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
|
.method public hidebysig instance void birmerulerrayjairbay() cil managed { .maxstack 3 .locals init ( [0] int32 i ) // 第 23 行 18 個字符到 27 個字符 // [23 18 - 23 27] il_0000: ldc.i4.0 // 定義 i ,代碼的 int i = 0; il_0001: stloc.0 // i il_0002: br.s il_0023 // 這里就是進入循環(huán) for ,在 il 不管 for 還是 while 都是差不多 // start of loop, entry point: il_0023 // [25 17 - 25 60] // 下面這個代碼就是 console.writeline("第" +i.tostring()+"個逗比"); 從代碼可以看到 // 需要先申請"第" il_0004: ldstr "第" // 然后把 i 放入棧 il_0009: ldloca.s i // 調(diào)用 int.tostring ,使用的是實例的方法 il_000b: call instance string [mscorlib]system.int32::tostring() // 把"個逗比"放入棧 il_0010: ldstr "個逗比" // 調(diào)用字符串組合方法,組合三個字符串,返回一個字符串。 // 把剛才入棧三個字符串出棧,返回的字符串入棧 il_0015: call string [mscorlib]system. string ::concat( string , string , string ) // 調(diào)用 console.writeline ,從棧拿到一個字符串輸出 il_001a: call void [mscorlib]system.console::writeline( string ) // 下面是 i++ 代碼 // [23 55 - 23 58] // 將指定索引處的局部變量加載到計算堆棧上,這里的索引是 0 ,在代碼的變量是 i 所以把 i 加載到計算堆棧 il_001f: ldloc.0 // i // 將整數(shù)值 1 作為 int32 推送到計算堆棧上 il_0020: ldc.i4.1 // 從堆棧出棧兩個數(shù)值進行相加,返回的值放在棧 il_0021: add // 從計算堆棧的頂部彈出當前值并將其存儲到指定索引處的局部變量列表中,這里索引是 0 ,在代碼的變量是 i ,所以 i = i + 1 的代碼就是這樣做 il_0022: stloc.0 // i // 從堆棧加載 i ,把 i 入棧 // [23 29 - 23 53] il_0023: ldloc.0 // i // 調(diào)用方法 daydrearnenawerlai 拿到返回值 il_0024: call int32 muhoubearwhedoofi.meepemorcear::daydrearnenawerlai() // 如果第一個值小于第二個值,則將控制轉(zhuǎn)移到目標指令,這里的第一個值就是 i ,第二個值就是 daydrearnenawerlai 的返回值。跳轉(zhuǎn)到標簽 il_0004 ,如果沒有小于,就繼續(xù)代碼。 il_0029: blt.s il_0004 // end of loop // [27 9 - 27 10] il_002b: ret } // end of method meepemorcear::birmerulerrayjairbay .method public hidebysig static int32 daydrearnenawerlai() cil managed { .maxstack 8 // 把一個值 放入堆棧,放入的是 10 ,然后從棧拿到值返回 // [36 13 - 36 23] il_0000: ldc.i4.s 10 // 0x0a il_0002: ret } // end of method meepemorcear::daydrearnenawerlai |
從上面代碼可以發(fā)現(xiàn),實際 daydrearnenawerlai 沒有被優(yōu)化掉,還是有這個方法。
總結(jié)
以上所述是小編給大家介紹的c# 循環(huán)判斷會進來幾次的實現(xiàn)代碼,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對服務器之家網(wǎng)站的支持!
原文鏈接:https://lindexi.gitee.io/lindexi/post/C-循環(huán)的判斷會進來幾次.html