C#4.0新增功能01 动态绑定 (dynamic 类型)
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 = ;
object obj = ; System.Console.WriteLine(dyn.GetType());
System.Console.WriteLine(obj.GetType());
}
}
输出结果:
若要查看编译时 dyn
与 obj
之间的区别,请在前面示例的声明和 WriteLine
语句之间添加下列两行:
dyn = dyn + ;
obj = obj + ;
尝试在表达式 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 = ; if (d is int)
{
return local;
}
else
{
return two;
}
}
}
在显式类型转换中,作为转换的目标类型。
static void convertToDynamic()
{
dynamic d;
int i = ;
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 = ;
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());
Console.WriteLine(ec.exampleMethod("value")); // 下面的语句会引发编译器异常。因为 exampleMethod 方法仅包含一个参数
//Console.WriteLine(ec.exampleMethod(10, 4)); dynamic dynamic_ec = new ExampleClass();
Console.WriteLine(dynamic_ec.exampleMethod()); // 因为 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 = ; if (d is int)
{
return local;
}
else
{
return two;
}
}
}
}
// 输出结果:
// Local variable
// 2
// Local variable
以上示例中,编译器的作用是将有关每个语句的预期作用的信息一起打包到类型化为 dynamic
的对象或表达式。 在运行时,将对存储的信息进行检查,并且任何无效的语句都将导致运行时异常。
大多数动态操作的结果是其本身 dynamic
。 例如,如果将鼠标指针放在以下示例中使用的 testSum
上,则 IntelliSense 将显示类型“(局部变量)dynamic testSum” 。
dynamic d = ;
var testSum = d + ; System.Console.WriteLine(testSum);
结果不为 dynamic
的操作包括:
- 从
dynamic
到另一种类型的转换。 - 包括类型为
dynamic
的自变量的构造函数调用。
例如,以下声明中 testInstance
的类型为 ExampleClass
,而不是 dynamic
:
var testInstance = new ExampleClass(d);
动态对象和其他类型之间的转换非常简单。 这样,开发人员将能够在动态行为和非动态行为之间切换。
任何对象都可隐式转换为动态类型,如以下示例所示。
dynamic d1 = ;
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[, ]).Value2 = "Name";
Excel.Range range2008 = (Excel.Range)excelApp.Cells[, ];
// 在引入 dynamic 之后,对 value 属性的访问以及到 excel.range 的转换将由运行时 COM 绑定器处理
excelApp.Cells[, ].Value = "Name";
Excel.Range range2010 = excelApp.Cells[, ];
其他技术请参考
Title | 说明 |
---|---|
dynamic | 描述 dynamic 关键字的用法。 |
动态语言运行时概述 | 提供有关 DLR 的概述,DLR 是一种运行时环境,它将一组适用于动态语言的服务添加到公共语言运行时 (CLR)。 |
演练:创建和使用动态对象 | 提供有关如何创建自定义动态对象以及创建访问 IronPython 库的对象的分步说明。 |
如何:通过使用 Visual C# 功能访问 Office 互操作对象 | 演示如何创建一个项目,该项目使用命名参数和可选参数、dynamic 类型以及可简化对 Office API 对象的访问的其他增强功能。 |
C#4.0新增功能01 动态绑定 (dynamic 类型)的更多相关文章
- C#2.0新增功能01 分布类与分部方法
连载目录 [已更新最新开发文章,点击查看详细] 分部类型 拆分一个类.一个结构.一个接口或一个方法的定义到两个或更多的文件中, 每个源文件包含类型或方法定义的一部分,编译应用程序时将把所有部分组 ...
- C#5.0新增功能01 异步编程
连载目录 [已更新最新开发文章,点击查看详细] 如果需要 I/O 绑定(例如从网络请求数据或访问数据库),则需要利用异步编程. 还可以使用 CPU 绑定代码(例如执行成本高昂的计算),对编写异步 ...
- C#3.0新增功能01 自动实现的属性
连载目录 [已更新最新开发文章,点击查看详细] 在 C# 3.0 及更高版本,当属性访问器中不需要任何其他逻辑时,自动实现的属性会使属性声明更加简洁. 它们还允许客户端代码创建对象. 当你声明以 ...
- C#3.0新增功能03 隐式类型本地变量
连载目录 [已更新最新开发文章,点击查看详细] 从 Visual C# 3.0 开始,在方法范围内声明的变量可以具有隐式“类型”var. 隐式类型本地变量为强类型,就像用户已经自行声明该类型,但 ...
- C#基础拾遗系列之二:使用ILSpy探索C#7.0新增功能点
C#基础拾遗系列之二:使用ILSpy探索C#7.0新增功能点 第一部分: C#是一种通用的,类型安全的,面向对象的编程语言.有如下特点: (1)面向对象:c# 是面向对象的范例的一个丰富实现, 它 ...
- C#2.0新增功能06 协变和逆变
连载目录 [已更新最新开发文章,点击查看详细] 在 C# 中,协变和逆变能够实现数组类型.委托类型和泛型类型参数的隐式引用转换. 协变保留分配兼容性,逆变则与之相反. 以下代码演示分配兼容性.协 ...
- 说说C# 8.0 新增功能Index和Range的^0是什么?
前言 在<C# 8.0 中使用 Index 和 Range>这篇中有人提出^0是什么意思?处于好奇就去试了,结果抛出异常.查看官方文档说^0索引与 sequence[sequence.Le ...
- C#基础拾遗系列之二:C#7.0新增功能点
第一部分: C#是一种通用的,类型安全的,面向对象的编程语言.有如下特点: (1)面向对象:c# 是面向对象的范例的一个丰富实现, 它包括封装.继承和多态性.C#面向对象的行为包括: 统一的类型系统 ...
- C#7.0新增功能点
原文地址: https://www.cnblogs.com/runningsmallguo/p/8972678.html 第二部分:C#7.0新增的功能 (1)数字字面量的提升: C#7中的数字文字 ...
随机推荐
- 开源libco库:单机千万连接、支撑微信8亿用户的后台框架基石
微信于2013年开源的ibco库,是微信后台大规模使用的c/c++协程库,2013年至今稳定运行在微信后台的数万台机器上.libco在2013年的时候作为腾讯六大开源项目首次开源,ibco支持后台敏捷 ...
- Delphi 屏幕抓图技术的实现
摘 要:本文以Delphi7.0作为开发平台,给出了网络监控软件中的两种屏幕抓图技术的设计方法和步骤.介绍了教师在计算机机房内教学时,如何监控学生计算机显示器上的画面,以保证教学的质量和效果. 引言 ...
- 三个臭皮匠,顶上一个诸葛亮——在Google Ideathon上Design Thinking分享
4月26日很荣幸的被邀请参加Google Ideathon做Design Thinking的分享. 这次主要分享了Design Thinking的基本方法流程,以及在真实项目的运用.现在整理一下当时选 ...
- python集合的内建函数
s.issubset(t) 如果s 是t 的子集,则返回True,否则返回False s.issuperset(t) 如果t 是s 的超集,则返回True,否则返回False s.union(t) 返 ...
- Python连载13-shutile模块(续)和zipfile模块
一.shutil模块(续) 1.函数:upack_archive() (1)用法:解包操作 (2)格式:shutil.unpack_archive("归档文件地址“,”解包之后的地址“) ( ...
- Laravel --- artisan创建表以及填充表数据流程总结
1.创建建表文件 php artisan make:migration create_comments_table 打开database/migrations/xxx_create_comments_ ...
- 编解码器之战:AV1、HEVC、VP9和VVC
视频Codec专家Jan Ozer在Streaming Media West上主持了一场开放论坛,邀请百余名观众参与热门Codec的各项优势与短板.本文整理了讨论的主要成果,基本代表了AV1.HEVC ...
- ZooKeeper学习之路(四)—— Java 客户端 Apache Curator
一.基本依赖 Curator是Netflix公司开源的一个Zookeeper客户端,目前由Apache进行维护.与Zookeeper原生客户端相比,Curator的抽象层次更高,功能也更加丰富,是目前 ...
- Element-ui安装之MessageBox详解
1.首先根据官方文档进行Element-ui的安装,这个过程很简单(通过webpack-simple) 1) vue init webpack-simple element-ui 2) cd elem ...
- Java基础篇01
01. 面向对象 --> 什么是面向对象 面向对象 面向对象程序设计,简称OOP(Object Oriented Programming). 对象: 指人们要研究的任何事物,不管是物理上具体的事 ...