当前位置: 首页 > news >正文

艾伟_转载:把委托说透(1):开始委托之旅 委托与接口

委托,本是一个非常基础的.NET概念,但前一阵子在园子里却引起轩然大波。先是Michael Tao的随笔让人们将委托的写法与茴香豆联系到了一起,接着老赵又用一系列文章分析委托写法的演变,并告诫“嘲笑孔乙己的朋友们,你们在一味鄙视“茴”的四种写法的同时,说不定也失去了一个了解中国传统文化的机会呢!”。

在我个人看来,委托是.NET Framework中一个非常炫的特性,绝不会向有些评论里说的那样,根本没有机会接触。恰恰相反,我们几乎每天都会接触委托,使用委托。

其实园子里已经有了很多关于委托的文章,比较有代表性的有:

1. C# 中的委托和事件及其续

2. C#委托,事件理解入门 (译稿)

3. 委托揭秘

4. ……

本系列试图从个人对于委托的理解展开,对委托的内涵和外延均加以讨论。文中有何不妥或不正确的地方,欢迎大家拍砖斧正。

好了,下面让我从一个示例开始,一步一步引入委托的概念。

从示例开始

假设一个系统的用户登录模块有如下所示的代码

class User
{
    public string Name { get; set; }
    public string Password { get; set; }
}

class UserService
{
    public void Register(User user)
    { 
        if (user.Name == "Kirin")
        {
            Log("注册失败,已经包含名为" + user.Name + "的用户");
        }
        else
        {
            Log("注册成功!");
        }
    }
    privte void Log(string message)
    {
        Console.WriteLine(message);
    }
}

UserService类封装用户登录的逻辑,并根据不同的登录情况向控制台打印不同的日志内容。当程序关闭时,所记录的日志自然也随之消失。

客户端的代码为

class Program
{
    static void Main(string[] args)
    {
        User user = new User { Name = "Kirin", Password = "123" };
        UserService service = new UserService();
        service.Register(user);
        Console.ReadLine();
    }
}

 

使用策略模式

然而这样的设计肯定是无法满足用户的需求的,用户肯定希望能够查看以前的日志记录,而不仅仅是程序打开以后的内容。如果我们仅仅修改Log方法的实现,那么用户需求再次改变时我们该如何处理呢?难道要无休止地修改Log方法吗?

既然日志记录的方式是变化的根源,我们自然会想到将其进行封装。我们创建一个名为ILog的接口。

interface ILog
{
    void Log(string message);
}

并创建两个实现了ILog的类,ConsoleLog和TextLog,分别用来向控制台和文本文件输出日志内容。

class ConsoleLog : ILog
{
    public void Log(string message)
    {
        Console.WriteLine(message);
    }
}

 

class TextLog : ILog
{
    public void Log(string message)
    {
        using (StreamWriter sw = File.AppendText("log.txt"))
        {
            sw.WriteLine(message);
            sw.Flush();
            sw.Close();
        }
    }
}

在UserService类中添加一个ILog类型的属性LogStrategy。

class UserService
{
    public ILog LogStrategy { get; set; }
    public UserService()
    {
        LogStrategy = new ConsoleLog();
    }
    public void Register(User user)
    { 
        if (user.Name == "Kirin")
        {
            LogStrategy.Log("注册失败,已经包含名为" + user.Name + "的用户");
        }
        else
        {
            LogStrategy.Log("注册成功!");
        }
    }
}

客户端代码变为如下形式。

class Program
{
    static void Main(string[] args)
    {
        User user = new User { Name = "Kirin", Password = "123" };
        UserService service = new UserService { LogStrategy = new TextLog() };
        service.Register(user);
        Console.ReadLine();
    }
}

在声明UserService的时候,还可以将将LogStrategy设置为TextLog。这样在UserService进行逻辑处理时,使用的LogStrategy即为TextLog,日志将输出到文本文件中。

我们在干什么?我们在重构。重构的结果是什么?重构的结果是实现了一个简单的策略模式。

使用委托

然而策略模式仍然不能满足客户的需求,这是为什么呢?

1. 用户也许会希望自定义Log的实现。当然,你可以通过在客户代码处扩展ILog来实现自己的日志记录方式。如

class TextBoxLog : ILog
{
    private TextBox textBox;
    public TextBoxLog(TextBox textBox)
    {
        this.textBox = textBox;
this.textBox.Multiline = true; }
public void Log(string message) { textBox.AppendText(message); textBox.AppendText(Environment.NewLine); } }

但这种方案是否过于复杂呢?如果用户希望在ListView或其他控件上显示,是否需要逐个创建新类呢?并且这样的实现是否与客户端的耦合过于紧密呢?比如用户希望在ListView的各个列中显示日志内容、时间、来源等不同内容,那么在ListViewLog中对ListView硬编码是否很难重用呢?

2. 用户也许会希望同时使用多种日志记录方式。比如,同时向控制台、文本文件、客户端控件和事件查看器中输出日志。你当然可以在UserService中维护一个List,但这时UserService的职责过多,显然违反了SRP。

下面介绍本文的主角:委托。

