当前位置:  开发笔记 > 编程语言 > 正文

PHP程序61条面向对象分析设计的经验原则

你不必严格遵守这些原则,违背它们也不会被处以宗教刑罚。但你应当把这些原则看成警铃,若违背了其中的一条,那么警铃就会响起。-----ArthurJ.Riel(1)所有数据都应该隐藏在所在的类的内部。(2)类的使用者必须依赖类的共有接口,但类不能依赖它的使用者">

你不必严格遵守这些原则,违背它们也不会被处以宗教刑罚。但你应当把这些原则看成警铃,若违背了其中的一条,那么警铃就会响起 。 ----- Arthur J.Riel

 

(1)所有数据都应该隐藏在所在的类的内部。

 

(2)类的使用者必须依赖类的共有接口,但类不能依赖它的使用者。

 

(3)尽量减少类的协议中的消息。

 

(4)实现所有类都理解的最基本公有接口[例如,拷贝操作(深拷贝和浅拷贝)、相等性判断、正确输出内容、从ASCII描述解析等等]。

 

(5)不要把实现细节(例如放置共用代码的私有函数)放到类的公有接口中。

 

如果类的两个方法有一段公共代码,那么就可以创建一个防止这些公共代码的私有函数。

 

(6)不要以用户无法使用或不感兴趣的东西扰乱类的公有接口。

 

(7)类之间应该零耦合,或者只有导出耦合关系。也即,一个类要么同另一个类毫无关系,要么只使用另一个类的公有接口中的操作。

 

(8)类应该只表示一个关键抽象。

 

包中的所有类对于同一类性质的变化应该是共同封闭的。一个变化若对一个包影响,则将对包中的所有类产生影响,而对其他的包不  造成任何影响 .

 

(9)把相关的数据和行为集中放置。

 

设计者应当留意那些通过get之类操作从别的对象中获取数据的对象。这种类型的行为暗示着这条经验原则被违反了。

 

(10)把不相关的信息放在另一个类中(也即:互不沟通的行为)。

 

朝着稳定的方向进行依赖.

 

(11)确保你为之建模的抽象概念是类,而不只是对象扮演的角色。

 

(12)在水平方向上尽可能统一地分布系统功能,也即:按照设计,顶层类应当统一地共享工作。

 

(13)在你的系统中不要创建全能类/对象。对名字包含Driver、Manager、System、Susystem的类要特别多加小心。

 

规划一个接口而不是实现一个接口。

 

(14)对公共接口中定义了大量访问方法的类多加小心。大量访问方法意味着相关数据和行为没有集中存放。

 

(15)对包含太多互不沟通的行为的类多加小心。

 

这个问题的另一表现是在你的应用程序中的类的公有接口中创建了很多的get和set函数。

 

(16)在由同用户界面交互的面向对象模型构成的应用程序中,模型不应该依赖于界面,界面则应当依赖于模型。

 

(17)尽可能地按照现实世界建模(我们常常为了遵守系统功能分布原则、避免全能类原则以及集中放置相关数据和行为的原则而违背    这条原则) 。

 

(18)从你的设计中去除不需要的类。

 

一般来说,我们会把这个类降级成一个属性。

 

(19)去除系统外的类。

 

系统外的类的特点是,抽象地看它们只往系统领域发送消息但并不接受系统领域内其他类发出的消息。

 

(20)不要把操作变成类。质疑任何名字是动词或者派生自动词的类,特别是只有一个有意义行为的类。考虑一下那个有意义的行为是  否应当迁移到已经存在或者尚未发现的某个类中。

 

(21)我们在创建应用程序的分析模型时常常引入代理类。在设计阶段,我们常会发现很多代理没有用的,应当去除。

 

(22)尽量减少类的协作者的数量。

 

一个类用到的其他类的数目应当尽量少。

 

(23)尽量减少类和协作者之间传递的消息的数量。

 

(24)尽量减少类和协作者之间的协作量,也即:减少类和协作者之间传递的不同消息的数量。

 

(25)尽量减少类的扇出,也即:减少类定义的消息数和发送的消息数的乘积。

 

(26)如果类包含另一个类的对象,那么包含类应当给被包含的对象发送消息。也即:包含关系总是意味着使用关系。

 

(27)类中定义的大多数方法都应当在大多数时间里使用大多数数据成员。

 

