面向对象设计原则之开闭原则

194 篇文章 12 订阅
189 篇文章 394 订阅

      开闭原则是面向对象的可复用设计的第一块基石,它是最重要的面向对象设计原则。开闭原则由Bertrand  Meyer1988年提出,其定义如下:

开闭原则(Open-Closed Principle, OCP):一个软件实体应当对扩展开放,对修改关闭。即软件实体应尽量在不修改原有代码的情况下进行扩展。

      在开闭原则的定义中,软件实体可以指一个软件模块、一个由多个类组成的局部结构或一个独立的类

      任何软件都需要面临一个很重要的问题,即它们的需求会随时间的推移而发生变化。当软件系统需要面对新的需求时,我们应该尽量保证系统的设计框架是稳定的。如果一个软件设计符合开闭原则,那么可以非常方便地对系统进行扩展,而且在扩展时无须修改现有代码,使得软件系统在拥有适应性和灵活性的同时具备较好的稳定性和延续性。随着软件规模越来越大,软件寿命越来越长,软件维护成本越来越高,设计满足开闭原则的软件系统也变得越来越重要。

      为了满足开闭原则,需要对系统进行抽象化设计,抽象化是开闭原则的关键。在JavaC#等编程语言中,可以为系统定义一个相对稳定的抽象层,而将不同的实现行为移至具体的实现层中完成。在很多面向对象编程语言中都提供了接口、抽象类等机制,可以通过它们定义系统的抽象层,再通过具体类来进行扩展。如果需要修改系统的行为,无须对抽象层进行任何改动,只需要增加新的具体类来实现新的业务功能即可,实现在不修改已有代码的基础上扩展系统的功能,达到开闭原则的要求。

      Sunny软件公司开发的CRM系统可以显示各种类型的图表,如饼状图和柱状图等,为了支持多种图表显示方式,原始设计方案如图1所示:

初始设计方案结构图

      在ChartDisplaydisplay()方法中存在如下代码片段:

......
if (type.equals("pie")) {
    PieChart chart = new PieChart();
    chart.display();
}
else if (type.equals("bar")) {
    BarChart chart = new BarChart();
    chart.display();
}
......

      在该代码中,如果需要增加一个新的图表类,如折线图LineChart,则需要修改ChartDisplay类的display()方法的源代码,增加新的判断逻辑,违反了开闭原则。

      现对该系统进行重构,使之符合开闭原则。

       在本实例中,由于在ChartDisplay类的display()方法中针对每一个图表类编程,因此增加新的图表类不得不修改源代码。可以通过抽象化的方式对系统进行重构,使之增加新的图表类时无须修改源代码,满足开闭原则。具体做法如下:

      (1) 增加一个抽象图表类AbstractChart,将各种具体图表类作为其子类;

      (2)  ChartDisplay类针对抽象图表类进行编程,由客户端来决定使用哪种具体图表。

      重构后结构如图2所示:

图2 重构后的结构图

      在图2中,我们引入了抽象图表类AbstractChart,且ChartDisplay针对抽象图表类进行编程,并通过setChart()方法由客户端来设置实例化的具体图表对象,在ChartDisplaydisplay()方法中调用chart对象的display()方法显示图表。如果需要增加一种新的图表,如折线图LineChart,只需要将LineChart也作为AbstractChart的子类,在客户端向ChartDisplay中注入一个LineChart对象即可,无须修改现有类库的源代码。     

       注意:因为xmlproperties等格式的配置文件是纯文本文件,可以直接通过VI编辑器或记事本进行编辑,且无须编译,因此在软件开发中,一般不把对配置文件的修改认为是对系统源代码的修改。如果一个系统在扩展时只涉及到修改配置文件,而原有的Java代码或C#代码没有做任何修改,该系统即可认为是一个符合开闭原则的系统。

 【作者:刘伟  http://blog.csdn.net/lovelion
  • 100
    点赞
  • 95
    收藏
    觉得还不错? 一键收藏
  • 48
    评论
C#面向对象设计原则主要包括以下几个方面: 1. 单一职责原则(Single Responsibility Principle, SRP): 一个类或模块应该有且只有一个责任。这意味着一个类或模块应该只有一个引起它变化的原因。通过遵守SRP,可以使类或模块更加高内聚,易于维护和扩展。 2. 开闭原则(Open-Closed Principle, OCP): 软件实体(类、模块、函数等)应该对扩展开放,对修改关闭。这意味着一个软件实体应该通过扩展来实现变化,而不是通过修改已有的代码来实现。通过遵守OCP,可以使系统更加稳定,易于扩展和维护。 3. 里氏替换原则(Liskov Substitution Principle, LSP): 所有引用父类的地方必须能够透明地使用其子类的对象,即子类对象必须能够替换父类对象而不影响程序的正确性。遵守LSP可以提高代码的可靠性和可复用性。 4. 接口隔离原则(Interface Segregation Principle, ISP): 不应该强迫客户端依赖于它们不使用的接口。接口应该尽量小而专门,客户端只需依赖于他们需要的接口。通过遵守ISP,可以减少类之间的耦合度,提高代码的灵活性和可维护性。 5. 依赖倒置原则(Dependency Inversion Principle, DIP): 高层模块不应该依赖于低层模块,二者都应该依赖于抽象。抽象不应该依赖于具体实现,具体实现应该依赖于抽象。遵守DIP可以提高代码的可测试性和可扩展性。 6. 迪米特法则(Law of Demeter, LoD): 一个对象应该对其他对象有尽可能少的了解,只与直接的朋友通信。遵守LoD可以减少类之间的依赖关系,提高代码的复用性和可维护性。 7. 合成复用原则(Composition/Aggregation Reuse Principle, CARP): 尽量使用对象组合和聚合,而不是继承来达到代码重用的目的。通过遵守CARP,可以使系统更加灵活,易于扩展和维护。 以上是C#面向对象设计原则,遵守这些原则可以帮助开发人员编写出更具灵活性、易维护性的代码。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [C# 面向对象的基本原则](https://download.csdn.net/download/weixin_38704485/12818402)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [C#面向对象设计模式纵横谈(视频与源码)](https://download.csdn.net/download/Lancelot_Liu/18469133)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [C#面向对象设计的七大原则](https://download.csdn.net/download/weixin_38697328/12797679)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 48
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值