我们首先来创建一个名为Log的委托,它接收一个string类型的参数。

public delegate void Log(string message);

然后在UserService类中添加一个Log委托类型的属性LogDelegate。

class UserService
{
    public Log LogDelegate { get; set; }


// … }

在客户端,我们直接声明两个静态方法,它们都包含一个string类型的参数,并且没有返回值。

static void LogToConsole(string message)
{
    Console.WriteLine(message);
}
static void LogToTextFile(string message)
{ 
    using (StreamWriter sw = File.AppendText("log.txt"))
    {
        sw.WriteLine(message);
        sw.Flush();
        sw.Close();
    }
}

客户端声明UserService的代码变为

static void Main(string[] args)
{
    User user = new User { Name = "Kirin", Password = "123" };
    UserService service = new UserService();
    service.LogDelegate = LogToConsole;
    service.LogDelegate += LogToTextFile;
    service.Register(user);
    Console.ReadLine();
}

在构造委托时,我们还可以使用匿名方法和Lambda表达式,在老赵的文章中详细阐述了这些写法的演变。

对于何时使用委托,何时使用接口(即策略模式),MSDN中有明确的描述:

在以下情况下,请使用委托:

    当使用事件设计模式时。

    当封装静态方法可取时。

    当调用方不需要访问实现该方法的对象中的其他属性、方法或接口时。

    需要方便的组合。

    当类可能需要该方法的多个实现时。

在以下情况下,请使用接口:

    当存在一组可能被调用的相关方法时。

    当类只需要方法的单个实现时。

    当使用接口的类想要将该接口强制转换为其他接口或类类型时。

    当正在实现的方法链接到类的类型或标识时:例如比较方法。

您可能觉得上面的例子阐述委托和接口有些过于牵强,事实上有些时候的确很难选择使用接口还是委托。Java中没有委托,但所有委托适用的情况同样可以使用包含单一方法的接口来实现的。在某种程度上,可以说委托是接口(仅定义了单一方法)的一种轻量级实现,它更灵活,也更方便。

到此为止,我们一步一步用委托重构了最初的代码。再接下来的随笔中,我们将开始更深一步的讨论。

转载于:https://www.cnblogs.com/waw/archive/2011/08/29/2157140.html

相关文章:

  • centos7常见问题(更新。。。)
  • 洛谷 P3388 【模板】割点(割顶)题解
  • 大型网站的监控、报警与故障转移
  • mjpg-streamer译文
  • 一起谈.NET技术,.NET Framework源码研究系列之---Delegate
  • gnu下的arm汇编伪指令:.word说明
  • re
  • python循环语句
  • DHCP中继
  • docker
  • 为何投奔BSD
  • 如何查看linux系统安装时间
  • Win XP多用户管理-单机多用户+网络多用户
  • github廖雪峰git
  • sql笔试
  • 2017届校招提前批面试回顾
  • classpath对获取配置文件的影响
  • create-react-app项目添加less配置
  • ES10 特性的完整指南
  • Java程序员幽默爆笑锦集
  • JSONP原理
  • puppeteer stop redirect 的正确姿势及 net::ERR_FAILED 的解决
  • Python_网络编程
  • spring security oauth2 password授权模式
  • vuex 学习笔记 01
  • windows下使用nginx调试简介
  • 百度小程序遇到的问题
  • 看域名解析域名安全对SEO的影响
  • 如何使用 JavaScript 解析 URL
  • 小程序、APP Store 需要的 SSL 证书是个什么东西?
  • 学习笔记TF060:图像语音结合,看图说话
  • [Shell 脚本] 备份网站文件至OSS服务(纯shell脚本无sdk) ...
  • const的用法,特别是用在函数前面与后面的区别
  • 长三角G60科创走廊智能驾驶产业联盟揭牌成立,近80家企业助力智能驾驶行业发展 ...
  • ​ubuntu下安装kvm虚拟机
  • ###51单片机学习(2)-----如何通过C语言运用延时函数设计LED流水灯
  • (ZT)薛涌:谈贫说富
  • (办公)springboot配置aop处理请求.
  • (翻译)Entity Framework技巧系列之七 - Tip 26 – 28
  • (附源码)springboot助农电商系统 毕业设计 081919
  • (论文阅读31/100)Stacked hourglass networks for human pose estimation
  • (一)基于IDEA的JAVA基础1
  • (转)详解PHP处理密码的几种方式
  • (转贴)用VML开发工作流设计器 UCML.NET工作流管理系统
  • .net mvc 获取url中controller和action
  • .NET 反射的使用
  • .net/c# memcached 获取所有缓存键(keys)
  • .net6+aspose.words导出word并转pdf
  • .NetCore Flurl.Http 升级到4.0后 https 无法建立SSL连接
  • .net通用权限框架B/S (三)--MODEL层(2)
  • .net中的Queue和Stack
  • @31省区市高考时间表来了,祝考试成功
  • @DateTimeFormat 和 @JsonFormat 注解详解
  • @EnableConfigurationProperties注解使用
  • [ vulhub漏洞复现篇 ] Apache APISIX 默认密钥漏洞 CVE-2020-13945