一 引子

都说面向对象的4大支柱是抽象,封装,继承与多态。但是一些初涉编程的开发人员,体会不到继承与多态的妙用,本文就试以一个经典实例来诠释继承与多态的用武之地。本实例的需求来自《重构》一书。

二 需求

1. 任务说明

我们的需求是一个影片出租的小应用,该应用会记录每个顾客的消费金额并打印出来。
程序输入为:顾客租的影片及对应的租期;
程序的处理为:根据顾客租用影片时间及影片类型,计算费用;输出:打印消费单。
影片有三种类型:普通影片、儿童影片及新上映影片。
另外,模仿时下潮流,程序还提供了积分制度,为常客计算积分 ,积分会根据影片是否为新上映影片而不同。

租赁费用计算:

影片类型为儿童片,两天以内费用为2,超出两天的时间,每天的费用为1.5

影片类型为新片,每天的费用为3

影片类型为普通片,三天以内费用为1.5,超出三天,每天的费用为1.5

积分计算:

每次租赁影片,积分加一,如果影片为新片且租赁时间大于1天,则多加一分

2. 本实例为控制台程序,运行界面如下:

三 非继承多态实现方式

根据需求,我们定义三个类,分别是movie类,Rental类(代表一条租用记录)和Customer类(租碟顾客)

其中movie类的代码如下:

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Linq;
 4 using System.Text;
 5
 6 namespace _1.cntbed
 7 {
 8     public enum TYPE
 9     {
10         REGULAR,
11         NEW_RELEASE,
12         CHILDRENS
13     }
14
15     class Movie
16     {
17         private string _title;  //movie name
18         TYPE _typeCode; //price code
19
20         public Movie()
21         {
22             _title = "unname";
23             _typeCode = 0;
24         }
25
26         public Movie(string title, TYPE typeCode)
27         {
28             _title = title;
29             _typeCode = typeCode;
30         }
31
32         public TYPE getTypeCode()
33         {
34             return (TYPE)_typeCode;
35         }
36
37         void setTypeCode(TYPE arg)
38         {
39             _typeCode = arg;
40         }
41
42         public string getTitle()
43         {
44             return _title;
45         }
46     }
47 }

View Code

Rental类的代码如下,租用记录中包含了一个movie对象,以及一个租期成员(积分和租金与此有关):

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{class Rental{private Movie _movie;int _daysRented;public Rental(Movie movie, int daysRented){_movie = movie;_daysRented = daysRented;}public int getDaysRented(){return _daysRented;}public Movie getMovie(){return _movie;}}
}

View Code

