开放封闭原则(OCP,Open Closed Principle)是面向对象原则的核心。由于软件设计本身所追求的墓边就是封装变化,降低耦合,而开放封闭原则就是对这一目标的直接体现。(你必须知道的.NET p48页)

以下用例子说明,同样来自该书。

假设在一个柜台业务处理系统,首先有客户:

    public class Client
{
public string Name{get;set;}
private string ClientType { get; set; }
}
ClientType表示客户的类型,如"存款用户","转账用户","取款用户"
public class BusyBankStaff
{
private readonly BankProcess bankProcess = new BankProcess();
public void HandleProcess(Client client)
{
switch (client.ClientType)
{
case "存款用户":
bankProcess.Deposit();
break;
case "转账用户":
bankProcess.Transfer();
break;
case "取款用户":
bankProcess.DrawMoney();
break;
}
}
}

业务处理类

 public class BankProcess
{
public void Deposit() { }
public void Transfer() { }
public void DrawMoney() { }
}

问题是,如果银行多了一种业务类型,比如代购公积金,那么,必然地,BankProcess要修改为

public class BankProcess
{
public void Deposit() { }
public void Transfer() { }
public void DrawMoney() { }
public void BuyFund(){}
}

,并且BusyBankStaff中坏味道的switch语句又要增加一个条件。

其实,这样类的设计违反了开放封闭原则。

所谓开放封闭,是指,对修改封闭——一个类一旦写好,就不能再修改;对扩展开放——如果有新的需求,可以在不修改原系统的基础上方便增加。

怎么改?抽象!对可能或经常变化的部分使用接口将其封装。在此例中,BusyBankStaff依赖于BankProcess类,将其改为依赖于一个IBankProcess接口

public interface IBankProcess
{
void Process();
}

然后,不同的业务类型都实现该接口

public class DepositProcess:IBankProcess
{
public void Process(){}
}
public class TransferProcess:IBankProcess
{
public void Process(){}
}

等等。

此时BusyBankStaff可以摇身一变,变成EasyBankStaff

public class EasyBankStaff
{
private readonly IBankProcess bankProcess = new BankProcess();
public void HandleProcess(Client client)
{
bankProcess=client.CreateProcess();
bankProcess.Process();
}
}

注意到,业务的分配由银行的业务员转为客户,让客户自己依据自己的类型创建相应的bankProcess对象——这就是现实中客户依据业务类型的不同在排队取号机前取不同的业务号码。

现在,若新增一个业务,只要增加一个实现IBankProcess接口的类即可。

public class BuyFund:IBankProcess
{
public void Process(){}
}

大功告成了吗?不!

若观察系统,我们会发现客户类此时已经转变为

   public class Client
{
public string Name{get;set;}
private string ClientType { get; set; } public IBankProcess CreateProcess()
{
switch(clientType)
{
case"存款用户":
return new DepositProcess();
case "转账用户":
return new TransferProcess(); }
}
}

又是一个违反OCP的类,又是switch的坏味道。

怎么办?同样是抽象!


将client类抽象为接口

public interface IClient
{
IBankProcess CreateProcess();
}

然后不同类型的客户都实现这一接口

public class DepositClient:Client
{
IBankProcess CreateProcess
{
return new DepositProcess();
} }
public class TransferClient:Client
{
IBankProcess CreateProcess
{
return new TransferProcess();
} }

如果有新的客户类型,那么就新增一个这样的类即可。完成!

