• C# Dynamic与Newtonsoft.Json的应用


    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     }

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

  • 相关阅读:
    【Leetcode】【Easy】Remove Duplicates from Sorted List
    【Leetcode】【Easy】Pascal's Triangle II
    【Leetcode】【Easy】Pascal's Triangle
    【Leetcode】【Easy】Binary Tree Level Order Traversal II
    【Leetcode】【Easy】Binary Tree Level Order Traversal
    【Leetcode】【Easy】Maximum Depth of Binary Tree
    【Leetcode】【Easy】Minimum Depth of Binary Tree
    【Leetcode】【Easy】Balanced Binary Tree
    【Leetcode】【Easy】Symmetric Tree
    如何使用Action.Invoke()触发一个Storyboard
  • 原文地址:https://www.cnblogs.com/linxmouse/p/11847614.html
Copyright © 2020-2023  润新知