设计模式设计模式七大原则.ppt
设计模式02,郑州大学软件学院 赵哲,主要内容,设计模式准备:7大设计原则,郑州大学软件学院 赵哲,七大设计原则,单一职责原则开-闭原则里氏代换原则依赖倒转原则接口隔离原则合成/聚合复用原则迪米特原则,郑州大学软件学院 赵哲,单一职责原则,一个类只有一种职责SRP Single Responsibility Principle prinspl原则:若想到多于一个的动机去改变一个类,那么这个类就具有多于一个的职责软件设计真正要做的许多内容,就是发现职责并且把职责相互分离为什么?高内聚、低耦合、多复用,郑州大学软件学院 赵哲,一个例子,郑州大学软件学院 赵哲,思考问题,这个类中有几个职责?用户是否登陆的方法跟用户玩游戏的方法是否有关?获得用户信息跟其他两个职责是否有关?如何优化代码?,郑州大学软件学院 赵哲,优化完成UML类图,郑州大学软件学院 赵哲,内聚高还是低?耦合度呢?,开-闭原则,对扩展开放、对更改封闭Open for extension Closed for modification开放-封闭原则是面向对象设计的核心所在做一个例子:写个代码片段:完成对两个数字的加分和减法,郑州大学软件学院 赵哲,开-闭原则,大部分人的做法:,郑州大学软件学院 赵哲,若要添加乘法和除法呢?,开-闭原则,郑州大学软件学院 赵哲,符合开闭原则,里氏代换原则,Barbara Liskow 在1988年发表的1972年Barbara成为麻省理工学院的教授是美国获得计算机科学博士学位的第一人,郑州大学软件学院 赵哲,11,里氏代换原则,Liskow Substitution Principle(LSP):子类必须能够替换掉它们的父类型只要是基类出现的地方,一定能够出现子类!LSP指导继承,是继承的基石龙生龙,凤生凤,老鼠的儿子会打洞。,郑州大学软件学院 赵哲,12,LSP,例子:一个例子:长方形和正方形某软件 while(长=宽),将宽+1.父类没问题,但子类就回出现死循环正方形不可以继承长方形,郑州大学软件学院 赵哲,13,LSP,例子孙悟空勾掉生死簿中所有猴子的名字,那么猕猴、石猴、长臂猿等的名字是否都勾掉了?乘马,乘白马、乘骊马注意LSP反之不成立!墨子 小取娣,美人也,爱娣,非爱美人也.盗,人也;恶盗,非恶人也,郑州大学软件学院 赵哲,14,一个例子,郑州大学软件学院 赵哲,依赖,虚线普通箭头表示依赖依赖和关联的区别A和B关联,A只认识B,知道B的属性和方法,可以双关联,但不建议使用。A依赖BB的改变影响A的结果。通常依赖是单向的,郑州大学软件学院 赵哲,郑州大学软件学院 赵哲,如果父类的某些方法在子类中已经发生畸变,则建议断开父子关系!,郑州大学软件学院 赵哲,总结LSP,拒绝子类有自己的个性一旦子类有了“个性”,则与父类之间的关系难以调和会让代码的耦合变得扑朔迷离,郑州大学软件学院 赵哲,依赖倒转原则,要针对接口编程,不要针对实现编程。Program to an interface,not an implementation.高层模块不应该依赖底层模块,两者应该都依赖抽象High-level module should not depend upon bottom module.both should depend upon abstractions,郑州大学软件学院 赵哲,20,依赖倒转原则,减少类之间的耦合度降低风险一个例子司机开车,郑州大学软件学院 赵哲,依赖倒转原则,司机还要开宝马,如何实现?增加宝马类,但是需要修改司机类,郑州大学软件学院 赵哲,Driver,driver(Benz bz)driver(BMW bmw),如何解决,针对接口编程,郑州大学软件学院 赵哲,实现场景代码:,张三开奔驰IDriver 张3=new Driver();ICar benz=new Benz();张3.driver(benz);,郑州大学软件学院 赵哲,练习,最初:光明农场养牛扩展:多种牛场、喂养多种动物最初类图如下,按照依赖倒转原则画出正确类图,郑州大学软件学院 赵哲,接口隔离原则,Interface Segregation Principle ISP客户端不应该依赖它不需要的方法类间的依赖关系应该建立在最小的接口上,郑州大学软件学院 赵哲,一个例子,一个接口:内容是雇佣者有吃饭和做工作的方法雇佣的男、女都可以使用该接口有一天,雇佣者来了一个机器人。机器人不吃饭,郑州大学软件学院 赵哲,例子-错误,Interface IWorkervoid eat();void work();Class Worker:IWorker public void eat()/eat 相关代码 public void work()/work相关代码,郑州大学软件学院 赵哲,28,ISP,例子的缺陷胖接口,造成了对接口的污染如何解决?客户端不应该依赖它不需要的方法 类间的依赖关系应该建立在最小的接口上 一个接口只做一件事,郑州大学软件学院 赵哲,29,例子-符合ISP,Interface IWorkerEat void eat();Interface IWorkerWorkvoid work();Class Worker:IWorkerEat,IWorkerWork public void eat()/eat 相关代码 public void work()/work相关代码Class Robot:IWorkerWork public void work()/work相关代码,郑州大学软件学院 赵哲,30,面向接口编程,而非实现编程,ISP,胖接口容易导致哑方法瘦接口更健康!,郑州大学软件学院 赵哲,31,合成/聚合复用原则,Composite/Aggregate Reuse PrincipleCARP 合成和聚合合成是强烈的关联,部分(将)和整体(主)生命周期一样,整体(主)对部分(将)有支配权,包括创建和销毁。聚合是部分和整体的普通关联,郑州大学软件学院 赵哲,合成和聚合,刘邦和韩信,郑州大学软件学院 赵哲,33,CARP原则,合成:class Personprivate Hand hand;public Person()hand=new Hand();,郑州大学软件学院 赵哲,34,CARP原则,聚合:class Person private Computer computer;private A a;public setComputer()computer=new Computer();public setA()a=new a();,郑州大学软件学院 赵哲,35,CARP原则,尽量使用合成/聚合,而不是使用继承 继承的优点新的实现较为容易,因为基类的大部分功能可以通过继承的关系自动进入子类。修改和扩展继承而来的实现较为容易。继承的缺点继承将基类的实现细节暴露给子类。称“白箱”复用。如果基类发生改变,那么子类的实现也不得不发生改变。多继承不利于维护,郑州大学软件学院 赵哲,36,CARP特点,优点:依赖少,条条框框的限制少几乎可以用到任何环境中容易实现修改和扩展容易缺点:对象多,需要管理,郑州大学软件学院 赵哲,37,迪米特原则,The Law of DemeterLoD不要和陌生人说话又叫最少知识原则小国寡民,邻国相望,鸡犬之声相闻,民至老死,不相往来,郑州大学软件学院 赵哲,迪米特法则,一个对象应当对其他对象有尽可能少的了解,不和陌生人说话最少知识、不相往来talk only to your immediate friends 建议多使用Private和Protect,而少使用public,郑州大学软件学院 赵哲,39,迪米特法则,郑州大学软件学院 赵哲,40,迪米特法则的优点,相当弱的类耦合利于复用只要有可能类就可以设计成不变类,郑州大学软件学院 赵哲,41,迪米特法则的缺点,造成大量的小方法,散落在系统的各个角落这些方法仅仅是传递间接的调用容易引起第一次了解系统人的困惑,郑州大学软件学院 赵哲,42,迪米特法则和设计模式,Faade模式和Mediator模式都符合迪米特法则,郑州大学软件学院 赵哲,43,总结,泛化、依赖、关联、聚合、组合泛化(继承)class Aclass B:A依赖:class A public void Function(B b)class B,郑州大学软件学院 赵哲,关联class A B b=new B();class B 聚合class A B b=new B();C c=new C();class B class C,郑州大学软件学院 赵哲,合成class APublic A()B b=new B();class B 聚合和合成的区别:生命周期。合成中,随着A类的生命周期结束,B也结束。必须在A类里面使用。聚合中,随着A类的生命周期结束,B,C不一定结束。,郑州大学软件学院 赵哲,总结7大原则,单一职责原则开-闭原则里氏代换原则依赖倒转原则接口隔离原则合成/聚合复用原则迪米特原则,郑州大学软件学院 赵哲,