【.NET深呼吸】动态类型(扩充篇)

前面两文中,老周已向大家介绍了关于动态类型对象的两种级别的使用方案,本篇呢,老周再讲一个自定义动态类型的例子。

前面给大家演示的例子中,动态类型中包装的是字典类型来存储数据的,这一次咱们换一种风味,老吃酸的不好,这回就吃点辣的吧,火锅就不吃了,据说火锅的汤底里面有罂粟果的皮,吸食微型鸦片不好。

本例在自定义的动态类型中包装一个XML文档,就用XML来存数据吧。

还是老方法,从DynamicObject类派生出来。

这里我主要实现以下几个功能:

1、可以设置属性,这个是肯定要的,不然怎么赋值数据呢。

        public override bool TrySetMember(SetMemberBinder binder, object value)
        {
            // 查找相应的元素
            var q = from x in xdoc.Root.Elements("item")
                    where ((string)x.Attribute("key")) == binder.Name
                    select x;
            if (q.Count() > 0)
            {
                // 如果元素存在,就修改
                XElement ele = q.First();
                ele.SetAttributeValue("key", value);
            }
            else
            {
                // 如果元素不存在,就添加
                XElement ele = new XElement("item");
                ele.SetAttributeValue("key", binder.Name);
                ele.SetAttributeValue("value", value.ToString());
                xdoc.Root.Add(ele);
            }
            return true;
        }

相信大家还记得,实现不同功能,就是重写不同的虚方法。如果操作成功,就返回true,如果不成功就返回false。在实现中,首先用LINQ筛选XML根下面的item元素,每个item元素存一条数据,元素中key特性表示动态设置的属性名,value表示属性值。也就是说,每个属性的存储结构是这样的:

   <item key = "Age"  value = "105"  />

2、实现属性的get功能,有了赋值,当然也要允许取值,不然会阴阳失调的。

        public override bool TryGetMember(GetMemberBinder binder, out object result)
        {
            result = string.Empty;
            // 筛选
            var q = from x in xdoc.Root.Elements("item")
                    where ((string)x.Attribute("key")) == binder.Name
                    select x.Attribute("value").Value;
            if (q.Count() > 0)
            {
                // 取值
                result = q.First();
                return true;
            }
            return false;
        }

原理和上面差不多,用LINQ查出包含指定成员的名字,binder的Name属性就是被调用的动态成员的名字。查询到对应的项后,把值返回。

3、实现转换功能,想让动态类型支持类型转换,可以重写TryConvert方法。

        public override bool TryConvert(ConvertBinder binder, out object result)
        {
            result = null;
            // 如果是隐式转换,只支持转换为字符串类型
            if (binder.Explicit == false)
            {
                if (binder.Type != typeof(string))
                {
                    return false;
                }
                result = xdoc.Root.ToString();
                return true;
            }
            // 如果是显式转换,只支持转换为XElement类型
            else
            {
                if (binder.Type != typeof(XElement))
                {
                    return false;
                }
                result = xdoc.Root;
                return true;
            }

            return false;
        }

注意这个binder,它有一个属性叫Explicit,大家还记得老周前不久写过有关自定义转换的烂文,转换有隐式和显示,这个属性如果为true,表明这个动态对象正在被显式转换为其他类型,像这样:

oo = (string)dynamicObj;

如果属性为false,就表明正在进行隐式转换,像:

oo = dynamicObj;

binder的Type属性表示要转换的目标类型的type,如果要转换为int,那就是typeof(int)。转换后的值赋值给result参数。

我这里实现的功能是:隐式的话,转换为string;显式的话,转为XElement对象。

最后,在类中初始化一下包装的XML文档。

        private XDocument xdoc = null;

        public MyXmlDynamic()
        {
            // 初始化文档
            xdoc = new XDocument(new XElement("root"));
        }

好了,这个自定义的动态类型完成了,当然了,接下来就是测试。

            dynamic d = new MyXmlDynamic();
            d.Name = "王老三";
            d.Desc = "大坏蛋";
            d.Age = 55;

            Console.WriteLine("Name = {0}, Desc = {1}, Age = {2}", d.Name, d.Desc, d.Age);

            // 测试隐转
            string xml = d;
            Console.WriteLine("\n隐式转换为XML字符串:\n" + xml);

            // 测试显转
            XElement x = (XElement)d;
            Console.WriteLine("\n\n显式转换为XML元素:\n" + x);

于是,得到如下结果:

至此,有关动态类型的话题,老周就讲完了,不知道这位网友看懂否?看不懂也没关系,可以慢慢去研究,多动手干活,就会明白了。编程这玩意儿,很多时候就是依靠动手去试出来的,光读理论没多大用处,更何况,编程是技术性的东西,写再多的理论也没什么用,老周十几年来,一如既往地对编程理论不感兴趣。

所以,有人就跟老周说,老周,你写的东西太不深入了;老周,你的书太没有深度了。

反正我不懂什么叫深度,什么叫深入,老周只想着写出来,人人都能看懂。写那么生僻难懂做什么,不就是敲键盘、写代码吗,是吧,搞那么复杂干吗。