(28)类包含的对象数目不应当超过开发者短期记忆的容量。这个数目常常是6。

 

当类包含多于6个数据成员时,可以把逻辑相关的数据成员划分为一组,然后用一个新的包含类去包含这一组成员。

 

(29)让系统功能在窄而深的继承体系中垂直分布。

 

(30)在实现语义约束时,最好根据类定义来实现。这常常会导致类泛滥成灾,在这种情况下,约束应当在类的行为中实现,通常是在  构造函数中实现,但不是必须如此。

 

(31)在类的构造函数中实现语义约束时,把约束测试放在构造函数领域所允许的尽量深的包含层次中。

 

(32)约束所依赖的语义信息如果经常改变,那么最好放在一个集中式的第3方对象中。

 

(33)约束所依赖的语义信息如果很少改变,那么最好分布在约束所涉及的各个类中。

 

(34)类必须知道它包含什么,但是不能知道谁包含它。

 

(35)共享字面范围(也就是被同一个类所包含)的对象相互之间不应当有使用关系。

 

(36)继承只应被用来为特化层次结构建模。

 

(37)派生类必须知道基类,基类不应该知道关于它们的派生类的任何信息。

 

(38)基类中的所有数据都应当是私有的,不要使用保护数据。

 

类的设计者永远都不应该把类的使用者不需要的东西放在公有接口中。

 

(39)在理论上,继承层次体系应当深一点,越深越好。

 

(40)在实践中,继承层次体系的深度不应当超出一个普通人的短期记忆能力。一个广为接受的深度值是6。

 

(41)所有的抽象类都应当是基类。

 

(42)所有的基类都应当是抽象类。

 

(43)把数据、行为和/或接口的共性尽可能地放到继承层次体系的高端。

 

(44)如果两个或更多个类共享公共数据(但没有公共行为),那么应当把公共数据放在一个类中,每个共享这个数据的类都包含这个类。

 

(45)如果两个或更多个类有共同的数据和行为(就是方法),那么这些类的每一个都应当从一个表示了这些数据和方法的公共基类继承。

 

(46)如果两个或更多个类共享公共接口(指的是消息,而不是方法),那么只有他们需要被多态地使用时,他们才应当从一个公共基类  继承。

 

(47)对对象类型的显示的分情况分析一般是错误的。在大多数这样的情况下,设计者应当使用多态。

 

(48)对属性值的显示的分情况分析常常是错误的。类应当解耦合成一个继承层次结构,每个属性值都被变换成一个派生类。

 

(49)不要通过继承关系来为类的动态语义建模。试图用静态语义关系来为动态语义建模会导致在运行时切换类型。

 

(50)不要把类的对象变成派生类。对任何只有一个实例的派生类都要多加小心。

 

(51)如果你觉得需要在运行时刻创建新的类,那么退后一步以认清你要创建的是对象。现在,把这些对象概括成一个类。

 

(52)在派生类中用空方法(也就是什么也不做的方法)来覆写基类中的方法应当是非法的。

 

(53)不要把可选包含同对继承的需要相混淆。把可选包含建模成继承会带来泛滥成灾的类。

 

(54)在创建继承层次时,试着创建可复用的框架,而不是可复用的组件。

 

(55)如果你在设计中使用了多重继承,先假设你犯了错误。如果没犯错误,你需要设法证明。

 

(56)只要在面向对象设计中用到了继承,问自己两个问题:(1)派生类是否是它继承的那个东西的一个特殊类型?(2)基类是不是派生类的一部分?

 

(57)如果你在一个面向对象设计中发现了多重继承关系,确保没有哪个基类实际上是另一个基类的派生类。

 

(58)在面向对象设计中如果你需要在包含关系和关联关系间作出选择,请选择包含关系。

 

(59)不要把全局数据或全局函数用于类的对象的薄记工作。应当使用类变量或类方法。

 

(60)面向对象设计者不应当让物理设计准则来破坏他们的逻辑设计。但是,在对逻辑设计作出决策的过程中我们经常用到物理设计准则。

 

(61)不要绕开公共接口去修改对象的状态。


