.NET独有的精巧泛型设计模式

在.NET发展史中,2.0是具有里程碑意义的一个版本。从这个版本,.NET青出于蓝(Java),而胜于蓝。在.NET 2.0带来的诸多新特性中,我认为泛型是最重要,没有之一。

虽然泛型出现已有多年,连Java都早已借鉴引入了泛型(虽然是语法糖),可是用泛型的编程思维方式并没有得到相应的普及。一方面是由于过去大量的Framework仍然是在非泛型时代写成的,另一方面泛型的设计模式没有得到发展,改变的时候该到了。

来举一个例子说明这两点。我们如果写过网络数据抓取的代码,应该熟悉这样的代码:

var request = WebRequest.Create("http://www.cnblogs.com/") as HttpWebRequest;

或者这么写,也是一样:

var request = HttpWebRequest.Create("http://www.cnblogs.com/") as HttpWebRequest;

大家可想过,为什么每次都要as一下?

类似的情况还有,比如做图像处理的弟兄会熟悉:

var bm = Image.FromFile("e:\\me.jpg") as Bitmap;

var bm = Bitmap.FromFile("e:\\me.jpg") as Bitmap;

我想过,但没想明白。上面两种写法,都是调用父类的工厂方法,实际返回了一个子类的实例。显然,即使不了解OCP,凭直觉也应该想到,父类的实现中不应该被子类所决定。写WebRequest和Image的前辈可能也觉得直接返回子类实例不妥,所以阴险地把方法签名的返回类型改成了父类。

虽然这种行径值得严重鄙视。但.NET程序员大都是人云亦云,照葫芦画瓢的好学生,所以这个问题多年了也没有修改。

理想的设计应该是这样:父类的每个子类,都有独立的工厂方法,返回其自身的实例。这样做法,在泛型出现前非常笨拙,得不偿失,但有了泛型,就可以精巧地实现。

以模拟Image类为例,Image和BitMap实现如下:

class Image<T> where T:Image<T>, new()
{
    public string Path { get; set; }

    public static T FromFile(string path)
    {
        return new T() { Path = path };
    }
}

class Bitmap:Image<Bitmap>
{
}

Image自身的工厂方法,就没有存在的必要了。

可以简单地测试一下:

var path = @"e:\me.jpg";
var bm = Bitmap.FromFile(path); ;

Console.WriteLine(bm.Path);
Console.WriteLine(bm.GetType().Name);

输出结果如下:

Path: e:\me.jpg
Type: Bitmap

为了让大家更熟悉一下,再举一个实现数据结构中的二叉树作例子。

传统的树节点类,无论无论C/C++/Java都是类似这样:

class TreeNode
{
    public TreeNode LeftChild { get; set; }

    public TreeNode RightChild { get; set; }

    public TreeNode Parent { get; set; }

    public int Value { get; set; }
}

大家知道,二叉树又分好几种,AVL树、B树、红黑树等等。实现特殊的二叉树数据结构,势必要继承TreeNode。由于树节点的类型中,有类型为基类的成员,所以在子类操作这些成员时,往往也要强制转换类型,这比Image和WebRequest的例子,只在实例创建时转换类型还麻烦。

这就该泛型模式一显身手的好机会了,请看其父类型的实现:

/// <typeparam name="T">Type of the node.</typeparam>
/// <typeparam name="K">Type of the node value.</typeparam>
class TreeNode<T,K> where T:TreeNode<T,K> where K: IComparable<K>
{
    public T LeftChild { get; set; }

    public T RightChild { get; set; }

    public T Parent { get; set; }

    public K Value { get; set; }
}

之后,实现任何一种特殊二叉树结构,比如RBTreeNode代表红黑树节点,可以这样:

class RBTreeNode : TreeNode<RBTreeNode,Int32>
{
    /// <summary>
    /// 树节点颜色,是否为红。
    /// </summary>
    public bool IsRed { get; set; }

    public override string ToString()
    {
        return this.Value + "," + (this.IsRed ? "R" : "B");
    }
}

这个是AVL树:

class AvlTreeNode : TreeNode<AvlTreeNode,Int32>
{
    /// <summary>
    /// 节点的平衡度
    /// </summary>
    public int Balance { get; set; }

    public override string ToString()
    {
        return "Balance: " + Balance + ", Value: " + this.Value;
    }
}

不但完全符合OCP原则,而且再也不需要as来强制转换节点类型了。

这肯定不是我的首创,其实.NET Framework中已经不少这样的设计,比如IComparable<T>接口。也有不少优秀的框架采用了类似的设计,比如大石头同学的ORM框架NewLife.XCode。

看上去也很简单吧,但是很多人思维还停留在面向对象语言刚诞生的阶段,还不习惯用这种设计模式。我认为这种写法足够典型和通用,足以得上一种设计模式,而且是.NET特殊优势,独特魅力。

说到设计模式,其实GOF提出的23种设计模式多年了,已经过时,出现了许多新模式(比如并发编程方面,参考Wiki Design Pattern)。旧有的模式中,有的已经包含在.NET语言特性中,有的模式实现方式已经改头换面。尤其在泛型出现后,许多模式的实现可以变得简洁许多,优雅许多。

不要一遍遍炒过去的冷饭,设计模式应该与时俱进,永远是充满新鲜活力的话题。

时间: 2024-10-24 21:06:00

.NET独有的精巧泛型设计模式的相关文章

Cocos2d-x从入门到精通第四课 -- 《Cocos2d-x中的FileUtils》d

