【设计模式】之开闭原则(OCP)
开闭原则是面向对象设计的一个重要原则,其定义如下:
开闭原则(Open-Closed Principle, OCP):一个软件实体应当对扩展开放,对修改关闭。即软件实体应尽量在不修改原有代码的情况下进行扩展。
在软件的生命周期内,因为变化、升级和维护等原因需要对软件原有代码进行修改时,可能会给旧代码中引入错误,也可能会使我们不得不对整个功能进行重构,并且需要原有代码经过重新测试。那么势必会对软件的开发带来额外的风险和成本, 这是OCP原则要规范设计的一个主要原因,所有的设计原则都是对软件的开发,设计以及维护带来好处的,OCP也不例外。
OCP原则是面形对象软件设计的基本原则,其旨在指导如何构建稳定的,灵活的,易于维护的软件。其实这个原则也是我们面向对象设计的一个终极要求,更是一个引导,在软件设计过程中要达到OCP原则其实你需要很好的遵守其他的设计原则,换句话说如果其它的设计原则都达到了那么OCP原则自然就符合了,也就是说OCP原则是其他原则综合使用的一个考量,一个检验。
假如我们要设计一个叫做动物的类(Animal)在这个类中我们有一个方法叫Sound, Sound 方法主要用于发出动物的叫声,通常我们的设计代码如下:
public class Animal
{
public void Sound(string animal)
{
switch (animal)
{
case "dog":
System.Console.WriteLine("woof woof woof...");
break;
case "cat":
Console.WriteLine("miaow miaow miaow...");
break;
}
}
}
客户端的调用代码如下:
class Program
{
static void Main(string[] args)
{
Animal animal=new Animal();
animal.Sound("dog");
Console.ReadKey();
}
}
调用返回的结果:

这样看起来似乎很完美,如果想要什么动物发生客户端就传入该动物的名字然后调用Sound方法就可以了。 客户今天只养了两种动物,狗和猫,如果有一天他在养一头羊,他想听到羊的叫声怎么办呢? 直接的想法是在Sound的方法中加一个case子句,写上羊的叫声如下:
public class Animal
{
public void Sound(string animal)
{
switch (animal)
{
case "dog":
System.Console.WriteLine("woof woof woof...");
break;
case "cat":
Console.WriteLine("miaow miaow miaow...");
break;
case "sheep":
Console.WriteLine("mee-mee mee-mee mee-mee...");
break;
}
}
}
客户端调用如下:
static void Main(string[] args)
{
Animal animal=new Animal();
animal.Sound("sheep");
Console.ReadKey();
}
输出:

这看起来似乎是很完美,但是我们回过头想一下,好像哪里不对劲,如果后面客户需要加更多的动物怎么办呢?,是不是这个case要写很长很长,Sound方法要每次都要修改,每次都要全部编译整个工程还要重新部署所有的代码,这中间的风险很大,很容易出现操作上的失误,或者代码修改出现bug,要命的是每次都要把整个代码重新测试一遍,给升级带来了很多的工作量,以及潜在的风险。其实再回头看看,我们这个设计是违反OCP原则的, OCP告诉我们对“修改关闭,对扩展开放“,很显然我们这里修改了代码。同时也违背了SRP原则“一个类或方法只负责干一件事情“,显然Sound 犯法的职责太多。那么我们有没有办法来重构代码,让其遵守这些原则,每次修改该最少的代码即尽可能的减少工作量呢? 答案是肯定的。
我们抽取一个接口叫IAnimal:
public interface IAnimal
{
void Sound();
}
再分别定义三个类 Dog, Cate 和Sheep 并继承IAnimal 接口:
public class Dog : IAnimal
{
public void Sound()
{
Console.WriteLine("woof woof woof...");
}
} public class Cat : IAnimal
{
public void Sound()
{
Console.WriteLine("miaow miaow miaow...");
}
} public class Sheep:IAnimal
{
public void Sound()
{
Console.WriteLine("mee-mee mee-mee mee-mee...");
}
}
客户端如果想听到狗的叫声的代码调用如下:
static void Main(string[] args)
{
IAnimal animal=new Dog();
animal.Sound(); Console.ReadKey();
}
输出:

