C#編譯器對(duì)局部變量的優(yōu)化指南
前言
C# 的編譯器可以對(duì)代碼進(jìn)行優(yōu)化,所以,我們?cè)趯懘a的時(shí)候,可以更多地考慮一下代碼的易讀性問題。
不考慮基本的對(duì)齊和換行美化。看一下局部變量?jī)?yōu)化問題。
C# 示例代碼
例如,我們有一段如下的代碼:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace ConsoleApp1
{
class Program
{
static void Main(string[] args)
{
var s = DoSomething();
Console.WriteLine(s);
}
static string DoSomething()
{
var s1 = "Hello, world.";
var s2 = s1.ToUpper();
return s2;
}
}
}
在 DoSomething() 這個(gè)方法中,里面定義了兩個(gè)局部變量:
- s1
- s2
在 Main() 方法中,定義了一個(gè)局部變量:
- s
定義 s1 和 s2 是為了提高代碼的可讀性,它們會(huì)導(dǎo)致生成冗余的代碼,降低執(zhí)行效率嗎?
我們分別在 Debug 模式下和 Release 模式下進(jìn)行編譯,使用 ILDasm 查看生成的中間代碼。
Debug 模式下生成的中間代碼
在 Debug 下編譯之后,DoSomething() 生成的中間代碼如下,可以看到實(shí)際上有 3 個(gè)局部變量。除了我們自己定義的 s1 和 s2 之外,還有一個(gè)生成的 V_2,代碼的尺寸為 20。
.method private hidebysig static string DoSomething() cil managed
{
// Code size 20 (0x14)
.maxstack 1
.locals init ([0] string s1,
[1] string s2,
[2] string V_2)
IL_0000: nop
IL_0001: ldstr "Hello, world."
IL_0006: stloc.0
IL_0007: ldloc.0
IL_0008: callvirt instance string [mscorlib]System.String::ToUpper()
IL_000d: stloc.1
IL_000e: ldloc.1
IL_000f: stloc.2
IL_0010: br.s IL_0012
IL_0012: ldloc.2
IL_0013: ret
} // end of method Program::DoSomething
看一下 Main() 方法。
有我們定義的 s 這一個(gè)局部變量,代碼尺寸為 15 個(gè)字節(jié)。
.method private hidebysig static void Main(string[] args) cil managed
{
.entrypoint
// Code size 15 (0xf)
.maxstack 1
.locals init ([0] string s)
IL_0000: nop
IL_0001: call string ConsoleApp1.Program::DoSomething()
IL_0006: stloc.0
IL_0007: ldloc.0
IL_0008: call void [mscorlib]System.Console::WriteLine(string)
IL_000d: nop
IL_000e: ret
} // end of method Program::Main
Release 模式下生成的中間代碼
而在 Release 模式下,實(shí)際上,DoSomething() 中所有的局部變量都被優(yōu)化掉了。代碼尺寸也只有 11 個(gè)字節(jié)。
.method private hidebysig static string DoSomething() cil managed
{
// Code size 11 (0xb)
.maxstack 8
IL_0000: ldstr "Hello, world."
IL_0005: callvirt instance string [mscorlib]System.String::ToUpper()
IL_000a: ret
} // end of method Program::DoSomething
還可以看一下 Main() 方法,這個(gè)局部變量 s 也被優(yōu)化掉了。代碼尺寸也只有 11 字節(jié)了。
.method private hidebysig static void Main(string[] args) cil managed
{
.entrypoint
// Code size 11 (0xb)
.maxstack 8
IL_0000: call string ConsoleApp1.Program::DoSomething()
IL_0005: call void [mscorlib]System.Console::WriteLine(string)
IL_000a: ret
} // end of method Program::Main
結(jié)論
編譯器會(huì)盡可能對(duì)代碼進(jìn)行優(yōu)化,我們可以為了提高代碼的易讀性增加一些局部變量,這并不會(huì)導(dǎo)致生成冗余代碼并導(dǎo)致執(zhí)行性能的下降。
到此這篇關(guān)于C#編譯器對(duì)局部變量?jī)?yōu)化的文章就介紹到這了,更多相關(guān)C#編譯器對(duì)局部變量?jī)?yōu)化內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C#實(shí)現(xiàn)復(fù)制數(shù)據(jù)庫(kù) C#將A數(shù)據(jù)庫(kù)數(shù)據(jù)轉(zhuǎn)到B數(shù)據(jù)庫(kù)
這篇文章主要為大家詳細(xì)介紹了C#復(fù)制數(shù)據(jù)庫(kù),將數(shù)據(jù)庫(kù)數(shù)據(jù)轉(zhuǎn)到另一個(gè)數(shù)據(jù)庫(kù),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11
C#中泛型容器Stack<T>的用法并實(shí)現(xiàn)”撤銷/重做”功能
這篇文章介紹了C#中泛型容器Stack<T>的用法并實(shí)現(xiàn)”撤銷/重做”功能,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-10-10
c# StringBuilder.Replace 方法 (Char, Char, Int32, Int32)
c# StringBuilder.Replace 方法 (Char, Char, Int32, Int32)...2007-08-08
Unity實(shí)現(xiàn)移動(dòng)端手勢(shì)解鎖功能
這篇文章主要為大家詳細(xì)介紹了Unity實(shí)現(xiàn)移動(dòng)端手勢(shì)解鎖功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-07-07
C#調(diào)用OpenCV開發(fā)簡(jiǎn)易版美圖工具【推薦】
本文主要介紹在WPF項(xiàng)目中使用OpenCVSharp3-AnyCPU開源類庫(kù)處理圖片,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2019-10-10
C#用表達(dá)式樹構(gòu)建動(dòng)態(tài)查詢的方法
這篇文章主要介紹了C#用表達(dá)式樹構(gòu)建動(dòng)態(tài)查詢的方法,幫助大家更好的理解和學(xué)習(xí)c#,感興趣的朋友可以了解下2020-12-12
運(yùn)用示例簡(jiǎn)單講解C#取消令牌CancellationTokenSource
這篇文章運(yùn)用示例簡(jiǎn)單講解C#取消令牌CancellationTokenSource,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08