《你必须知道的.NET》书中对OCP(开放封闭)原则的阐述的更多相关文章

  1. 必须知道的Spring Boot中的一些Controller注解

    这篇文章是抄其他人的,原址:https://cloud.tencent.com/developer/article/1082720 本文旨在向你介绍在Spring Boot中controller中最基 ...

  2. [你必须知道的.NET]第二十六回:认识元数据和IL(下)

    发布日期:2009.03.04 作者:Anytao © 2009 Anytao.com ,Anytao原创作品,转贴请注明作者和出处. 说在,开篇之前 书接上回: 第二十四回:认识元数据和IL(上), ...

  3. MVC中你必须知道的13个扩展点

    MVC中你必须知道的13个扩展点 pasting 转:http://www.cnblogs.com/kirinboy/archive/2009/06/01/13-asp-net-mvc-extensi ...

  4. 解惑《你必须知道的.net》——C#继承关系中【方发表】的创建和调用

    前言: 现在正在读<你必须知道的.net>(第二版)一书,看到IL语言那一章,将call.callvirt和calli时候,书中举了一个例子,是一个三层继承的例子,我一开始看的时候就有点懵 ...

  5. [你必须知道的.NET]第二十五回:认识元数据和IL(中)

    发布日期:2009.02.25 作者:Anytao © 2009 Anytao.com ,Anytao原创作品,转贴请注明作者和出处. 说在,开篇之前 书接上回[第二十四回:认识元数据和IL(上)], ...

  6. 每个项目中,你必须知道的11个Java第三方类库。

    Java第三方library ecosystem是一个很广阔的范畴.不久前有人撰文:每个项目中,你必须知道的11个Java第三方类库. 单元测试 1.DBUnit DBunit是一个基于junit扩展 ...

  7. [你必须知道的.NET]第三十回:.NET十年(下)

    发布日期:2009.05.11 作者:Anytao © 2009 Anytao.com ,Anytao原创作品,转贴请注明作者和出处. /// <summary> /// 本文部分内容,已 ...

  8. [你必须知道的.NET]第二十九回:.NET十年(上)

    发布日期:2009.05.08 作者:Anytao © 2009 Anytao.com ,Anytao原创作品,转贴请注明作者和出处. /// <summary> /// 本文部分内容,已 ...

  9. 从零开始学习jQuery(剧场版) 你必须知道的javascript

    原文:从零开始学习jQuery(剧场版) 你必须知道的javascript 一.摘要 本文是jQuery系列教程的剧场版, 即和jQuery这条主线无关, 主要介绍大家平时会忽略的一些javascri ...

随机推荐

  1. 【网络资料】Astar算法详解

    关于A*算法,很早就想写点什么,可是貌似天天在忙活着什么,可事实又没有做什么,真是浮躁啊!所以今晚还是来写一下总结吧! A*算法是很经典的只能启发式搜索算法,关于只能搜索算法和一般的搜索算法(例如DF ...

  2. [LeetCode] Remove Duplicates from Sorted Array

    Given a sorted array, remove the duplicates in place such that each element appear only once and ret ...

  3. Lingo语法

    基本语法 ! 注释,末尾需要分号 ~ 分隔符 集成员无论用何种字符字符标记,它的索引都是从1开始连续计数 在数据声明中输入两个相连的逗号表示该位置对应的集成员的属性值未知. init: endinit ...

  4. Windows MDL原理总结

    http://blog.csdn.net/tbwood/article/details/5400419 http://www.cnblogs.com/jack204/archive/2011/12/2 ...

  5. 智能车学习(八)——菜单的实现

    一.代码分享 1.头文件 #ifndef __MENU_H #define __MENU_H /***********宏定义************/ //页面声明 typedef enum Menu ...

  6. 如何离线下载Chrome的安装包

    打开Chrome官网(自行搜索)点击下载后下载的是联网安装包,这对部分上网不方便的用户造成了一定的麻烦. http://www.google.cn/chrome/browser/desktop/ind ...

  7. Linux内核创建一个新进程的过程

    “平安的祝福 + 原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 ” 进程在 ...

  8. (转载)Bash 中的特殊字符大全

    转自:https://linux.cn/article-5657-1.html Linux下无论如何都是要用到shell命令的,在Shell的实际使用中,有编程经验的很容易上手,但稍微有难度的是she ...

  9. 使用nginx做负载均衡的session共享问题

    查了一些资料,看了一些别人写的文档,总结如下,实现nginx session的共享PHP服务器有多台,用nginx做负载均衡,这样同一个IP访问同一个页面会被分配到不同的服务器上,如果session不 ...

  10. kylin1.5新特性 new aggregation group

    终于啃完并理解了,我果然弱鸡.new aggregation group,是kylin 1.5的新特性:老版本中的agg是需要选中所有可能被使用的纬度字段,以供查询:但存在高纬度的查询需求,例如查询某 ...