Customer类的代码如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{class Customer{private string _name;List<Rental> _rentals = new List<Rental>();public Customer(string name){_name = name;}public void addRental(Rental arg){_rentals.Add(arg);}string getName(){return _name;}public string statement(){double totalAmount = 0;  //总共的租金int frequentRenterPoints = 0;//积分string result = "\r-------------------------------------------\n\r " +"租碟记录--- " + getName() + "\n";foreach (Rental iter in _rentals){double thisAmount = 0;Rental each = iter;switch (each.getMovie().getTypeCode()){case TYPE.REGULAR:thisAmount += 2;//2天之内2元if (each.getDaysRented() > 2)thisAmount += (each.getDaysRented() - 2) * 1.5;break;case TYPE.NEW_RELEASE:thisAmount += each.getDaysRented() * 3;break;case TYPE.CHILDRENS:thisAmount += 1.5;//3天之内1.5元if (each.getDaysRented() > 3)thisAmount += (each.getDaysRented() - 3) * 1.5;break;}frequentRenterPoints++;//对于每一种类型的影片,一次租用积分加1if ((each.getMovie().getTypeCode() == TYPE.NEW_RELEASE) &&each.getDaysRented() > 1)frequentRenterPoints++;result += "\n\t" + each.getMovie().getTitle() + "\t" + thisAmount;totalAmount += thisAmount;}result += "\n共消费 " + totalAmount + " 元" + "\n您增加了 " + frequentRenterPoints + " 个积分\n";return result;}}
}

View Code

主程序代码如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{class Program{static void Main(string[] args){Console.Write("影碟店客户租碟明细");Movie m1 = new Movie("致我们终将逝去的青春", TYPE.NEW_RELEASE);Movie m2 = new Movie("我是特种兵之利刃出鞘", TYPE.REGULAR);Movie m3 = new Movie("熊出没之环球大冒险", TYPE.CHILDRENS);Rental r1 = new Rental(m1, 4);Rental r2 = new Rental(m1, 2);Rental r3 = new Rental(m3, 7);Rental r4 = new Rental(m2, 5);Rental r5 = new Rental(m3, 3);Customer c1 = new Customer("孙红雷");c1.addRental(r1);c1.addRental(r4);Customer c2 = new Customer("林志玲");c2.addRental(r1);c2.addRental(r3);c2.addRental(r2);Customer c3 = new Customer("刘德华");c3.addRental(r3);c3.addRental(r5);Customer c4 = new Customer("孙俪");c4.addRental(r2);c4.addRental(r3);c4.addRental(r5);Console.Write(c1.statement());Console.Write(c2.statement());Console.Write(c3.statement());Console.Write(c4.statement());}}
}

View Code

四 继承多态实现方式

我们定义一个Movie父类,每种影片类型均定义一个Movie子类(ChildrensMovie,NewReleaseMovie,RegularMovie),同时定义一个Movie工厂类(MovieFactoryMethod)。代码如下:

新的Movie类的代码如下:Movie类提供积分计算和租金计算的默认实现。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{public enum TYPE{REGULAR,NEW_RELEASE,CHILDRENS}public class Movie{protected string _title;  //movie nameTYPE _priceCode; //price codepublic Movie(){_title = "unname";_priceCode = 0;}public Movie(string title, TYPE priceCode){_title = title;_priceCode = priceCode;}public virtual double getCharge(int daysRented){return 0;//收费
        }public virtual int getFrequentRenterPoints(int daysRented)//积分
        {return 1;}public TYPE getPriceCode(){return (TYPE)_priceCode;}void setPriceCode(TYPE arg){_priceCode = arg;}public string getTitle(){return _title;}}
}

View Code

ChildrensMovie子类的代码如下:重写租金计算方法(多态性),积分计算方法从父类继承。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{public class ChildrensMovie : Movie{public ChildrensMovie (string title){_title = title;}public override double getCharge(int daysRented){double result = 1.5;if (daysRented > 3)result += (daysRented - 3) * 1.5;return result;}}
}

View Code

NewReleaseMovie子类的代码如下:重写租金计算方法(多态性)和积分计算方法(多态性)。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{public class NewReleaseMovie:Movie{public NewReleaseMovie (string title){_title = title;}public override double getCharge(int daysRented){return daysRented * 3;}public override int getFrequentRenterPoints(int daysRented){return (daysRented > 1) ? 2 : 1;}}
}

View Code

RegularMovie子类的代码如下:重写租金计算方法(多态性),积分计算方法从父类继承。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{public class RegularMovie : Movie{public RegularMovie(string title){_title = title;}public override double getCharge(int daysRented){double result = 2;if (daysRented > 2)result += (daysRented - 2) * 1.5;return result;}}
}

View Code

Rental类的代码保持不变:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{class Rental{private Movie _movie;int _daysRented;public Rental(Movie movie, int daysRented){_movie = movie;_daysRented = daysRented;}public int getDaysRented(){return _daysRented;}public Movie getMovie(){return _movie;}}
}

View Code

MovieFactoryMethod类的代码如下:根据不同的影片类型,返回相应的派生类对象,注意这里的函数的返回值是父类。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{public class MovieFactoryMethod{public Movie MakeMovie(string strTitle, TYPE arg){switch (arg){case TYPE.REGULAR:return new RegularMovie(strTitle);case TYPE.CHILDRENS:return new ChildrensMovie(strTitle);case TYPE.NEW_RELEASE:return new NewReleaseMovie(strTitle);default://cout << "Incorrect Price Code" << endl;return null;}}}
}

View Code

新的Customer代码如下:变得简单了,不用关心List里的Movie类对象的真正类型,会自动调用相应派生类的方法

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{class Customer{private string _name;List<Rental> _rentals = new List<Rental>();public Customer(string name){_name = name;}public void addRental(Rental arg){_rentals.Add(arg);}string getName(){return _name;}public string statement(){double totalAmount = 0;  //总共的租金int frequentRenterPoints = 0;//积分string result = "\r-------------------------------------------\n\r " +"租碟记录--- " + getName() + "\n";foreach (Rental iter in _rentals){Rental each = iter;frequentRenterPoints += each.getMovie().getFrequentRenterPoints(each.getDaysRented());result += "\n\t" + each.getMovie().getTitle() + "\t" + each.getMovie().getCharge(each.getDaysRented());totalAmount += each.getMovie().getCharge(each.getDaysRented()); }result += "\n共消费 " + totalAmount + " 元" + "\n您增加了 " + frequentRenterPoints + " 个积分\n";return result;}}
}

View Code

最后,主程序的代码如下:注意Rental类接受一个Movie类的参数,但是我们可以传递给他一个派生类对象。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace _1.cntbed
{class Program{static void Main(string[] args){Console.Write("影碟店客户租碟明细");MovieFactoryMethod mfm = new MovieFactoryMethod();Movie m1 = mfm.MakeMovie("致我们终将逝去的青春", TYPE.NEW_RELEASE);Movie m2 = mfm.MakeMovie("我是特种兵之利刃出鞘", TYPE.REGULAR);Movie m3 = mfm.MakeMovie("熊出没之环球大冒险", TYPE.CHILDRENS);Rental r1 = new Rental(m1, 4);Rental r2 = new Rental(m1, 2);Rental r3 = new Rental(m3, 7);Rental r4 = new Rental(m2, 5);Rental r5 = new Rental(m3, 3);Customer c1 = new Customer("孙红雷");c1.addRental(r1);c1.addRental(r4);Customer c2 = new Customer("林志玲");c2.addRental(r1);c2.addRental(r3);c2.addRental(r2);Customer c3 = new Customer("刘德华");c3.addRental(r3);c3.addRental(r5);Customer c4 = new Customer("孙俪");c4.addRental(r2);c4.addRental(r3);c4.addRental(r5);Console.Write(c1.statement());Console.Write(c2.statement());Console.Write(c3.statement());Console.Write(c4.statement());}}
}

View Code

五  总结

函数的返回值是父类,我们却可以返回一个派生类对象;函数的参数是父类,我们却可以传入一个派生类对象。foreach循环遍历List<>,程序会自动根据List里保存的对象的真正类型,引用相应的方法。

我们这个需求,无论租何种影片,租多长时间,都送一积分,故在Movie基类提供了积分计算方法getFrequentRenterPoints()的默认实现;NewReleaseMovie类型的影片积分计算方法有所不同,故重写了getFrequentRenterPoints()方法;关于租金计算方法getCharge,大家可以试着自行分析。

考虑增加支持一种新影片类型-TVB电视剧:积分和租金的计算规则如下:

租金计算方式:借7天之内收5元,超过7天,之后每天收2元
积分:只要租了就积1分,然后每达到3天的倍数积1分(比如1-2天积1分,3-5天积2分)

要求在以上2个小框架中,分别实现该需求,然后,回过头来看看,那种方式更加方便,就可以更好的体会到继承和多态的强大之处了。

这里先揭示一下,在非继承多态框架下,实现新增一种影片类型或现有影片类型的积分或租金计算规则改变了,Customer都需要进行改动;而在继承多态框架下,实现新增一种影片类型或现有影片类型的积分或租金计算规则改变了,Customer无需任何改动。在继承多态框架下,若要新增一种影片类型,则只需新增一个Movie派生类;现有影片类型的积分或租金计算规则改变了,则只需重写相应派生类型的积分或租金计算函数。

最后说一句,大型项目中,Customer的维护者和Movie家族类的维护者有可能不是同一个人,这样,需求的变更对于Customer的维护者来说是透明的,Customer的维护者可能都不知道何时增加了几种影片类型。

六 源码下载

demo代码

作者:宋波
出处:http://www.cnblogs.com/ice-river/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接。
正在看本人博客的这位童鞋,我看你气度不凡,谈吐间隐隐有王者之气,日后必有一番作为!旁边有“推荐”二字,你就顺手把它点了吧,相得准,我分文不收;相不准,你也好回来找我!

转载于:https://www.cnblogs.com/ice-river/p/3573735.html

一个经典实例理解继承与多态原理与优点(附源码)---面向对象继承和多态性理解得不够深刻的同学请进...相关推荐

  1. SHA3系列(KECCAK)哈希算法原理及实现(附源码)

    相关文章: (本文持续更新中) SHA3系列(KECCAK)哈希算法原理及实现(附源码) SHA512系列哈希算法原理及实现(附源码) SHA224和SHA256哈希算法原理及实现(附源码) 国密SM ...

  2. SHA224和SHA256哈希算法原理及实现(附源码)

    相关文章: SHA224和SHA256哈希算法原理及实现(附源码) 国密SM3哈希算法原理及实现(附源码) SHA1哈希算法原理及实现(附源码) MD5哈希算法原理及实现(附源码) MD4哈希算法原理 ...

  3. 国密SM3密码杂凑算法原理及实现(附源码)

    相关文章: 国密SM3哈希算法原理及实现(附源码) SHA1哈希算法原理及实现(附源码) MD5哈希算法原理及实现(附源码) MD4哈希算法原理及实现(附源码) MD2哈希算法原理及实现(附源码) M ...

  4. SHA512系列哈希算法原理及实现(附源码)

    相关文章: SHA512系列哈希算法原理及实现(附源码) SHA224和SHA256哈希算法原理及实现(附源码) 国密SM3哈希算法原理及实现(附源码) SHA1哈希算法原理及实现(附源码) MD5哈 ...

  5. linux直流电机测试,带霍尔传感器编码器的直流减速电机测速原理讲解(附源码)...

    查看: 14294|回复: 83 带霍尔传感器编码器的直流减速电机测速原理讲解(附源码) 高级会员, 积分 891, 距离下一级还需 109 积分 积分金钱891 注册时间2019-4-22 在线时间 ...

  6. Halcon转OpenCV实例--去除纸张中的颜色笔迹/墨迹(附源码)

    导读 本文主要介绍一个去除纸张中颜色笔迹/墨迹的实例,并将Halcon实现转为OpenCV. 实例来源 实例来源于51Halcon论坛的讨论贴: https://www.51halcon.com/fo ...

  7. 超详讲解图像拼接/全景图原理和应用 | 附源码

    点击上方"小白学视觉",选择加"星标"或"置顶" 重磅干货,第一时间送达 概述 图像拼接是计算机视觉中最成功的应用之一.如今,很难找到不包含 ...

  8. 如何将图片中的一个任意四边形区域的图像转化为矩形【附源码】

    将一幅图像中的任意四边形区域映射为矩形区域.比如,我们有时拍照片,因为角度的问题,本来是矩形的广告牌,在照片中可能表现为一个任意的四边形.事实上,在现实中本来是矩形的物体,在照片中往往呈现为任意四边形 ...

  9. 数字信号 fft c源码_如何制作一个简单的人体动态识别微信小程序(附源码)

    知乎小白第一次写专栏,还请多指教. 先放成果. GitHub源码: lrioxh/HAR-applet-of-Wechat​github.com b站演示视频: 居然不需要服务器?!如何制作一个简单的 ...

最新文章

  1. R语言plotly可视化:plotly可视化箱图、相同数据集对比使用不同分位数算法的可视化差异(quartilemethod参数、linear、inclusive、exclusive)
  2. JDK1.8源码(三)——java.lang.String 类
  3. J2EE的13种核心技术
  4. ubuntu QT 编译报错 -1: error: cannot find -lGL问题的解决方法
  5. 计算机网络是将地理知识,计算机网络的基础知识精选.ppt
  6. linux下设置oracle开机自启动
  7. 【安卓开发】AS神奇的报错:Cannot find AVD system path. Please define ANDROID_SDK_ROOT
  8. ai python 代码提示插件_Python 还能实现哪些 AI 游戏?附上代码一起来一把!
  9. hihoCoder 1513: 小Hi的烦恼(五维偏序+bitset)
  10. Emacs Lisp程序单步调试
  11. Maven生命周期和插件
  12. 数据类型和Json格式[zt]
  13. 创建华为云服务器实验报告,华为云正式发布云端实验室,真正实现云服务实验云上做...
  14. Android 音乐资源管理与播放
  15. krnln.fnr和shell.fne_电脑开机显示failedtoloadkernllibrary什么意思啊
  16. AttributeError: Can‘t get attribute ‘SPPF‘ on <module ‘models.common‘ from ‘D:\\ModelTest\\yolov5-5
  17. java dh_java DH加密算法备忘
  18. Linux下TBB安装及编译
  19. 剑网三一个服务器最多有多少人,人比怪多,剑网三缘起刚开服,升级最大的阻碍居然是玩家...
  20. 前程无忧网站,职位信息一步到位函数爬取!!!真一步到位

热门文章

  1. 小学计算机技术指导纲要,《中小学信息技术课程指导纲要(试行)》
  2. Opencv——DFT变换(实现两个Mat的卷积以及显示Mat的频域图像)
  3. obj[]与obj._Ruby中带有示例的Array.rassoc(obj)方法
  4. 使用OpenCV python模块读取图像并将其另存为灰度系统
  5. 将数据、代码、栈放入不同的栈(8086)
  6. java怎么获取当前日期_JAVA中获取当前系统时间
  7. uva 701——The Archeologists\' Dilemma
  8. 二叉树的相关题(叶子结点个数,最大深度,找特殊值结点(值不重复),判断两个树是否相同,判断两个数是否为镜像树,是否为子树,)
  9. Linux网络编程服务器模型选择之循环服务器
  10. lseek、stat、access、chmod、strtol、truncate、unlink