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

一 引子

都说面向对象的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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/399690.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

推荐几款好用的云笔记软件

一直钟爱印象笔记&#xff0c;程序员的电脑上必装的软件&#xff0c;但最近期限到了&#xff0c;再也不能像以前无限制的上传文件&#xff0c;续费也比去年的只要九块九一年高出了很多倍&#xff0c;因此&#xff0c;注册试用了其他的笔记&#xff0c;发现云笔记众多&#xff0…

DataTables怎么给某一列加上click事件

$(#example tbody).on(click,tr td:nth-child(3), function (e) {var name $(this).text();alert(name); } ); https://ask.csdn.net/questions/178026

oschina下载工具

http://www.oschina.net/project/tag/97/download-tools

读《启示录》有感-----1

书本封面很有特色&#xff0c;已经标注这本书内容的真谛&#xff1a;表明了这本书要做到的东西&#xff0c;看这本书能得到的东西 怎样打造用户喜爱的产品? 好产品基本条件&#xff1a;价值、可用性、可行性&#xff0c;三者缺一不可。 做一个合格的产品经理需要哪些能力&…

Sublime Text 3 初试牛刀

每次我在其他视频网站上看学习视频的时候&#xff0c;看着老师用的编辑器高大上档次&#xff0c;而我一般用Notepad&#xff0c;和Dreamweaver去编辑网页&#xff0c;需要每一行代码&#xff0c;打进去&#xff0c;效率低。最近看到sublime编辑器&#xff0c;在网上搜了一下说是…

js 去除左右空格

//去左右空格; function trim(s){return s.replace(/(^\s*)|(\s*$)/g, ""); }

在没有域环境的情况下配置完整安装的SharePoint2010和2013

完整安装SharePoint2010。完成后先不要运行配置向导。配置数据库。SharePoint会安装一个POWERSHELL在这里14\CONFIG\POWERSHELL\Registration。运行该目录下的psconsole会打开一个命令行窗口。执行New-SPConfigurationDatabase【回车】DatabaseName <config database name&g…

mysql: union / union all / 自定义函数用法详解

mysql&#xff1a; union / union all http://www.cnblogs.com/wangyayun/p/6133540.html mysql&#xff1a;自定义函数用法详解 http://www.cnblogs.com/caoruiy/p/4485273.html 转载于:https://www.cnblogs.com/bj20170624/p/7605426.html

[C++学习历程]基础部分 C++中的函数学习

本文地址&#xff1a;http://blog.csdn.net/sushengmiyan/article/details/20305815 作者&#xff1a;sushengmiyan 一。静态变量&#xff1a; 局部变量是线程到达定义的地方的时候进行初始化&#xff0c;如果定义在函数中&#xff0c;那么每次函数调用的时候&#xff0c;都会进…

js 表单自动提交

var form document.getElementById(formid);form.submit();

linux下安装Mysql(干货!!!)解决mysql 1130问题,远程登录问题

转载自&#xff1a;http://www.cnblogs.com/xxoome/p/5864912.html linux版本&#xff1a;CentOS7 64位 1、下载安装包“mysql-5.6.33-linux-glibc2.5-x86_64.tar.gz” # 安装依赖 yum -y install perl perl-devel autoconf libaio 2、把下载的安装包移动到/usr/local/下。…

linux命令:ftp

1. 登录&#xff1a; ftp IP_ADDR &#xff1b; 根据提示输入USER_NAME PASS_WORD 或&#xff1a; ftp -i -n IP_ADDR user USER_NAME PASS_WORD ftp -i -n 172.17.17.17 user PUB 123456 2. 下载文件 下载文件通常用get和mget这两条命令。 a) get 格式&a…

flex 有关数据类型强制转

flex 编辑页面里变量的强制类型转化时&#xff0c;竟然不能用 as 比如&#xff1a;在mxml里&#xff0c; private var aa:String"89"; private var bb:Number(Number)aa; 正确 private var aa:String"89"; private var bb:intaa as int ; 不正确 估计这两种…

Openstack Neutron : 安全

目录 - iptable&#xff1a;起源 - tables - chains - rules - 方向 - Security group 安全组&#xff1a; - Firewall 防火墙&#xff1a; - 更高的安全 - 无处安放的安全 - 公共安全 当业务从传统环境迁移到云上之后&a…

SQL语句汇总(三)——聚合函数、分组、子查询及组合查询

https://www.cnblogs.com/ghost-xyx/p/3811036.html SQL语句汇总&#xff08;三&#xff09;——聚合函数、分组、子查询及组合查询 拖了一个星期&#xff0c;终于开始写第三篇了。走起&#xff01; 聚合函数&#xff1a; SQL中提供的聚合函数可以用来统计、求和、求最值等等…

iOS应用国际化教程(2014版)

本文转载至 http://www.cocoachina.com/industry/20140526/8554.html 这篇教程将通过一款名为iLikeIt的应用带你了解最基础的国际化概念&#xff0c;并为你的应用添加国际化的支持。该示例应用有一个标签和一个You Like&#xff1f;按钮&#xff0c;用户无论何时点击You Like?…

公众平台商户接入(微信支付)功能申请教程

场景及类型介绍 商家可以申请公众账号支付和APP&#xff08;应用客户端&#xff09;支付两种接入微信支付方式。 公众账号支付&#xff1a;用户在微信公众帐号内使用微信支付消费&#xff0c;案例&#xff1a;易迅、QQ充值。 APP&#xff08;应用客户端&#xff09;支付&#x…

datatables 自定义按钮及响应点击事件

按钮 {"targets": -1,"class": "but_xq","data": null,"bSortable": false,"defaultContent": "<p><a id\"edit\" href "#\">修改</a > <a id\"del\" …

wxPython python3.x下载地址

2019独角兽企业重金招聘Python工程师标准>>> wxPython python3.x下载地址 http://wxpython.org/Phoenix/snapshot-builds/ 转载于:https://my.oschina.net/laugh2last/blog/504688

hadoop2.2.0 core-site.xml--security properties

<!--- security properties --> <property><name>hadoop.security.authorization</name><value>false</value><description>Is service-level authorization enabled?</description> </property> 注释&#xff1a; <…