高质量编程的金玉良言——开放-封闭原则
2015-08-31 13:28
232 查看
什么是“开放-封闭”原则?
软件的类、模块、函数等交付后,如果需要增加功能,不要去修改原来的代码,而是通过新增加类的方法去扩展功能。
所谓的“开放”就是指开放接口,也就是通过增加子类的方法来增加功能;
“封闭”指的是封闭修改,也就是代码一旦提交,就不要再去修改它。
总而言之,“开放-封闭”原则的意思就是:在我们设计一个类的时候要尽量把这个类的功能考虑全面,然后一旦写好了交付给第三方使用之后就不要再去修改。如果确实需要给一个类增加功能,我们增加一个含有新功能的类,原来的代码不要动它。
但是,在实际开发中,“封闭”是相对的,一个类写完后并不能确保它未来永远都不会被修改。因此,在设计类的时候,应该要“猜”这个类未来会作哪些扩展,然后把这些扩展抽象出一个公共父类,父类中有这些子类都会执行的函数,只不过不同的子类函数实现不一样;这样,增加新功能时只要新建一个类,继承父类,覆盖父类中的函数,然后客户端需要使用新功能时,只要将新的子类对象赋给父类引用,再通过父类引用调用函数即可。
何时应对变化?
我们在设计或开发类的时候如果没有将未来可能扩展的功能考虑到,这没关系,只要在开发或新增需求的时候,一旦发现需要扩展功能并且需要修改原来代码,我们就立刻给这个类抽象出父类,让原本的功能和新增的功能都继承这个父类。
比如,我们一开始拿到需求是这样的:
开发一个软件,营业员只要输入原价,就能根据“满100返50”计算出折扣以后的价钱。
我们只需要创建一个折扣类,该类里面有一个打折函数,这个函数需要输入原价,然后它会按照“满100返50”的规则计算出折后价。
过了一阵子后,超市的需求改了:
由于超市隔三差五就要办促销活动,但每次促销活动的方式不一样,所以需要开发一个软件,营业员只要输入原价再选择活动类型后,就能计算出折扣以后的价钱。
此时我们发现,折扣的形式有多种多样,而现在我们的折扣类中只提供了“满100返50”的打折方式,如果要增加其他打折方式就要往折扣类里面增加新的打折方式,如果过了一阵子又要增加打折方式,就又要在折扣类里面增加代码,这就违背了“封闭修改”的原则。
正确的做法是,当我们发现需要增加打折功能的时候,我们将抽象出一个打折类,这个类中有一个抽象函数:discount(),然后让各种不同的打折方式类继承这个类,并重写各自的打折方式。类这样设计之后,当超市又要增加新的打折方式的时候,我们只需要创建一个新的打折类,去继承打折父类,并重写打折函数;然后客户端就可以调用这个新的功能了。
软件的类、模块、函数等交付后,如果需要增加功能,不要去修改原来的代码,而是通过新增加类的方法去扩展功能。
所谓的“开放”就是指开放接口,也就是通过增加子类的方法来增加功能;
“封闭”指的是封闭修改,也就是代码一旦提交,就不要再去修改它。
总而言之,“开放-封闭”原则的意思就是:在我们设计一个类的时候要尽量把这个类的功能考虑全面,然后一旦写好了交付给第三方使用之后就不要再去修改。如果确实需要给一个类增加功能,我们增加一个含有新功能的类,原来的代码不要动它。
但是,在实际开发中,“封闭”是相对的,一个类写完后并不能确保它未来永远都不会被修改。因此,在设计类的时候,应该要“猜”这个类未来会作哪些扩展,然后把这些扩展抽象出一个公共父类,父类中有这些子类都会执行的函数,只不过不同的子类函数实现不一样;这样,增加新功能时只要新建一个类,继承父类,覆盖父类中的函数,然后客户端需要使用新功能时,只要将新的子类对象赋给父类引用,再通过父类引用调用函数即可。
何时应对变化?
我们在设计或开发类的时候如果没有将未来可能扩展的功能考虑到,这没关系,只要在开发或新增需求的时候,一旦发现需要扩展功能并且需要修改原来代码,我们就立刻给这个类抽象出父类,让原本的功能和新增的功能都继承这个父类。
比如,我们一开始拿到需求是这样的:
开发一个软件,营业员只要输入原价,就能根据“满100返50”计算出折扣以后的价钱。
我们只需要创建一个折扣类,该类里面有一个打折函数,这个函数需要输入原价,然后它会按照“满100返50”的规则计算出折后价。
过了一阵子后,超市的需求改了:
由于超市隔三差五就要办促销活动,但每次促销活动的方式不一样,所以需要开发一个软件,营业员只要输入原价再选择活动类型后,就能计算出折扣以后的价钱。
此时我们发现,折扣的形式有多种多样,而现在我们的折扣类中只提供了“满100返50”的打折方式,如果要增加其他打折方式就要往折扣类里面增加新的打折方式,如果过了一阵子又要增加打折方式,就又要在折扣类里面增加代码,这就违背了“封闭修改”的原则。
正确的做法是,当我们发现需要增加打折功能的时候,我们将抽象出一个打折类,这个类中有一个抽象函数:discount(),然后让各种不同的打折方式类继承这个类,并重写各自的打折方式。类这样设计之后,当超市又要增加新的打折方式的时候,我们只需要创建一个新的打折类,去继承打折父类,并重写打折函数;然后客户端就可以调用这个新的功能了。
相关文章推荐
- 很让人受教的提高php代码质量的方法
- Spring AOP编程中--@aspect 标签,切面编程
- Qt编译安装方法详解:包含数据库驱动的编译安装方法
- 07_集合
- 2016阿里巴巴java笔试题
- 06_IO输入输出流
- spring.ftl文件不明白--Spring整合FreeMarker本地化动态设置
- InputStreamReader和OutputStreamWriter的用法
- 05_常用API
- quartz Cron自动生成的 相关代码
- Python多线程
- 04_面向对象概述
- JAVA字符串格式化-String.format()的使用
- 03_java数组和冒泡排序
- 64位Win8环境的Python数值分析库安装(原创)
- Lua5.0 语法解析之路
- 02_java基础语法
- C也可以通过精心封装某些函数功能实现重用,那C++的类有什么优点吗(从面向对象的三大属性进行分析)
- C# 之 Int16 Int32 Int64 的区别
- Java学习集锦