这下是不是比开始好了很多,并且他还很好的满足了单一职责原则(SRP),每个类只负责一种动物发出的声音,职责单一了, 但是我们发现如果我们想听到猫的叫声还是要修改Main方法中的调用代码, 还要编译部署,风险还是有点大,工作量还是有点大,那么我们能不能不修改代码只需要改个配置来达到修改Main方法调用的结果呢?这样每次就不用编译只需要修改一下配置就好了呢? 答案是肯定的, 我们利用反射加配置就可以了。 这里我们先加一个工具类用于反射。代码如下:
public class ObjectBuildFactory<T>
{
public static T Instance(string key)
{
Type obj = Type.GetType(key);
if (obj == null) return default(T); T factory = (T)obj.Assembly.CreateInstance(obj.FullName); return factory;
}
}
写配置文件如下:
<appSettings>
<add key="Animal" value="ConsoleApp1.Dog"/>
</appSettings>
调用并通过反射创建对象,调用Dog的叫声如下:
static void Main(string[] args)
{
string key = ConfigurationManager.AppSettings["Animal"]; IAnimal animal = ObjectBuildFactory<IAnimal>.Instance(key); animal.Sound(); Console.ReadKey();
}
输出:

好了如果希望听到羊的叫声,只需要改一下我们的配置文件就可以了:
<appSettings>
<add key="Animal" value="ConsoleApp1.Sheep"/>
</appSettings>

其它的代码不需要任何修改直接运行输出如下:

好了这回满足OCP了。
那么好了如果客户期望在增加一种动物,我们应该怎么办呢? 这下就变得非常简单了,我们需要如下两个步骤来完成:
1.增加一个类继承IAnimal接口并实现Sound方法。
2.修改配置文件。
例如我们增加一个动物鸭子代码如下:
public class Duck : IAnimal
{
public void Sound()
{
Console.WriteLine("quack quack quack...");
}
}
配置:
<appSettings>
<add key="Animal" value="ConsoleApp1.Duck"/>
</appSettings>
输出:

