C#中Dynamic关键字

dynamic关键字和动态语言运行时(DLR)是.Net 4.0中新增的功能。

什么是"动态"?

  编程语言有时可以划分为静态类型化语言和动态类型化语言。C#和Java经常被认为是静态化类型的语言,而Python、Ruby和JavaScript是动态类型语言。

  一般而言,动态语言在编译时不会对类型进行检查,而是在运行时识别对象的类型。这种方法有利有弊:代码编写起来更快、更容易,但无法获取编译器错误,只能通过单元测试和其他方法来确保应用正常运行。

  C#最初是作为纯静态语言创建的,但是C#4添加了一些动态元素,用于改进与动态语言和框架之间的互操作性。C# 团队考虑了多种设计选项,但最终确定添加一个新关键字来支持这些功能:dynamic。

  dynamic关键字可充当C#类型系统中的静态类型声明。这样,C#就获得了动态功能,同时仍然作为静态类型化语言而存在。

  由于编译时不会去检查类型,所以导致IDE的IntellSense失效。

dynamic、Object还是Var?

  那么,dynamic、Object和var之间的实际区别是什么?何时使用它们?

  先说说var,经常有人会拿dynamic和var进行比较。实际上,var和dynamic完全是两个概念,根本不应该放在一起做比较。

  var实际上编译器抛给我们的语法糖,一旦被编译,编译器就会自动匹配var变量的实际类型,并用实际类型来替换该变量的声明,等同于我们在编码时使用了实际类型声明。而dynamic被编译后是一个Object类型,编译器编译时不会对dynamic进行类型检查。

  再说说Object,上面提到dynamic类型再编译后是一个Object类型,同样是Object类型,那么两者的区别是什么呢?

  除了在编译时是否进行类型检查之外,另外一个重要的区别就是类型转化,这也是dynamic很有价值的地方,dynamic类型的实例和其他类型的实例间的转换是很简单的,开发人员能够很方便地在dyanmic和非dynamic行为间切换。任何实例都能隐式转换为dynamic类型实例,见下面的例子:

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;

C# 使用dynamic类型来访问JObject对象

dynamic是C#里面的动态类型,可在未知类型的情况访问对应的属性,非常灵活和方便。

使用Json.Net可以把一个Json字符串转换成一个JObject对象,如果有已知强类型,如果有已知对应的强类型,可以直接转成对应的类型。但如果没有,要访问Json里面对应的数据的时候,就显得比较麻烦。我们可以借助DynamicObject来访问对应的属性。

DynamicObject

我们要创建一个动态类,用于访问JObject,代码如下:

 1 public class JObjectAccessor : DynamicObject
 2     {
 3         private JToken obj;
 4
 5         public JObjectAccessor(JToken obj)
 6         {
 7             this.obj = obj;
 8         }
 9
10         public override bool TryGetMember(GetMemberBinder binder, out object result)
11         {
12             result = null;
13             var val = obj?[binder.Name];
14             if (val == null) return false;
15             result = Populate(val);
16             return true;
17         }
18
19         private object Populate(JToken token)
20         {
21             var val = token as JValue;
22             if (val != null) return val.Value;
23             else if (token.Type == JTokenType.Array)
24             {
25                 var list = new List<object>();
26                 foreach (var item in token as JArray)
27                 {
28                     list.Add(Populate(item));
29                 }
30
31                 return list;
32             }
33             else return new JObjectAccessor(token);
34         }
35     }

接下来就可以开始使用它了:

 1 class Program
 2     {
 3         static void Main(string[] args)
 4         {
 5             string json = @"{'name': '小明','contact': {'city': '上海','phone': '18888888888'},'shopping_list': [{'type': '笔记本','price': 7888}, {'type':'火车', 'price':1.5}]}";
 6             JObject jobj = JObject.Parse(json);
 7             dynamic obj = new JObjectAccessor(jobj);
 8
 9             Console.WriteLine($"{obj.name}: {obj.contact.city} {obj.contact.phone}");
10             Console.WriteLine($"{obj.shopping_list[0].type}: {obj.shopping_list[0].price}");
11         }
12     }

运行一下程序,看一下输出结果:

12-30 10:57
查看更多