你可能不知道的C#特性
-
yield
通常用於迭代器中,向IEnumerable物件提供值或者結束迭代。
如:
yield return expression; yield break;
-
var
用於定義隱式型別的變數。
var i = 5; var s = "Hello";
注意隱式型別(Implicitly typed)並不是“動態型別“,由編譯器在編譯時候決定具體型別。
default除了在switch語句中提供預設的條件之外,還作為關鍵字用於泛型中。比如:
T temp = default(T);
由於事先不知道T型別是引用還是值型別,無法用於和null(只有引用型別可以)或者數字(值型別才能和其比較)進行比較。用default關鍵字,會確保如果是引用型別則返回null,如果是值型別則返回0。
關鍵字global::用於引用全域性名稱空間。
class TestClass : global::TestApp { }
用於多執行緒中表示這個欄位可以由多個同時執行的執行緒修改。更多有關如何使用volatile多執行緒的例子,可以參考如何:建立和終止執行緒。
public volatile int i;C#中可以用extern修飾符宣告外部實現的方法。常用於Interop服務使用非託管程式碼與DllImport屬性一起使用(同時要宣告static),如:
[DllImport("avifil32.dll")] private static extern voidAVIFileInit();C#中可以同時引用兩個型別名完全相同的程式集(常見同一控制元件的多個版本),這時候使用外部命令列指定別名引用,比如:
/r:GridV1=grid.dll /r:GridV2=grid20.dll在程式中使用它們的時候,需要用關鍵extern來引用它們:
extern alias GridV1; extern alias GridV2;GridV1和GridV2會被引用進和全域性名稱空間同級別的額外空間,使用GridV1::Grid或者GridV2::Grid可以得到不同版本的Grid型別。
語法
??操作符
null合併運算子,用於定義可以為null值的引用型別的預設值。如果左運算元不為null,則返回左運算元,反之返回右運算元。
int y = x ?? -1;
同時這裡也是null運算子:
static int? GetNullableInt() { return null; } //... or int? x =null;
where T:new()
我們知道可以用where關鍵字來對泛型定義進行約束,比如:
public class MyClass<T, U> where T : class where U : struct {}同時我們可以對泛型定義中包含建構函式的約束,new()約束告訴編譯器提供的任何型別的引數都含有一個無引數的(預設)構造方法。
public class MyGenericClass<T> where T : IComparable, new() { T item = new T(); }語言特性
Nullable型別
當我們要描述一個值型別可以不存在的的時候,使用nullable型別取代值型別。
使用?修飾符來宣告一個nullable型別,比如:
int? a = 1;
C#通過Nullable類和Nullable結構體來支援nullable型別,Nullable結構體含有兩個有用屬性HasValue和Value來判斷和獲取當前型別是否還有值。
這個名詞對.NET的朋友可能有點生疏,wikipedia給的翻譯是“柯里化”- 個人不太喜歡這個名字,英文是currying。curry化是函數語言程式設計的一種實現技術。大致過程,就是把本來接受多個引數的函式,變成只有第一個引數的函式,然後返回新函式接受餘下的引數。有點拗口,而且對函數語言程式設計或curry化介紹的中文資料也相對較少,我讀過一篇對函數語言程式設計和curry化解釋的比較清楚的文章還有這篇英文的部落格,推薦大家看看。
static class Program { static Func<TArg1, Func<TArg2, TResult>> Curry<TArg1, TArg2, TResult>(this Func<TArg1, TArg2, TResult> f) { return a1 => a2 => f(a1, a2); } static voidMain() { Func<int, int, int> add = (x, y) => x + y; var curriedAdd = add.Curry(); Console.WriteLine(curriedAdd(13)(29)); } }最後,如果大家對用C#實現函數語言程式設計有興趣,可以在MSDN Code中找到更多的程式碼實現。
匿名型別
匿名型別可用來將一組只讀屬性封裝到單個物件中,而無需首先顯式定義一個型別。型別名稱由編譯器生成,並且不能在原始碼級使用,型別也由編譯器推斷決定。常和LINQ查詢表示式的select子句結合使用,初始化其他型別的屬性組成的物件。
var productQuery = from prod in products select new { prod.Color, prod.Price }; //建立由Color和Price屬性組成的新物件 foreach (var v in productQuery) { Console.WriteLine("Color={0}, Price={1}", v.Color, v.Price); }
擴充套件方法
C#3.0中可以對CLR擴充型別,比如,你可能需要得到一句英文字串中有多少個單詞,以往可以寫一個處理string的方法,現在你可以直接為string型別新增一個計算單詞數量的方法:namespace ExtensionMethods { public static class MyExtensions {public static int WordCount(this String str) { returnstr.Split(new char[] { ' ', '.', '?' }, StringSplitOptions.RemoveEmptyEntries).Length; } } }通過this我們把這個WordCount方法擴充到了CLR內建型別,你可以直接像用普通方法一樣使用它:
using ExtensionMethods; //... string s = "Hello Extension Methods"; int i = s.WordCount();更多介紹可以參考MSDN或者ScottGu的部落格。
方法和屬性
List.ForEach()
Array或List有個很實用的ForEach方法,可以直接傳入一個方法對集合中元素操作。如:
List<String> names = new List<String>(); names.Add("Bruce"); names.Add("Alfred"); names.ForEach(Print); names.ForEach(delegate(String name) { Console.WriteLine(name); });private static void Print(string s) { Console.WriteLine(s); }
GetValueOrDefault
對於nullable型別的物件,除了Value和HasValue兩個常用屬性外,還可以使用GetValueOrDefault方法來獲得當前值或者預設值:
float? mySingle = 12.34f; float? yourSingle = -1.0f; yourSingle = mySingle.GetValueOrDefault(-222.22f); // yourSingle=12.34 mySingle = null; yourSingle = mySingle.GetValueOrDefault(); // yourSingle=0