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

一 引子

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

Sublime Text 3 初试牛刀

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

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

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

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/下。…

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…

wxPython python3.x下载地址

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

爬山算法和模拟退火算法简介(转)

源&#xff1a;爬山算法和模拟退火算法简介 一. 爬山算法 ( Hill Climbing ) 介绍模拟退火前&#xff0c;先介绍爬山算法。爬山算法是一种简单的贪心搜索算法&#xff0c;该算法每次从当前解的临近解空间中选择一个最优解作为当前解&#xff0c;直到达到一个局部最优解。 爬山算…

How to connect oracle databse

1. 下載客戶端Oracle Developer Tools for Visual Studio_32bit 安裝后通過配置tnsnames.ora指定連接 C:\app\user name\product\11.2.0\client_1\Network\Admin\SERVER(DESCRIPTION(ADDRESS(PROTOCOLTCP)(HOSTIP Address)(PORT1521))(CONNECT_DATA(SIDSID))) 重啓后可以在VS20…

wordpress实现搜索页关键词高亮

http://www.hehaibao.com/wordpress-search-word-highlight/ 今天给网站搜索页加了关键词高亮功能&#xff0c;分享出来&#xff0c;希望对小伙伴们有所帮助。 实现效果如下图&#xff1a; 那么我们直接先上主要代码&#xff1a; 1 2 3 4 5 6 7 8 9 10 11<?php$s trim(ge…

数组和指针的区别

1、访问方式不同 ⑴数组的下标引用&#xff08;读一次内存&#xff09; char ary[5] "hello",c; ...... c ary[2]; ⑵对指针的引用&#xff08;读两次内存&#xff09; char *P,c; ...... c *p;   ⑶对指针进行下标引用&#xff08;读两次内存&#xff09; char…

SQL图像查看器 —— SQL Image Viewer

有时候往数据库里面存储了一些图片&#xff0c;但是如果不写读取程序的话&#xff0c;就不知道存储的对不对。 或者查看SQL数据库里面二进制看不懂&#xff0c;这个看图片很直观的。 就需要SQL Image Viewer这么一个?B的软件了 自己去找下载地址吧&#xff0c;我没有破解版。 …

wordpress插件使用教程 – 使用Google Language Translator 实现网站多语言在线翻译

https://www.wordpresshy.com/17600 在使用WordPress进行企业建站的时候会碰到一些网站需要中英双语言&#xff0c;这时候就需要WPML插件的使用。需要使用WPML进行网站翻译的站长可以点击查看以下文章 : WordPress中翻译网站&#xff08;WPML插件&#xff09;的使用 但是WPML的…

Oracle服务器连接

1 Oracle服务器的两类连接方式Oracle数据库连接有多种方式&#xff0c;按照客户端和服务器端是否同机运行可分成两大类&#xff0c;一是本地连接&#xff0c;二是通过网络连接。本地连接。顾名思义&#xff0c;就是客户端程序和服务器程序运行在同一台机器上。安装Oracle时&…

Object C学习笔记13-Dictionary字典

通过Array数组和Set集合的学习和理解&#xff0c;可以想象得到Dictionary也分为两种情况了&#xff0c;那就是可变和不可变两种类型的。的确如此&#xff0c;在Object C中提供了两个字典类&#xff0c;分别为NSDictionary 和 NSMutableDictionary. 在.NET中我们也学习过Diction…

零基础逆向工程24_C++_01_类_this指针_继承本质_多层继承

1 类内的成员函数和普通函数的对比 1.1 主要是从参数传递、压栈顺序、堆栈平衡来总结. 1.参数传递&#xff1a;成员函数多传一个this指针 2.压栈顺序&#xff1a;成员函数会将this指针压栈&#xff0c;在函数调用取出 3.堆栈平衡&#xff1a;普通函数是外平栈 对比图如下&#…

Wordpress基础:精简头部wp_head

https://www.cnblogs.com/tinyphp/p/5859167.html Wordpress基础&#xff1a;精简头部wp_head 在Wordpress里 <?php wp_head(); ?> wp_head()是一个重要的函数&#xff0c;它允许插件开发者向你的站点动态地添加CSS和javascript&#xff0c;如果我们不在模板中引入这…

Nodejs架构之json空处理

2019独角兽企业重金招聘Python工程师标准>>> 判断一个json是否存在某个对象。 正确的做法是 data {"mch_id":"12345678","hb_type":"NORMAL","hblist":"dddd"}; if(data ! null && data.hbli…