你要是希望老周写点玄学、道学、佛学、儒学、美学、书学相关的文章,那就不同了,那样老周也可以写得很抽象,为啥,因为美学、玄学本来就是抽象。人家庄爷爷说了,不抽象的美都是俗气,抽象的美才是天地大美。

所以说嘛,写什么样的内容就用什么样的方法,写编程的东西,老周是拒绝抽象的。

示例源代码下载

时间: 2024-12-23 16:12:39

【.NET深呼吸】动态类型(扩充篇)的相关文章

【.net深呼吸】动态类型(高级篇)

前面老周给大家介绍了动态类型使用的娱乐级别用法,其实,在很多情景下,娱乐级别的用法已经满足需求了. 如果,你想自己来控制动态类型的行为和数据的存取,那么,就可以考虑用今天所说的高大上技术了.比如,你希望自己弄个字典来存取数据,又或者,你不想用字典,你想用XML来存取数据,那么就必须自己来实现动态对象的行为了. 实现的原理就是从DynamicObject类(位于System.Dynamic命名空间)派生出你自己的类.注意看,这个类的构造函数是protected的,也就是你无法把它实例化,所以,你要

C#如何根据DataTable生成泛型List或者动态类型list

背景:在项目中,sql语句检索返回DataTable,然后根据检索结果做进一步的操作,本篇文章即是介绍如何将DataTable快速生成泛型List返回. 假设存在如下学生类: 1 public class student 2 { 3 public int ID { get; set; } 4 public string StuName { get; set; } 5 public string CityCode { get; set; } 6 } 存在如下数据表Student: 1.泛型List

【转】解析JDK 7的动态类型语言支持

http://www.infoq.com/cn/articles/jdk-dynamically-typed-language Java虚拟机的字节码指令集的数量自从Sun公司的第一款Java虚拟机问世至JDK 7来临之前的十余年时间里,一直没有发生任何变化[1].随着JDK 7的发布,字节码指令集终于迎来了第一位新成员——invokedynamic指令.这条新增加的指令是JDK 7实现“动态类型语言(Dynamically Typed Language)”支持而进行的改进之一,也是为JDK 8

Python的动态类型

动态类型:                                                      在python中,类型是在运行过程中自动决定的,而不是通过代码声明的. 变量:                                                                                     变量是一个系统表的元素,拥有指向对象的连接的空间 python中,类型的概念是存在于对象中的,而不是变量中,变量是通用的. 变量的使用

OC多态,动态类型绑定

// //  main.m //  OC7类 // //  Created by Zoujie on 15/8/23. //  Copyright (c) 2015年 Zoujie. All rights reserved. // #import <Foundation/Foundation.h> #import "Fraction.h"//导入头文件 #import "Complex.h" #define Choose  0 int main(int 

.net 3.5 c#构建动态类型最佳实践

为什么要在.net3.5下面构建,有几个原因: win7自带.net 3.5 ,用户无须安装 我用的是vs2008,不想升级 .net 4.0引入了关键字,但是我想.net 3.5下面也有类似功能 这是构建orm的基础,一言以蔽之,市面上的orm都不好用,我想重新造轮子 暂时没想到 最终结果是这样的 属性名是字符串,属性值随便什么类型. 下图是我写了两个晚上的orm,多表查询生成的是动态类型List,同时映射到多个实体类,在展现层的代码量极少,便于工具生成. 我对orm的设想如下: 基础的增删改

python学习笔记17(动态类型)

动态类型 在我们接触的对象中,有一类特殊的对象,是用于存储数据的,常见的该类对象包括各种数字,字符串,表,词典.在C语言中,我们称这样一些数据结构为变量,而在Python中,这些是对象. 对象是储存在内存中的实体.但我们并不能直接接触到该对象.我们在程序中写的对象名,只是指向这一对象的引用(reference). 引用和对象分离,是动态类型的核心.引用可以随时指向一个新的对象: a = 3 a = 'python' 第一个语句中,3是储存在内存中的一个整数对象,通过赋值,引用a指向对象3. 第二

Python 动态类型

Python 动态类型 1.Python中,类型是在运行过程中自动决定的,并不需要提前在代码中声明. 2.所有变量必须在使用前明确的赋值,否则将会产生错误.#例:NameError: name 'a' is not defined 3.赋值语句 a=3 代表了:创建一个对象代表3:创建一个变量a:将变量与新的对象3相连.实际上是到对象内存空间的一个指针! 4.上面的第3条可翻译为:变量是一个系统表的元素,拥有指向对象的链接的空间:对象是分配的一块内存,有足够空间来表示他们所代表的值:引用是自动形

c++知识点总结--静态与动态类型

对象的静态类型(static type),就是它在程序中被声明时所采用的类型 对象的动态类型(dynamic type)是指“目前所指对象的类型” vitual 函数是动态绑定而来,调用一个virtual函数时,究竟调用哪一个函数实现,取决于发出调用的那个对象的动态类型 class Shape{ public: enum ShapeColor{ Red, Green, Blue = 1, Org }; void printColor(){ cout << Red << Green