1. 程式人生 > >你可能不知道的C#特性

你可能不知道的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結構體含有兩個有用屬性HasValueValue來判斷和獲取當前型別是否還有值。

這個名詞對.NET的朋友可能有點生疏,wikipedia給的翻譯是“柯里化”- 個人不太喜歡這個名字,英文是currying。curry化是函數語言程式設計的一種實現技術。大致過程,就是把本來接受多個引數的函式,變成只有第一個引數的函式,然後返回新函式接受餘下的引數。有點拗口,而且對函數語言程式設計或curry化介紹的中文資料也相對較少,我讀過一篇對函數語言程式設計和curry化解釋的比較清楚的文章還有這篇英文的部落格,推薦大家看看。

所有能實現閉包的語言都可以實現curry化。在C#2.0中可以通過匿名委託來實現,在3.0中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<intintint> 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