很简单达到了我们的设计目的。
总结:开闭原则(OCP)是我们在面向对象设计过程中必须注入潜意识的一个原则,在设计的过程中要时时刻刻,如影随形,一旦发现违背就要立即重构,不然代码就会变的越来越不易于理解,越来越不易于维护了。
【设计模式】之开闭原则(OCP)的更多相关文章
- C#软件设计——小话设计模式原则之:开闭原则OCP
前言:这篇继续来看看开闭原则.废话少说,直接入正题. 软件设计原则系列文章索引 C#软件设计——小话设计模式原则之:依赖倒置原则DIP C#软件设计——小话设计模式原则之:单一职责原则SRP C#软件 ...
- 设计原则:开闭原则(OCP)
1.什么是开闭原则 开闭原则的英文是Open Closed Principle,缩写就是OCP.其定义如下: 软件实体(模块.类.方法等)应该"对扩展开放.对修改关闭". 从定义上 ...
- 7.10 其他面向对象设计原则1: 开-闭原则OCP
其他面向对象设计原则1: 开-闭原则OCP Open-Closed Principle (OCP)5.1 设计变坏的前兆 Signs of Rotting Design 僵硬性 Rigidit ...
- 深入理解JavaScript系列(7):S.O.L.I.D五大原则之开闭原则OCP
前言 本章我们要讲解的是S.O.L.I.D五大原则JavaScript语言实现的第2篇,开闭原则OCP(The Open/Closed Principle ). 开闭原则的描述是: Software ...
- 聊一聊开闭原则(OCP).
目录 简述 最早提出(梅耶开闭原则) 重新定义(多态开闭原则) 深入探讨 OCP的两个特点 对外扩展开放(Open for extension) 对内修改关闭 抽象 关闭修改.对外扩展? 简述 在面向 ...
- 【面向对象设计原则】之开闭原则(OCP)
开闭原则是面向对象设计的一个重要原则,其定义如下: 开闭原则(Open-Closed Principle, OCP):一个软件实体应当对扩展开放,对修改关闭.即软件实体应尽量在不修改原有代码的情况下进 ...
- JavaScript 开闭原则OCP
代码: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3. ...
- 设计模式笔记:开闭原则(OCP,The Open-Closed Principle)
1. 开闭原则概述 开闭原则(OCP,The Open-Closed Principle)两个主要特征: (1)对扩展开放(open for extension):模块的行为的可以扩展的,当应用的需求 ...
- 设计模式原则(6)--Open-Closed Principle(OCP)--开闭原则
作者QQ:1095737364 QQ群:123300273 欢迎加入! 1.定义: 一个软件实体应当对扩展开放,对修改关闭.即软件实体应尽量在不修改原有代码的情况下进行扩展. 2.使用场 ...
随机推荐
- 手机自动化测试:appium源码分析之bootstrap十三
手机自动化测试:appium源码分析之bootstrap十三 poptest(www.poptest.cn)是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开 ...
- linux从入门到精通学习-NFS
NFS网络文件系统 功能 nfs[network file system] 网络文件系统 是FreBSD系统支持的一种系统,允许在网络 上与其它人共享使用文件或文件夹 采用C/S模式 端口号 在vim ...
- 基于MVC和Bootstrap的权限框架解决方案 二.添加增删改查按钮
上一期我们已经搭建了框架并且加入了列表的显示, 本期我们来加入增删改查按钮 整体效果如下 HTML部分,在HTML中找到中意的按钮按查看元素,复制HTML代码放入工程中 <a class=&qu ...
- 将 JSP 中数组传递给 js
<% String[] name = { "w ", "a ", "n ", "g"}; % ...
- memcached分布式缓存
1.memcached分布式简介 memcached虽然称为“分布式”缓存服务器,但服务器端并没有“分布式”功能.Memcache集群主机不能够相互通信传输数据,它的“分布式”是基于客户端的程序逻辑算 ...
- 最近公共祖先LCA(Tarjan算法)的思考和算法实现
LCA 最近公共祖先 Tarjan(离线)算法的基本思路及其算法实现 小广告:METO CODE 安溪一中信息学在线评测系统(OJ) //由于这是第一篇博客..有点瑕疵...比如我把false写成了f ...
- IOS——触摸事件 视图检测和事件传递
iPhone上有非常流畅的用户触摸交互体验,能检测各种手势:点击,滑动,放大缩小,旋转.大多数情况都是用UI*GestureRecognizer这样的手势对象来关联手势事件和手势处理函数.也有时候,会 ...
- java面试题(一)
1.面向对象的特征有哪些方面? 答:面向对象的特征主要有以下几个方面: - 抽象:抽象是将一类对象的共同特征总结出来构造类的过程,包括数据抽象和行为抽象两方面.抽象只关注对象有哪些属性和行为,并不关注 ...
- HTTP 协议
HTTP 协议对应 Web 开发者来说都必须要了解的,无论技术背景或首选编程语言是什么,"请求-响应" 对话是驱动 Web 上通信的基础. HTTP 概述 HTTP 协议是 Hyp ...
- bzoj2876 [Noi2012]骑行川藏
Description 蛋蛋非常热衷于挑战自我,今年暑假他准备沿川藏线骑着自行车从成都前往拉萨.川藏线的沿途有着非常美丽的风景,但在这一路上也有着很多的艰难险阻,路况变化多端,而蛋蛋的体力十分有限,因 ...