c# dynamic用法_dynamic_cast详解

c# dynamic用法_dynamic_cast详解dynamic是FrameWork4.0的新特性。dynamic的出现让C#具有了弱语言类型的特性。编译器在编译的时候不再对类型进行检查,编译期默认dynamic对象支持你想要的任何特性。比如,即使你

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

dynamic是FrameWork4.0的新特性。dynamic的出现让C#具有了弱语言类型的特性。编译器在编译的时候不再对类型进行检查,编译期默认dynamic对象支持你想要的任何特性。比如,即使你对GetDynamicObject方法返回的对象一无所知,你也可以像如下那样进行代码的调用,编译器不会报错:

dynamic dynamicObject = GetDynamicObject();
Console.WriteLine(dynamicObject.Name);
Console.WriteLine(dynamicObject.SampleMethod());

说到正确用法,那么首先应该指出一个错误用法:

常有人会拿var这个关键字来和dynamic做比较。实际上,var和dynamic完全是两个概念,根本不应该放在一起做比较。var实际上是编译期抛给我们的“语法糖”,一旦被编译,编译期会自动匹配var 变量的实际类型,并用实际类型来替换该变量的申明,这看上去就好像我们在编码的时候是用实际类型进行申明的。而dynamic被编译后,实际是一个object类型,只不过编译器会对dynamic类型进行特殊处理,让它在编译期间不进行任何的类型检查,而是将类型检查放到了运行期。

这从visual studio的编辑器窗口就能看出来。以var声明的变量,支持“智能感知”,因为visual studion能推断出var类型的实际类型,而以dynamic声明的变量却不支持“智能感知”,因为编译器对其运行期的类型一无所知。对dynamic变量使用“智能感知”,会提示“此操作将在运行时解析”。

关于dynamic变量是一个object变量这一点,可以通过IL代码得到验证,这里不再贴出IL代码。当然,编译器也对dynamic声明进行了处理,以区别直接object变量。

dynamic是做为简化互操作性而被MSDN中大肆渲染,我感觉正是基于这一点,才被部分开发人员误解:因为很多开发人员不会接触COM+、OFFICE二次开发之类的编码,所以急需要一个dynamic的应用理由。那么,在日常开发中,我认为dynamic很有价值的一点是:

类型转换
Dynamic类型的实例和其他类型的实例间的转换是很简单的,开发人员能够很方便地在dyanmic和非dynamic行为间切换。任何实例都能隐式转换为dynamic类型实例,见下面的例子:
dynamic d1 = 7;
dynamic d2 = "a string";
dynamic d3 = System.DateTime.Today;
dynamic d4 = System.Diagnostics.Process.GetProcesses();

Conversely, an implicit conversion can be dynamically applied to any expression of type dynamic.

反之亦然,类型为dynamic的任何表达式也能够隐式转换为其他类型。(英文的翻译)

int i = d1;
string str = d2;
DateTime dt = d3;
System.Diagnostics.Process[] procs = d4;

方法中含有dynamic类型参数的重载问题
如果调用一个方法是传递了dynamic类型的对象,或者被调用的对象是dynamic类型的,那么重载的判断是发生在运行时而不是编译时。
 
动态语言运行时(dynamic language runtime DLR)
动态语言运行时是.NET Framework 4 Beta 1中的一组新的API,它提供了对c#中dynamic类型的支持,也实现了像IronPython和IronRuby之类的动态程序设计语言。

dynamic可以简化反射

以前我们这样使用反射:

public class DynamicSample
{
public string Name { get; set; }

public int Add(int a, int b)
{
return a + b;
}
}
DynamicSample dynamicSample = new DynamicSample(); //create instance为了简化演示,我没有使用反射
var addMethod = typeof(DynamicSample).GetMethod("Add");
int re = (int)addMethod.Invoke(dynamicSample, new object[] { 1, 2 });

现在,我们有了简化的写法:

dynamic dynamicSample2 = new DynamicSample();
int re2 = dynamicSample2.Add(1, 2);

我们可能会对这样的简化不以为然,毕竟看起来代码并没有减少多少,但是,如果考虑到效率兼优美两个特性,那么dynamic的优势就显现出来了。编译器对dynamic进行了优化,比没有经过缓存的反射效率快了很多。如果非要比较,可以将上面两者的代码(调用Add方法部分)运行1000000就可以得出结论。

 

 

dynamic关键字才出来的时候,觉得真是没什么用,谁总是和com交互来交互去啊,唯恐避之不及啊。

        后来逐渐算是有了一些使用心得,发现这货还真是犀利啊,故在此举几个例子,起抛砖引玉之用。

1.替代XXX.GetType().GetProperty(“YYY”).GetValue(XXX)

static object GetPerson()
{
    return new Person { Name = "Leo" };
} 

有时候难免会遇到这种返回object的倒霉代码(特别是跟反射有关的时候),这时我们又要访问其中的某个属性,那个费劲啊,现在有了dynamic感觉好多了。

object objPerson = GetPerson();
var objName =  objPerson.GetType().GetProperty("Name").GetValue(objPerson);
Console.WriteLine(objName);

dynamic dynPerson = GetPerson();
var dynName = dynPerson.Name;
Console.WriteLine(dynName);

另一个好处是性能会得到一程度的提升:

            Watch = new Stopwatch();
            Watch.Start();
            for (int i = 0; i < 1000000; i++)
            {
                objName = objPerson.GetType().GetProperty("Name").GetValue(objPerson);
            }
            Watch.Stop();
            Console.WriteLine(Watch.Elapsed);


            Watch.Restart();
            for (int i = 0; i < 1000000; i++)
            {
                dynName = dynPerson.Name;
            }
            Watch.Stop();
            Console.WriteLine(Watch.Elapsed);