FileUtils的作用 课程视频教程地址:http://edu.csdn.net/course/detail/1342/20982?auto_start=1 FileUtils是Cocos2d-x里面的文件管理类.它对我们游戏中的资源文件起到管理的作用,可以说是游戏资源管理的大管家.FileUtils可以进行读写文件,可以设置可搜索路径,可以获得资源文件的绝对和相对路径,可以判断文件和文件夹是否存在,可以获得资源文件的大小,等.很多对文件操作的所需的功能FileUtils都有对应的接口.所以说

【设计模式】面向对象小结——接口、泛型和委托

上一篇文章,通过一张概括性导图,总结性的讲解了一下我对面向对象的理解,以及重写与重载的区别. 这篇文章的主要内容是写,我对面向对象功能:接口.泛型和委托的理解..很多人都认为,我不使用它们,照样能实现程序想要的功能,为什么还要用它们?可以这么说,如果要想混IT行业,那么,我们就要成为专业人员.那么,我们所写的代码就不仅仅是实现其功能那么简单了,我们还要提高程序的性能,更重要的是,我们所写的代码是给别人看的,要让别人看得懂,实现团队合作..接口.泛型和委托让我们的代码更结构化.专业化..    

四、集合与泛型、委托与事件-----《大话设计模式》

一.集合与泛型   数组 集合(ArrayList) 泛型 优点 连续存储.快速从头到尾遍历和修改元素 使用大小可按需动态增加 类型安全:省去拆箱和装箱操作 缺点 创建时必须制定数组变量的大小: 两个元素之间添加元素比较困难 类型不安全,接受所有类型的数据: 导致一直进行拆箱和装箱操作,带来很大的性能消耗   public partial class Form1 : Form { IList arrayAnimal_list = new ArrayList(); //声明并初始化集合 IList

[04]C++系统工程师 cpp工程师 C++面向对象 设计模式 STL 泛型 算法 系统设计

视频试看链接:https://pan.baidu.com/s/1skTNsqp 总目录:   频目录1: 视 视频目录2: 视频: 课件:  源码:

大话设计模式学习笔记——面向对象基础

前言 好记性不如烂"笔头"系列--大话设计模式学习笔记 目录 面向对象基础 面向对象基础 什么是类与实例 一切事物皆为对象,即所有的东西老师对象,对象就是可以看到.感觉到.听到.触摸到.尝到.或闻到的东西.准确地说,对象是一个自包含的实体,用一组可识别的特性和行为来标识.面向对象编程,英文叫 Object-Oriented Programming,其实就是针对对象来进行编程的意思.类就是具有相同属性和功能的对象的抽象集合.实例就是一个真实的对象.比如我们属于'人'类,而个人就是'人'类

我的设计模式总结

各位小牛大牛老鸟菜鸟们好,欢迎参观我的设计模式世界.这个世界我已经总结多年了,现在才刚刚成型.But I have a dream,梦想所有开发者都能一夜之间认清所有设计模式,还幻想以后大家认识设计模式时,必首先google本文,嘿嘿. 前辈同仁们已经总结过很多,至今首页上设计模式的文章仍然层出不穷.但我总认为,在GOF的23个设计模式提出多年了,该需要些变化和扩展了.特别适用于.NET(或Mono)的设计模式,好像没有系统的总结.新年伊始,推出这篇总结,我个人不喜欢人云亦云,对设计模式的整体,

《大话设计模式》笔记-基础知识2:面向对象基础知识

本书作者介绍每一种设计模式就是用类图(用的UML类图,其基础知识另文介绍)+面向对象语言小程序+人物对话解释知识点.本文就是介绍<大话设计模式>中所用到面向对象的基础知识(基于C#语言),方便阅读本书代码. 类与实例 (1)对象 一切事物皆为对象,对象就是看到.感觉到.听到.摸到等实实在在的东西.准确地说,对象是一个自包含的实体,用一组可识别的特性和行为来标识.面向对象编程,就是针对对象来进行编程的. (2)类 类就是具有相同的属性和功能的对象的抽象的集合.例子如下 "class&q

NHibernate框架与BLL+DAL+Model+Controller+UI 多层架构十分相似--『Spring.NET+NHibernate+泛型』概述、知识准备及介绍(一)

原文://http://blog.csdn.net/wb09100310/article/details/47271555 1. 概述 搭建了Spring.NET+NHibernate的一个数据查询系统.之前没用过这两个框架,也算是先学现买,在做完设计之 后花了一周搭建成功了.其中,还加上了我的一些改进思想,把DAO和BLL之中相似且常用的增删改查通过泛型T抽象到了DAO和BLL的父类中,其DAO 和BLL子类只需继承父类就拥有了这些方法.和之前的一个数据库表(视图)对应一个实体,一个实体对应一

工厂设计模式

工厂设计模式:是一种创建对象的模式,使代码的耦合.工厂模式就是给外部批量提供相同或者不同的产品,而外部不需要关心工厂是如何创建一个复杂产品的过程.所以工厂模式可以降低模块间的耦合,同时可以提高扩展性(当有新的产品出现时,只需要扩展工厂就行了,上层模块不敏感). 1:创建一个接口或者抽象类 2:实现接口(产品类) 3:创建一个工厂类:在工厂类里面创建对象(产品的实例化类) 4:在客户端创建工厂,得到我们想要创建的对象(客户端的引用) 1.)标准工厂方法模式 首先先介绍一下标准的工厂方法模式,不带任