C#特性-對象集合初始化器介紹
C# 3.0為你提供了對象集合初始化器:
/// <summary> /// 圖書類 /// </summary> public class Book { /// <summary> /// 圖書名稱 /// </summary> public string Title { get; set; } /// <summary> /// 單價(jià) /// </summary> public float Price { get; set; } /// <summary> /// 作者 /// </summary> public string Author { get; set; } /// <summary> /// ISBN號(hào) /// </summary> public string ISBN { get; set; } } //對象初始化器 Book book = new Book { Title="Inside COM",ISBN="123-456-789"};
現(xiàn)在你想初始化幾個(gè)就初始化幾個(gè),不需要出現(xiàn)這種情況:
public Book():this("") { } public Book(string title):this(title,0) { } public Book(string title, float price):this(title,price,"") { } public Book(string title, float price, string isbn) { this.Title = title; this.Price = price; this.ISBN = isbn; }
這一串的構(gòu)造方法都是為了應(yīng)付不同的初始化情況。
好了,來看看對象初始化器編譯器在后面為我們做了些什么呢?
使用Reflector反編譯程序集:
Book <>g__initLocal0 = new Book();
<>g__initLocal0.Title = "Inside COM";
<>g__initLocal0.ISBN = "123-456-789";
Book book = <>g__initLocal0;
C#編譯器生成了一個(gè)新的局部變量<>g__initLocal0,調(diào)用Book的默認(rèn)無參構(gòu)造方法初始化它,然后對它的屬性進(jìn)行賦值,最后將這個(gè)局部變量賦值給book。看到這里,我們應(yīng)該想到,要使用對象初始化器,那么這個(gè)對象必須有一個(gè)無參構(gòu)造方法,如果你給這個(gè)方法寫了一個(gè)有參構(gòu)造方法而將它的默認(rèn)無參構(gòu)造方法覆蓋了并且沒有提供一個(gè)新的無參構(gòu)造方法,那么使用對象初始化器編譯的時(shí)候是不會(huì)通過的(不過想不通,為啥C#編譯器生成這么一個(gè)奇怪的局部變量名字,還有為啥不直接使用book呢)。像下面的代碼不更好:
Book book = new Book();
book.Title = "Inside COM";
book.ISBN = "123-456-789";
后來我發(fā)現(xiàn)我是在debug模式下編譯的,換到release模式下變成了這樣:
Book <>g__initLocal0 = new Book();
<>g__initLocal0.Title = "Inside COM";
<>g__initLocal0.ISBN = "123-456-789";
被優(yōu)化了。上面介紹的就是對象初始化器了,那什么是集合初始化器呢?
IList<Book> books = new List<Book>();
//這里就使用了對象初始化器,學(xué)以致用吧
books.Add(new Book { Title = "Inside COM", ISBN = "123-456-789",Price=20 });
books.Add(new Book { Title = "Inside C#", ISBN = "123-356-d89",Price=100 });
books.Add(new Book { Title = "Linq", ISBN = "123-d56-d89", Price = 120 });
這樣的代碼沒少寫吧,實(shí)際上也許比這更復(fù)雜,有了C# 3.0我們睡覺都想笑:
IList<Book> books = new List<Book> {
new Book { Title = "Inside COM", ISBN = "123-456-789",Price=20 },
new Book { Title = "Inside C#", ISBN = "123-356-d89",Price=100 },
new Book { Title = "Linq", ISBN = "123-d56-d89", Price = 120 }
};
還是像剛才一樣,我們來欣賞一下C#編譯器為我們生成的代碼:
List<Book> <>g__initLocal0 = new List<Book>();
Book <>g__initLocal1 = new Book();
<>g__initLocal1.Title = "Inside COM";
<>g__initLocal1.ISBN = "123-456-789";
<>g__initLocal1.Price = 20f;
<>g__initLocal0.Add(<>g__initLocal1);
Book <>g__initLocal2 = new Book();
<>g__initLocal2.Title = "Inside C#";
<>g__initLocal2.ISBN = "123-356-d89";
<>g__initLocal2.Price = 100f;
<>g__initLocal0.Add(<>g__initLocal2);
Book <>g__initLocal3 = new Book();
<>g__initLocal3.Title = "Linq";
<>g__initLocal3.ISBN = "123-d56-d89";
<>g__initLocal3.Price = 120f;
<>g__initLocal0.Add(<>g__initLocal3);
從上面的代碼來看,編譯器自動(dòng)的調(diào)用了List的無參構(gòu)造方法,然后實(shí)例化一個(gè)個(gè)的Book,再一個(gè)個(gè)的Add進(jìn)去,和我們原來的做法沒有什么不同,但是,這是編譯器為我們做的,所以簡省了我們很多的編碼工作。
對象集合初始化器就算介紹完了。有人也許會(huì)說,不就是個(gè)syntx sugar么,有什么。是的,確實(shí)是個(gè)語法糖。在編譯器發(fā)展早期,編譯器科學(xué)家門一直在想方設(shè)法的優(yōu)化編譯器生成的代碼,這個(gè)時(shí)候,編譯器做的主要是對機(jī)器優(yōu)化,因?yàn)槟莻€(gè)時(shí)候機(jī)器的時(shí)間非常寶貴,機(jī)器運(yùn)算速度也不快,今天我們有了足夠好的機(jī)器了(但并不是說我們可以不關(guān)注性能的編寫程序),而且作為編寫軟件的人來說,比機(jī)器的時(shí)間寶貴得多,所以今天的編譯器也在向人優(yōu)化了,從編程語言的發(fā)展之路來講,今天的編程語言比昨天的語言更高級,也更人性化了,我們只要編寫更少的代碼,更符合人的思維的代碼,而只要關(guān)注我們值的關(guān)注的地方。體力活兒就交給編譯器吧。
附加:
剛開始想想這對象集合初始化器也許就一雞肋,沒啥用,不就減少一點(diǎn)點(diǎn)代碼么,像這種簡單的初始化工作,大部分代碼生成器都可以來干。后來在研究匿名類型的時(shí)候突然發(fā)現(xiàn),如果沒有這個(gè)對象初始化器,匿名類型是不是要復(fù)雜一些?或者就是難以實(shí)現(xiàn)?
var test = new{Key="test",Value="test"};如果沒有對象初始化器,匿名類型該怎么辦?
相關(guān)文章
C#使用DevExpress中的SplashScreenManager控件實(shí)現(xiàn)啟動(dòng)閃屏和等待信息窗口
這篇文章介紹了C#使用DevExpress中的SplashScreenManager控件實(shí)現(xiàn)啟動(dòng)閃屏和等待信息窗口的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05使用C#實(shí)現(xiàn)Windows組和用戶管理的示例代碼
這篇文章主要介紹了使用C#實(shí)現(xiàn)Windows組和用戶管理的示例代碼,幫助大家更好的理解和使用c#,感興趣的朋友可以了解下2021-01-01c# 實(shí)現(xiàn)圓形的進(jìn)度條(ProgressBar)
這篇文章主要介紹了c# 如何實(shí)現(xiàn)圓形的進(jìn)度條(ProgressBar),幫助大家更好的理解和學(xué)習(xí)使用c#,感興趣的朋友可以了解下2021-03-03