1.命名參數允許調用者通過提供參數的名稱來為其賦值,這樣參數的位置就不在重要了。可選參數允許在定義時為某些參數賦值,在調用時可以忽略這些“可選的”參數。命名參數和可選參數可以應用在方法、索引器、構造函數和委托。命名參數和可選參數與dynamic類型結合在一起使用,可以方便的訪諸如Office自動化API之類的COM API。
1.1 命名參數
命名參數的語法為:
參數名稱1:參數值1,參數名稱2:參數值2…
例如以下代碼: static void CreateUser(string name, string password)
static void Main(string[] args)
{
CreateUser(password:"adminpassword",name:"admin");
}
{
Console.WriteLine("name:{0},password:{1}", name, password);
}
可以看到由於在調用時使用了命名參數,參數的位置就不在重要了。
1.2. 可選參數
方法、構造函數、索引器和委托的定義可以指定其參數為必選的還是可選的,在調用時必須提供必選參數,但是可以省略可選參數。
還可以使用System.Runtime.InteropServices.OptionalAttribute特性類定義可選參數,該類從1.0時代就已經包含在基類庫中了。
每一個可選參數的定義都包含默認值(默認值必須是常量),如果在調用時沒有指定該參數,則使用默認值。例如以下代碼: ///
static void Main(string[] args)
{
CreateUser("admin","adminpassword",50);
}
/// 創建用戶
///
///
用戶名稱
///
用戶密碼
///
積分
///
是否鎖定
static void CreateUser(string name, string password,
int score=20,bool isLocked=false)
{
Console.WriteLine("name:{0},password:{1}", name, password);
}
在所有必須參數後面定義可選參數,如果在調用時提供了某個可選參數的值,那麼必須提供該可選參數之前所有可選參數的值(如果此參數之前有可選參數),而不允許使用逗號分隔的形式提供參數,即以下調用是錯誤的:
CreateUser(“admin”,“adminpassword”,,true);
而必須寫成:
CreateUser(“admin”,“adminpassword”,20,true);、
或者更好的解決辦法是使用命名參數:
CreateUser(“admin”,“adminpassword”,isLocked:true);
1.3. COM API訪問
命名和可選參數與dynamic和其他增強一起使得訪問COM API更加方便。例如在C#3.0或更早的版本中在調用某些COM API時,如果要省略某些參數需要使用Type.Missing,例如以下代碼(代碼系摘抄):
var excelApp = new Microsoft.Office.Interop.Excel.Application(); excelApp.get_Range("A1", "B4").AutoFormat(myFormat, Type.Missing,
var myFormat =
Microsoft.Office.Interop.Excel.XlRangeAutoFormat.xlRangeAutoFormatAccounting1;
Type.Missing, Type.Missing, Type.Missing, Type.Missing, Type.Missing);
但是有了命名和可選參數後,可以很簡單的寫成這樣:
excelApp.get_Range("A1", "B4").AutoFormat( Format: myFormat );
2. 類型等價支持(Type Equivalence Support)(此段為翻譯)
如果嵌入來自於強命名托管程序集的類型信息時,可以使在某一應用程序中的類型與獨立的發布版本中類型保持松散的連接。這意味著應用程序可以在不需要重新編譯每一個版本的情況下使用多個版本托管類庫中的類型。
類型嵌入經常用於COM交互,例如使用Microsoft Office中的自動化的應用程序。嵌入類型信息允許同一個應用程序在安裝了不同的Office版本的機器上運行。而且開發人員可以在完全托管解決方案中使用類型嵌入。
來自於某個程序中可以嵌入的類型需要滿足以下條件:
該程序集至少暴露一個公共接口。
該嵌入接口使用ComImport和Guid特性聲明
該程序集使用ImportedFromTypeLib和一個程序集級別的Guid特性標注(默認情況下Visual Basic和Visual C#模版已包含了程序集的Guid特性)。
在指定可以嵌入的公共接口後,可以創建實現了這些接口的類。客戶端程序可以在設計時引用包含了這些公共接口的程序何並且默認Embed Interop Types屬性為true以嵌入類型信息(在命令行使用/link編譯開關可以達到相同的效果),接下來客戶端可以創建這些接口的實例。如果您創建了強命名運行時程序集的新版本,客戶端不需要使用新的程序集重新編譯,相反,客戶端程序通過公共接口的嵌入類型信息繼續使用可用的程序集的版本。
2.1. 首先創建一個強命名接口類庫(根據滿足條件設置屬性)
[ComImport]
[Guid("8DA56996-A151-4136-B474-32784559F6DF")]
public interface ISampleInterface
{
void GetUserInput();
string UserInput { get; }
}
2.2. 創建強命名類庫,引用接口類庫並定義實現以上接口的類:
public classSampleClass: ISampleInterface public voidGetUserInput()
{
private stringp_UserInput;
public stringUserInput { get{ return p_UserInput; } }
{
Console.WriteLine("Please enter a value:");
p_UserInput = Console.ReadLine();
}
}
2.3. 創建客戶端應用程序,引用接口並使用反射的方法動態創建類型執行相應操作:
class Program
{
static void Main(string[] args)
{
Assembly sampleAssembly = Assembly.Load("TypeEquivalenceRuntime");
ISampleInterface sampleClass =
(ISampleInterface)sampleAssembly.CreateInstance("TypeEquivalenceRuntime.SampleClass");
sampleClass.GetUserInput();
Console.WriteLine(sampleClass.UserInput);
Console.WriteLine(sampleAssembly.GetName().Version.ToString());
Console.ReadLine();
}
}
4. 修改實現了接口在的客戶端類,增加新的方法並修改程序集版本號和文件版本號為2.0.0.0:
public DateTime GetDate()
{
return DateTime.Now;
}
5. 再次執行客戶端程序,觀察不同(客戶端將輸出新的版本號)。
在.NET全部使用托管代碼創建的程序集自動會識別更新,也就是說不需要使用額外的屬性定義,直接創建接口、實現接口類庫和客戶端類(或者沒有接口直接創建類庫在客戶端引用),在類庫更新後復制到客戶端引用的位置,客戶端會自動檢測到該更新,這也是.NET程序集為開發人員帶來的好處。但是使用類型等價支持的作用體現在什麼地方,我認為還是方便了COM API的訪問,因為COM可能是使用其他語言編寫的,沒有辦法做到像.NET程序集那樣自動感應版本變化,個人意見,期望高手解答。
3. 總結
Visual C#中提供了動態類型、命名參數、可選參數和類型等價支持,為編程帶來便利,對於訪問COM API來說更方便了。而且微軟多次提到了諸如Office之類的文字,是不是意味著微軟在不斷的鼓勵程序員不斷開發出其於Office的一些應用,亦或是現在其於Office的應用在不斷增加,還是應用程序中與Office的交互在不斷增加,通過增強的特性使這些工作更方便,來鞏固微軟件地位?一家之言,請高手不吝賜教。