版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、1第2章 SimDuck案例 - 类与接口设计辛国栋软件学院,宋健529,2目的r掌握类的设计r类与类之间的关系rUML中类图绘制方法r抽象方法、抽象类r覆盖的概念rSimuDuck游戏代码实现rJava中的注释rJava中抽象类与接口的概念r接口的实现r分离变化点原则r匿名对象r组合与继承的对比3从SimDuck游戏开始r问题描述rJ o e 上班的公司要做一上班的公司要做一套相当成功的模拟鸭子游套相当成功的模拟鸭子游戏戏S I m U D u c k。游。游戏中出现各种鸭子,戏中出现各种鸭子, 一边一边游泳戏水,一边呱呱叫。游泳戏水,一边呱呱叫。r此系统的内部设计使用了此系统的内部设计使用
2、了标准的标准的OO 技术,设计了技术,设计了一个鸭子超类,并让各种一个鸭子超类,并让各种鸭子继承自这个超类。鸭子继承自这个超类。4First Design:r游戏中出现各种鸭子,一边游泳戏水,一边呱呱叫m这些鸭子游泳戏水、呱呱叫,因此可以抽象出这些鸭子的超类,定义为Duck类,在Duck类中有swim,quack方法。m每种鸭子都有自己的外观,因此每种鸭子都应该实现自己的外观。5First Design:r补充内容补充内容UML:m软件工程师的图纸表达 UML(Unified Modeling Language )统一建模语言m类图的画法: 从上到下分为类名、属性和操作。类名必须有的 类如果有
3、属性,则每一属性都必须有一个名字,还可以有其它的描述信息,如可见性、数据类型、缺省值等; 类如果有操作,则每一个操作也都有一个名字,其它可选的信息包括可见性、参数的名字、参数类型、参数缺省值和操作的返回值的类型等6First Design:r补充内容类图的绘制方法:补充内容类图的绘制方法:Account- balance : double = 1+ Deposit(amount : double) : int+ ComputeInterest() : double可见性-或或 代表private+或或 代表public#或或 代表protected返回值类型操作名称斜体为抽象操作缺省值类名斜体
4、为抽象类属性名称参数列表7First Design:r补充内容类图的绘制方法:补充内容类图的绘制方法:m类之间的继承关系绘制方法 泛化(继承)关系用来描述类的一般和具体之间的关系is a kind of。 UML中,用一条带有空心大箭头的有向实线表示,箭头指向父类 SuperClassSuperClassSubClass8First Design:r每种鸭子都有display方法实现,有可能发生遗忘,仅仅靠程序员的素质来保证该方法的存在不合适。r因此最好能在Duck超类中声明一个display()方法;从而子类必须必须完成display()具体实现,否则就会出现语法错误。9First Desi
5、gn:所有的鸭子都会呱呱叫(Quack)也会游泳(Swim),所以由超类负责处理这部分的实现代码。每个子类型负责实现自己的display()行为,以在屏幕上显示该鸭子的外观因为每一种鸭子的外观都不同,所以display()方法是抽象的。许多其他类型的鸭子继承自Duck类 10First Design:RubberDuck叫声和MallardDuck、ReadHeadDuck叫声不一样,因此,覆盖超类中的quack()- java中的覆盖概念新建立一种鸭子:RubberDuck11代码实现:12source code of Duckpublic class Duck public Duck()
6、public void quack() System.out.println(Duck classs quack method); public void swim() System.out.println(Duck classs swim method); public abstract void display();abstract public class Duck 13source code of RedheadDuckpublic class RedheadDuck extends Duck /构造方法 public RedheadDuck() /* 多行注释 */ public v
7、oid display() /*todo Implement this duck and its inheritance class.Duck abstract method*/ System.out.println(This is RedheadDucks display!); 14source code of MallardDuckpublic class MallardDuck extends Duck public MallardDuck() public void display() System.out.println(This is MallardDucks display!);
8、 15source code of RubberDuck public class RubberDuck extends Duck public void quack() System.out.println(RubberDuck classs quack method, overided the supers method); public void display() System.out.println(This is MallardDucks display!); 16source code of MiniDuckSimulator public class MiniDuckSimul
9、ator public static void main(String args) MallardDuck mallardDuck = new MallardDuck(); RubberDuck rubberDuck = new RubberDuck(); / print information of MallardDuck System.out.println(The following are MallardDucks behavior); mallardDuck.swim(); mallardDuck.quack(); mallardDuck.display(); /print info
10、rmation of RubberDuck System.out.println(The following are RubberDucks behavior); rubberDuck.swim(); rubberDuck.quack(); rubberDuck.display(); /匿名对象的概念 new RedHeadDuck().display(); 17游戏规则增加:让鸭子飞r去年,公司的竞争压力加剧,公司主管认为该是创新的时候了, 他们需要在下周股东会议上展示一些真正让人印象深刻的东西来振奋人心。r主管认为, 此模拟程序需要会飞的鸭子, 将竞争者抛在后头在这个时候,joe的经理拍胸
11、脯告诉主管们,J o e 只需要一个星期就可以搞定, “毕竟, Joe 是一个O O程序员.这有什么困难?”18Joes design我只需要在D u ck 类中加上fly()方法,然后所有鸭子都会继承fly()。这是我大显身手,展示OO才华的时候了。所有的子类都会继承fly()。Joe加上的19设计出现严重错误“Joe,我正在股东会议上刚刚看了一下展示, 有一只”橡皮鸭子”在屏幕上飞来飞去,这是你开的玩笑吗?”source20Joe设计的问题rJoe忽略了一件事:并非Duck 所有的子类都会飞r当J o e 在D u c k 超类中加上新的行为, 这会使得某些子类也具有这个不恰当的行为。现在
12、可好了!SimUDuck程序中有一个会飞的非动物。r对代码所做的局部修改,影响层面可能不只局部(会飞的橡皮鸭)!21Joe realize this:rJoe体会到了一件事: 当涉及”维护”时,为了”复用”(reuse)目的而使用继承,结局并不完美。在超类中加上fly(),就会导致所有的子类都具备fly(),连那些不该具备fly()的子类也无法免除。22Joe thinks about inheritance我可以把橡皮鸭类中的fly() 方法覆盖掉, 就好像覆盖quack()的作法一样.把橡皮鸭类中的fly() 方法覆盖掉, 就好像覆盖quack()的作法一样.橡皮鸭子不会呱呱叫,所以把qu
13、ack() 的定义覆盖成吱吱叫(squeak)23Add a DecoyDuck in the classes可是, 如果以后我加入木质诱饵鸭Decoy Decoy Duck Duck ,又会如何?诱饵鸭是假鸭,不会飞也不会叫.这是继承层次中的另一个类。注意,诱饵鸭既不会飞也不会叫,可是橡皮鸭不会飞但会叫24How about an interface?rJ o e 认识到继承可能不是一个好的解决方法, 因为他刚刚拿到来自主管的备忘录,希望以后每六个月更新产品(至于更新的方法, 他们还没想到) 。rJ o e 知道规格会常常改变, 每当有新的鸭子子类出现,他就要被迫检查是否需要覆盖fly()
14、和quark(). 这简直是无穷尽的恶梦。r所以,他需要一个更清晰的方法,让某些(而不是全部)鸭子类型可飞或可叫每当有新的鸭子子类出现,他就要被迫检视并可能需要覆盖fly() 和quark()25Joes new idea for the designr我可以把fly() 取出来, 放进一个Flyable接口中。这么一来,只有会飞的鸭子才实现此接口。r同样的方式, 也可以用来设计一个Quackable接口,因为不是所有的鸭子都会叫26What do you think about this design27Duck类代码接口版本abstract public class Duck public
15、 Duck() public void swim() System.out.println(Duck classs swim method); public abstract void display();28DecoyDuck类代码public class DecoyDuck extends Duck public DecoyDuck() public void display() System.out.println(Im a duck Decoy);29What do you think about this design30FlyBehavior & quack类代码public in
16、terface Flyable public void fly();public interface Quackable public void quack();31What do you think about this design32MallardDuck类代码public class MallardDuck extends Duck implements Flyable,Quackable public MallardDuck() public void display() System.out.println(This is MallardDucks display!); publi
17、c void fly() System.out.println(MallardDucks fly); public void quack() System.out.println(MallardDucks quack;); 33RubberDuck类代码public class RubberDuck extends Duck implements Quackable public RubberDuck() public void quack() System.out.println(I am squeek); public void display() System.out.println(I
18、m a rubber duckie);34主类测试类:public class MiniDuckSimulator public static void main(String args) MallardDuck mallardDuck = new MallardDuck(); RedheadDuck redheadDuck = new RedheadDuck(); RubberDuck rubberDuck = new RubberDuck(); DecoyDuck decoyDuck = new DecoyDuck(); mallardDuck.fly(); mallardDuck.qua
19、ck(); redheadDuck.fly(); redheadDuck.quack(); System.out.println(RubberDuck has not implements the flyBehavior so it has only quack() method); rubberDuck.quack(); System.out.println(DecoyDuck has not implements the flyBehavior and QuackBehavior); decoyDuck.display(); 35This is not a good idea “这真是一个
20、超笨的主意,你没发现这么一来重复的代码会变多吗?如果你认为覆盖几个方法就算是差劲,那么对于在N个Duck的子类都要稍微修改一下飞行的行为,你又怎么说?!”36What would you do if you were Joe?r我们知道,并非”所有”的子类都具有飞行和呱呱叫的行为,所以继承并不是适当的解决方式。r虽然Flyable与Quackable 可以解决一部分问题(不会再有会飞的橡皮鸭),但是却造成代码无法复用,这只能算是从一个恶梦跳进另一个恶梦。r甚至,在会飞的鸭子中,飞行的动作可能还有多种变化.r如果能有一种建立软件的方法,好让我们需要改变软件时,可以在对既有的代码影响最小的情况下,
21、轻易地达成,花较少时间重新整理程序,而多让程序去做更酷的事。该有多好.37Zeroing in on the problemr现在我们知道使用继承有一些缺失, 因为改变鸭子的行为会影响所有种类的鸭子,而这并不恰当。rFlyable与Quackable接口一开始似乎还挺不错, 解决了问题( 只有会飞的鸭子才继承Flyable),但是Java 的接口不具有实现代码,所以继承接口无法达到代码的复用。m这意味着:无论何时你需要修改某个行为, 你被迫得往下追踪并修改每一个有定义此行为的类,一不小心,可能造成新的错误。38Design Principler找出应用中可能需要变化之处,把它们独立出来,不要和
22、那些不需要变化的代码混在一起。r换句话说,如果每次新的需求一来,都会变化到某方面的代码, 那么你就可以确定,这部分的代码需要被抽出来,和其他闻风不动的代码有所区隔。r这个原则的另一个思考方式:把会变化的部分取出并封装起来,以便以后可以轻易地扩充此部分,而不影响不需要变化的其他部分39Separating what changes from what stays the samer如何开始?m就我们目前所知,除了fly() 和quack ( ) 的问题之外,Duck类还算一切正常,似乎没有特别需要经常变化或修改的地方。所以,除了某些小改变之外,我们不打算对Duck类做太多处理。r现在,为了要分开
23、变化和不会变化的部分m我们准备建立两组类(完全远离Duck类), 一个是fly相关的,一个是quack相关的,每一组类将实现各自的动作。比方说,我们可能有一个类实现呱呱叫,另一个类实现吱吱叫另一个类实现安静40We do this:Duck 类仍是所有鸭子的超类,但是飞行和呱呱叫的行为已经被取出,放在别的类结构中。多种行为的实现被放在这里。现在飞行和呱呱叫都有它们自己的类。41Designing the Duck Behaviorsr如何设计类实现飞行和呱呱叫的行为?m我们希望一切能有弹性,毕竟,正是因为一开始的鸭子行为没有弹性,才让我们走上现在这条路。m我们还想能够指定行为到鸭子的实例,比方
24、说,想要产生绿头鸭实例,并指定特定类型的飞行行为给它。干脆顺便让鸭子的行为可以动态地改变好了m换句话说,我们应该在鸭子类中包含设定行为的方法,就可以在运行时动态地改变绿头鸭飞行行为42Designing the Duck Behaviorsr从现在开始,鸭子的行为将被放在分开的类中,此类专门提供某行为的实现。r如此,鸭子类就不再需要知道行为的实现细节43The reasonr我们利用接口代表每个行为,比方说, Flyable 与Quackable,而行为的每个实现都必须实现这些接口之一。m所以这次鸭子类不会负责实现Flyable与Quackable接口,反而是由其他类专门实现Flyable 与
25、Quackable, 这就称为行为类。m由行为类实现行为接口,而不是由Duck类实现行为接口。44The reasonr这样的作法迥异于以往m以前的作法是: 行为是继承自Duck 超类的具体实现而来,或是继承某个接口并由子类自行实现而来。 这两种作法都是依赖于实现,我们被实现绑得死死的,没办法更改行为(除非写更多代码)45The reasonr在我们的新设计中, 鸭子的子类将使用接口 Flyable 与Quackable所表示的行为, 所以实际的实现不会被绑死在鸭子的子类中。r换句话说,特定的实现代码,是位于实现Flyable 与Quackable 的特定类中46Implementing th
26、e Duck BehaviorsFlyBehaviorFlyBehavior接口,所有飞行类别都应该继承自它,所有新的飞行类别都必须实现fly()方法同样,这是一个QuackBehaviorQuackBehavior接口,只有一个quack()方法,所有的叫声类必须实现它这里实现了所有有翅膀的鸭子飞行动作这里实现了所有不会飞的鸭子的动作真的呱呱叫名为呱呱叫,其实是吱吱叫名为呱呱叫,其实不出声47Integrating the Duck Behaviorr关键在于,鸭子现在会将飞行和呱呱叫的动作委托(delegate)别人处理,而不是使用定义在自己类(或子类)内的方法。48Integrating
27、 the Duck Behaviorr作法是这样的:首先, 在鸭子中加入两个实例变量, 分别为fly Behavior 与quackBehavior ,声明为接口类型( 而不是具体类实现类型) , 每个变量会利用多态的方式在运行时引用正确的行为类型( 例如: FlyWithWings、Squeak . . . 等) 。49Integrating the Duck Behavior行为变量被声明为行为接口类型。用performFly( )performFly( ) 和performQuack( )performQuack( ) 取代DuckDuck类中的fly( )与quack()quack()
28、实例变量在运行时会引用特定的行为。50Integrating the Duck Behavior现在, 我们来实现performQuack ():public class Duck QuackBehavior quackBehavior; / 还有更多 public void performQuack() quackBehavior.quack(); m想进行呱呱叫的动作,Duck 对象只要叫quackBe havior 对象去呱呱叫就可以了。在这部分的代码中,我们不在乎QuackBehavior 接口的对象到底是什么,我们只关心该对象知道如何进行呱呱叫就够了每只鸭子都会引用实现QuackBe
29、havior接口的对象不亲自处理呱呱叫行为,而是委托quackBehavior对象帮Duck呱呱叫。51More integration现在来关心如何设定flyBehavior 与quackBehavior 的实例变量MallardDuck类:public class MallardDuck extends Duck public MallardDuck() quackBehavior = new Quack(); flyBehavior = new FlyWithWings(); public void display() System.out.println(“Im a real Mall
30、ard duck”); 绿头鸭使用QuackQuack类处理呱呱叫,所以当performQuack()performQuack()被调用,就把责任交给QuackQuack对象进行真正的呱呱叫。使用FlyWithWingsFlyWithWings作为其FlyBehaviorFlyBehavior类型。别忘了,因为MallardDuckMallardDuck继承自DuckDuck类,所以具有flyBehaviorflyBehavior与quackBehaviorquackBehavior实例变量52More integrationr所以,绿头鸭会真的呱呱叫,而不是吱吱叫,或叫不出声。这是怎么办到的
31、?当MallardDuck对象产生时,它的构造器会把继承来的quackBehavior 实例变量予以初始化成Quack 类型的新实例(Quack是QuackBehavior 的具体实现类)。r同样的处理方式也可以用在飞行行为上: MallardDuck的构造器将flyBehavior实例变量初始化成FlyWithWings类型的实例(FlyWithWings是FlyBehavior的具体实现类) 。53Testing the Duck code输入下面的Duck类( Duck. Java) 以及前面的MallardDuck类(MallardDuck.java)并编译之public class
32、Duck FlyBehavior flyBehavior; QuackBehavior quackBehavior; public Duck() public abstract void display(); public void performFly() flyBehavior.fly(); public void performQuack() quackBehavior.quack(); public void swim() System.out.println(“All ducks fl oat, even decoys!”); abstract为行为接口类型声明两个引用变量,所有鸭子
33、子类(在同一个package)都继承它们。委托给行为类54Testing the Duck code输入FlyBehavior接口(FlyBehavior.java)与两个行为实现类(FlyWithWings.java与FlyNoWay.java),并编译之。public interface FlyBehavior public void fly();public class FlyWithWings implements FlyBehavior public void fly() System.out.println(“Im fl ying!”); public class FlyNoWay
34、 implements FlyBehavior public void fl y() System.out.println(“I cant fl y”); 这是飞行行为的实现,给真会飞的鸭子用 .所有飞行行为类必须实现的接口这是飞行行为的实现,给不会飞的鸭子用包括橡皮鸭和诱饵鸭)55Testing the Duck code输入QuackBehavior接口(QuackBehavior.java)及其三个实现类(Quack.java、MuteQuack . java、Squeak.java),并编译之public interface QuackBehavior public void quac
35、k();public class Quack implements QuackBehavior public void quack() System.out.println(“Quack”); public class MuteQuack implements QuackBehavior public void quack() System.out.println(“”); public class Squeak implements QuackBehavior public void quack() System.out.println(“Squeak”); 56Testing the Du
36、ck code输入并编译测试类MiniDuckSimulator. javapublic class MiniDuckSimulator public static void main(String args) Duck mallard = new MallardDuck(); mallard.performQuack(); mallard.performFly(); 这会调用 MallardDuck 继承来的performQuack(),进而委托给该对象的 QuackBehavior对象处理。也就是说,调用继承来的 quackBehavior 的 quack()至于 performFly()
37、,也是一样的道理。57Testing the Duck code运行代码!58Setting behavior dynamicallyr动态地设定行为m在鸭子里建立了一堆动态的功能没有用到,就太可惜了!假设我们想在鸭子子类透过设定方法(setter method)设定鸭子的行为,而不是在鸭子的构造器内实例化。59Setting behavior dynamically在Duck类中,加入两个新方法:public void setFlyBehavior(FlyBehavior fb) flyBehavior = fb;public void setQuackBehavior(QuackBehavior qb) quackBehavior = qb
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- DB33T 2175.3-2018 人民法院诉讼服务规范 第3部分:热线服务
- 2025委托加工合同书
- 建设年产8000台智能化高效节水喷灌设备项目建议书立项备案审批
- 2024年度天津市公共营养师之三级营养师真题练习试卷A卷附答案
- 2024年度天津市公共营养师之二级营养师每日一练试卷B卷含答案
- 2024年度四川省公共营养师之四级营养师综合练习试卷B卷附答案
- 2024年度四川省公共营养师之三级营养师能力检测试卷B卷附答案
- 2024年度四川省公共营养师之二级营养师高分通关题型题库附解析答案
- 2019-2025年中国真丝化纤纺织品制造行业市场调查研究及投资前景预测报告
- 2024-2025年中国频谱理疗仪电商市场运行态势及行业发展前景预测报告
- 水利水电工程安全管理制度例文(三篇)
- 人教版2024-2025学年第一学期八年级物理期末综合复习练习卷(含答案)
- 《上帝掷骰子吗:量子物理史话》导读学习通超星期末考试答案章节答案2024年
- 病例报告表(CRF)模板
- 塔塔里尼调压器FLBM5介绍.ppt
- 国家开放大学毕业生登记表
- CCC例行检验和确认检验程序
- 初中物理竞赛教程(基础篇)第16讲比热容
- 亲子鉴定书(共3页)
- 容器支腿计算公式(支腿计算主要用于立式容器的支腿受力及地脚螺栓计算)
- 建设工程项目施工安全管理流程图3页
评论
0/150
提交评论