大致结果如下图,还是快了很多的:

c# dynamic用法_dynamic_cast详解

2.拯救接手接口没设计好的代码的倒霉孩子

        比如这里有N个WCF服务,返回了N个对象的集合,这几个对象没啥关系,其实又有一点关系,倒霉孩子又不会让Entity Framework生成的类自动继承某个接口(本文里用本地方法代替WCF服务)。

        这里来举一个例子,首先有下面2个倒霉的类,同样string类型的name是可以提取接口的(这里真的合适提取么……),同样名称但不同类型的ID,完全无关的Age和Price。

    public class Person
    {
        public int ID { get; set; }

        public string Name { get; set; }

        public int Age { get; set; }

        public static List<Person> GetPersonList()
        {
            return new List<Person>
            {
                new Person{ Name = "Leo1" , Age = 10 },
                new Person{ Name = "Leo2" , Age = 20 },
                new Person{ Name = "Leo3" , Age= 30 }
            };
        }
    }

    public class Car
    {
        public Guid ID { get; set; }

        public string Name { get; set; }

        public double Price { get; set; }

        public static List<Car> GetCarList()
        {
            return new List<Car>
            {
                new Car{ Name = "Focus1" , Price = 100 },
                new Car{ Name = "Focus2" , Price = 200 },
                new Car{ Name = "Focus3" , Price = 300 }
            };
        }
    }

        我用2个static方法返回不同类型的List<T>来模拟WCF中最普通的调用。

        static void Main(string[] args)
        {
            List<dynamic> list = new List<dynamic>();
            //用本地方法替代WCF服务,您假装是通过WCF获取的list
            Person.GetPersonList().ForEach((p) => list.Add(p));
            TestDynamic2(list,"Leo2");

            list = new List<dynamic>();
            //用本地方法替代WCF服务,您假装是通过WCF获取的list
            Car.GetCarList().ForEach((c) => list.Add(c));
            TestDynamic2(list,"Focus3");

            Console.ReadKey();
        }

        private static void TestDynamic2(List<dynamic> list,string name)
        {
            //可以无差别的使用ID和Name属性
            dynamic first = list.OrderBy(d => d.ID).FirstOrDefault(d => d.Name.Contains(name));

            //差别对待不同的属性,这里供参考,不建议这么写,这会导致依赖具体的类型
            if (first is Person)
            {
                Console.WriteLine(first.Age);
            }
            else
            {
                Console.WriteLine(first.Price);
            }
        }

 

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/168016.html原文链接:https://javaforall.net

(0)
上一篇 2022年8月4日 上午8:46
下一篇 2022年8月4日 上午8:46


相关推荐

  • python 实现发送邮件功能

    python 实现发送邮件功能一 前言在开始正题之前 我们先理一下常见的电子邮件协议 SMTP POP IMAP 都遵循 TCP IP 协议规范 至于 Exchange 是邮件服务器 不是收邮件和发邮件的协议 不要混淆概念 一般情况下不用 因为它是微软的产品 SMTP 即简单邮件传输协议 它是一组用于由源地址到目的地址传送邮件的规则 由它来控制信件的中转方式 简单来说是发送协议 POP 协议允许电子邮件客户端下载服务器上的邮件 但是在客户端的操作 如移动邮件 标记已读等 不会反馈到服务器上 属于收件协议

    2026年3月17日
    1
  • PyTorch建立resnet34和resnet101代码[通俗易懂]

    PyTorch建立resnet34和resnet101代码[通俗易懂]model.pyimporttorch.nnasnnimporttorchclassBasicBlock(nn.Module):expansion=1def__init__(self,in_channel,out_channel,stride=1,downsample=None):super(BasicBlock,self).__init__()self.conv1=nn.Conv2d(in_channels=

    2026年4月16日
    4
  • 一文搞懂│http 和 https 的通信过程及区别

    一文搞懂│http 和 https 的通信过程及区别拨云见日,带你学好网络安全通信基础

    2022年8月16日
    11
  • webpack配置文件_pack luggage

    webpack配置文件_pack luggage前言上一篇文章我们使用webpack打包成功了,但是每次都要自己手动输入打包的文件地址和打包到哪里去的地址,非常麻烦,所以这里介绍使用配置文件进行打包webpack.config.js首先我们创

    2022年8月7日
    9
  • Maven安装与配置

    Maven安装与配置一、需要准备的东西JDKEclipseMaven程序包二、下载与安装前往https://maven.apache.org/download.cgi下载最新版的Maven程序:2.将文件解压到D:\ProgramFiles\Apache\maven目录下:新建环境变量MAVEN_HOME,赋值D:\ProgramFiles\Apache\mave…

    2022年6月11日
    35
  • MySQL数据库基础知识_Mysql教程

    MySQL数据库基础知识_Mysql教程一、数据库基础知识在任何一个关系数据库中,第一范式是对关系模式的基本要求,不满足第一范式的数据库就不是关系数据库第二范式定义:若关系模式R∈1NFR\in1NF,且关系模式R的每个非主属性完全函数依赖于码(候选码),则R∈2NFR\in2NF第三范式定义:在2NF基础上,若一个关系模式中所有非主属性完全依赖于码并且不传递依赖于码,则R∈3NFR\in3NFBC范式定义:设关系模式R(U,F)

    2022年8月22日
    7

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号