C# 4 引入了一个新类型 dynamic
。 该类型是一种静态类型,但类型为 dynamic
的对象会跳过静态类型检查。 大多数情况下,该对象就像具有类型 object
一样。 在编译时,将假定类型化为 dynamic
的元素支持任何操作。 因此,不必考虑对象是从 COM API、从动态语言(例如 IronPython)、从 HTML 文档对象模型 (DOM)、从反射还是从程序中的其他位置获取自己的值。 但是,如果代码无效,则在运行时会捕获到错误。
在通过 dynamic
类型实现的操作中,该类型的作用是绕过编译时类型检查。 改为在运行时解析这些操作。 dynamic
类型简化了对 COM API(例如 Office Automation API)、动态 API(例如 IronPython 库)和 HTML 文档对象模型 (DOM) 的访问。
在大多数情况下,dynamic
类型与 object
类型的行为类似。 但是,如果操作包含 dynamic
类型的表达式,那么不会通过编译器对该操作进行解析或类型检查。 编译器将有关该操作信息打包在一起,之后这些信息会用于在运行时评估操作。 在此过程中,dynamic
类型的变量会编译为 object
类型的变量。 因此,dynamic
类型只在编译时存在,在运行时则不存在。
下面的示例将 dynamic
类型的变量与 object
类型的变量进行对比。 若要在编译时验证每个变量的类型,请将鼠标指针放在 WriteLine
语句中的 dyn
或 obj
上。 IntelliSense 对 dyn
显示“dynamic” ,对 obj
显示“object” 。
class Program { static void Main(string[] args) { dynamic dyn = 1; object obj = 1; System.Console.WriteLine(dyn.GetType()); System.Console.WriteLine(obj.GetType()); } }
输出结果:
若要查看编译时 dyn
与 obj
之间的区别,请在前面示例的声明和 WriteLine
语句之间添加下列两行:
dyn = dyn + 3; obj = obj + 3;
尝试在表达式 obj + 3
中添加整数和对象时,将报告编译器错误。 但是,对于 dyn + 3
,不会报告任何错误。 在编译时不会检查包含 dyn
的表达式,原因是 dyn
的类型为 dynamic
。
dynamic
关键字可以直接出现,也可以作为构造类型的组件在下列情况中出现:
-
在声明中,作为属性、字段、索引器、参数、返回值、本地变量或类型约束的类型。下面的类定义在多个不同的声明中使用
dynamic
。
class ExampleClass { // 动态字段 static dynamic field; // 动态属性 dynamic prop { get; set; } // 动态返回类型和动态类型参数 public dynamic exampleMethod(dynamic d) { // 局部动态变量 dynamic local = "Local variable"; int two = 2; if (d is int) { return local; } else { return two; } } }
-
在显式类型转换中,作为转换的目标类型。
static void convertToDynamic() { dynamic d; int i = 20; d = (dynamic)i; Console.WriteLine(d); string s = "Example string."; d = (dynamic)s; Console.WriteLine(d); DateTime dt = DateTime.Today; d = (dynamic)dt; Console.WriteLine(d); } // 输出结果: // 20 // Example string. // 7/25/2018 12:00:00 AM
- 在以下任何情况下:类型用作值(如
is
运算符或as
运算符右侧),或者用作构造类型中typeof
的参数。 例如,可以在下列表达式中使用dynamic
。
int i = 8; dynamic d; // 使用 is 操作符 // 在此处动态类型和object相似,The dynamic type behaves like object。除非 somevar 的值为 null ,否则以下表达式将返回true if (someVar is dynamic) { } // 使用 as 操作符 d = i as dynamic; // 使用typeof, 作为构造类型的一部分 Console.WriteLine(typeof(List<dynamic>)); // 以下语句导致编译器错误 //Console.WriteLine(typeof(dynamic));
dynamic
。 Main
方法也将编译时类型检查与运行时类型检查进行了对比。using System; namespace DynamicExamples { class Program { static void Main(string[] args) { ExampleClass ec = new ExampleClass(); Console.WriteLine(ec.exampleMethod(10)); Console.WriteLine(ec.exampleMethod("value")); // 下面的语句会引发编译器异常。因为 exampleMethod 方法仅包含一个参数 //Console.WriteLine(ec.exampleMethod(10, 4)); dynamic dynamic_ec = new ExampleClass(); Console.WriteLine(dynamic_ec.exampleMethod(10)); // 因为 dynamic_ec 是 dynamic 类型, 下面的调用(传递了2个参数)不会引发编译器异常。 // 但是在运行时会引发异常。 //Console.WriteLine(dynamic_ec.exampleMethod(10, 4)); } } class ExampleClass { static dynamic field; dynamic prop { get; set; } public dynamic exampleMethod(dynamic d) { dynamic local = "Local variable"; int two = 2; if (d is int) { return local; } else { return two; } } } } // 输出结果: // Local variable // 2 // Local variable
以上示例中,编译器的作用是将有关每个语句的预期作用的信息一起打包到类型化为 dynamic
的对象或表达式。 在运行时,将对存储的信息进行检查,并且任何无效的语句都将导致运行时异常。
大多数动态操作的结果是其本身 dynamic
。 例如,如果将鼠标指针放在以下示例中使用的 testSum
上,则 IntelliSense 将显示类型“(局部变量)dynamic testSum” 。
dynamic d = 1; var testSum = d + 3; System.Console.WriteLine(testSum);
结果不为 dynamic
的操作包括:
- 从
dynamic
到另一种类型的转换。 - 包括类型为
dynamic
的自变量的构造函数调用。
例如,以下声明中 testInstance
的类型为 ExampleClass
,而不是 dynamic
:
var testInstance = new ExampleClass(d);
动态对象和其他类型之间的转换非常简单。 这样,开发人员将能够在动态行为和非动态行为之间切换。
任何对象都可隐式转换为动态类型,如以下示例所示。
dynamic d1 = 7; dynamic d2 = "a string"; dynamic d3 = System.DateTime.Today; dynamic d4 = System.Diagnostics.Process.GetProcesses();
反之,隐式转换也可动态地应用于类型为 dynamic
的任何表达式。
int i = d1; string str = d2; DateTime dt = d3; System.Diagnostics.Process[] procs = d4;
dynamic
,或者方法调用的接收方的类型为 dynamic
,则会在运行时(而不是在编译时)进行重载决策。 在以下示例中,如果唯一可访问的 exampleMethod2
方法定义为接受字符串参数,则将 d1
作为参数发送不会导致编译器错误,但却会导致运行时异常。 重载决策之所以会在运行时失败,是因为 d1
的运行时类型为 int
,而 exampleMethod2
要求为字符串。ec.exampleMethod2("a string"); ec.exampleMethod2(d1);
dynamic
类型的基础结构,还提供了 IronPython 和 IronRuby 等动态编程语言的实现。 有关 DLR 的详细信息,请参阅动态语言运行时概述。C# 4 包括若干功能,这些功能改善了与 COM API(例如 Office 自动化 API)的互操作体验。 这些改进之处包括 dynamic
类型以及命名参数和可选参数的用法。
通过将类型指定为 object
,许多 COM 方法都允许参数类型和返回类型发生变化。 这样,就必须显式强制转换值,以便与 C# 中的强类型变量保持协调。 如果使用 /link(C# 编译器选项)选项进行编译,则可以通过引入 dynamic
类型将 COM 签名中出现的 object
看作是 dynamic
类型,从而避免大量的强制转换。 例如,以下语句对比了在使用 dynamic
类型和不使用 dynamic
类型的情况下如何访问 Microsoft Office Excel 电子表格中的单元格。
// 引入动态之前 ((Excel.Range)excelApp.Cells[1, 1]).Value2 = "Name"; Excel.Range range2008 = (Excel.Range)excelApp.Cells[1, 1];
// 在引入 dynamic 之后,对 value 属性的访问以及到 excel.range 的转换将由运行时 COM 绑定器处理 excelApp.Cells[1, 1].Value = "Name"; Excel.Range range2010 = excelApp.Cells[1, 1];
其他技术请参考
Title | 说明 |
---|---|
dynamic | 描述 dynamic 关键字的用法。 |
动态语言运行时概述 | 提供有关 DLR 的概述,DLR 是一种运行时环境,它将一组适用于动态语言的服务添加到公共语言运行时 (CLR)。 |
演练:创建和使用动态对象 | 提供有关如何创建自定义动态对象以及创建访问 IronPython 库的对象的分步说明。 |
如何:通过使用 Visual C# 功能访问 Office 互操作对象 | 演示如何创建一个项目,该项目使用命名参数和可选参数、dynamic 类型以及可简化对 Office API 对象的访问的其他增强功能。 |