推荐阅读
  • Web自动化测试:表单提交与页面跳转的高效实现
    Web自动化测试:表单提交与页面跳转的高效实现 ... [详细]
  • Issue with the Reserved Term HOSTS in System Configuration ... [详细]
  • 本文首先对信息漏洞的基础知识进行了概述,重点介绍了几种常见的信息泄露途径。具体包括目录遍历、PHPINFO信息泄露以及备份文件的不当下载。其中,备份文件下载涉及网站源代码、`.bak`文件、Vim缓存文件和`DS_Store`文件等。目录遍历漏洞的详细分析为后续深入研究奠定了基础。 ... [详细]
  • React组件是构成用户界面的基本单元,每个组件都封装了特定的功能和逻辑,具备高度的独立性和可复用性。通过将不同大小和功能的组件组合在一起,可以构建出复杂且功能丰富的页面,类似于拼图游戏中的各个部分,最终形成一个完整的视觉效果。 ... [详细]
  • 在面对不确定性的挑战时,卓越的操作者通常会采用七大策略来有效管理和减轻风险,这些策略同样适用于职业发展和个人生活。具体而言,这七大风险管理策略包括:1. 克服恐惧心理卓越的操作者能够正视并克服内心的恐惧,保持冷静和理性,从而做出更加明智的决策。这一能力不仅有助于在市场波动中保持稳定,也能在职业生涯和个人生活中发挥重要作用。 ... [详细]
  • 《题画山水屏风》译文与原文赏析:唐代诗人张九龄的艺术解读 ... [详细]
  • 《已亥杂诗 第139首》译文与原文鉴赏:清代文学家龚自珍的诗歌艺术探析 ... [详细]
  • 配置未被正确应用:导航栏图标设置失效问题分析与解决 ... [详细]
  • 新年伊始,正是学习的最佳时机。本文全面解析了CK1957-Zookeeper的核心概念与实践技巧,旨在帮助初学者快速掌握这一深度学习工具。通过详细的理论讲解和实际操作示例,读者可以更好地理解Zookeeper的工作原理及其在分布式系统中的应用。无论是新手还是有一定基础的学习者,都能从中受益匪浅。 ... [详细]
  • 微信支付授权目录配置详解及操作步骤
    在使用微信支付时,若通过WeixinJSBridge.invoke方法调用支付功能,可能会遇到“当前页面URL未注册”的错误提示,导致get_brand_wcpay_request:fail调用微信JSAPI支付失败。为解决这一问题,需要正确配置微信支付授权目录,确保支付页面的URL已成功注册。本文将详细介绍微信支付授权目录的配置步骤和注意事项,帮助开发者顺利完成支付功能的集成与调试。 ... [详细]
  • 探究Oracle数据库字符集编码的详细方法与实践
    本文深入探讨了Oracle数据库字符集编码的详细方法与实践。首先,通过执行 `SELECT USERENV('language') FROM DUAL;` 查询服务端字符集编码。其次,通过在注册表中搜索 `NLS_LANG` 参数来查看客户端字符集编码。此外,文章还介绍了如何在不同场景下正确配置和转换字符集,以确保数据的一致性和完整性。 ... [详细]
  • 如何使用Python高效绘制矩形图形
    本文详细介绍了如何利用Python的Turtle库高效绘制矩形图形,适合初学者快速上手。通过具体示例代码,帮助读者理解Turtle库的基本绘图方法和技巧,同时探讨了在不同应用场景中绘制矩形的实际操作,为后续复杂图形的绘制打下坚实基础。 ... [详细]
  • 如何高效使用“叫我起床”应用程序:基础操作指南
    对于可能还不熟悉“叫我起床”应用程序的用户,本文将详细介绍其基本操作方法,帮助您快速上手。通过本指南,您可以轻松掌握如何设置闹钟、调整音量和选择唤醒模式等核心功能,确保每天都能精神饱满地迎接新一天的挑战。 ... [详细]
  • 本文介绍了如何通过掌握 IScroll 技巧来实现流畅的上拉加载和下拉刷新功能。首先,需要按正确的顺序引入相关文件:1. Zepto;2. iScroll.js;3. scroll-probe.js。此外,还提供了完整的代码示例,可在 GitHub 仓库中查看。通过这些步骤,开发者可以轻松实现高效、流畅的滚动效果,提升用户体验。 ... [详细]
  • Laravel 与 Vue 实现前后端分离时的跨域解决方案探析
    Laravel 与 Vue 实现前后端分离时的跨域解决方案探析 ... [详细]
author-avatar
幸福的小馋豆
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有