`
k_lb
  • 浏览: 798426 次
  • 性别: Icon_minigender_1
  • 来自: 郑州
社区版块
存档分类
最新评论
  • kitleer: 据我所知,国内有款ETL调度监控工具TaskCTL,支持ket ...
    kettle调度

详解继承、多态、重载和重写

 
阅读更多

本文只是转载,从别处参考过来,仅供个人参考

什么是多态?它的实现机制是什么呢?重载和重写的区别在那里?这就是这一次我们要回顾的四个十分重要的概念:继承、多态、重载和重写。

继承

简单的说,继承就是在一个现有类型的基础上,通过增加新的方法或者重定义已有方法(下面会讲到,这种方式叫重写)的方式,产生一个新的类型。继承是面向对象的三个基本特征--封装、继承、多态的其中之一,我们在使用JAVA时编写的每一个类都是在继承,因为在JAVA语言中,java.lang.Object类是所有类最根本的基类(或者叫父类、超类),如果我们新定义的一个类没有明确地指定继承自哪个基类,那么JAVA就会默认为它是继承自Object类的。

我们可以把JAVA中的类分为以下三种:

  1. 普通类:使用class定义且不含有抽象方法的类。
  2. 抽象类:使用abstract class定义的类,它可以含有,也可以不含有抽象方法。
  3. 接口类:使用interface定义的类。


在这三种类型之间存在下面的继承规律:

  • 普通类可以继承(extends)普通类,可以继承(extends)抽象类,可以继承(implements)接口。
  • 抽象类可以继承(extends)普通类,可以继承(extends)抽象类,可以继承(implements)接口。
  • 接口只能继承(extends)接口。


请注意上面三条规律中每种继承情况下使用的不同的关键字extends和implements,它们是不可以随意替换的。大家知道,一个普通类继承一个接口后,必须实现这个接口中定义的所有方法,否则就只能被定义为抽象类。我在这里之所以没有对implements关键字使用“实现”这种说法是因为从概念上来说它也是表示一种继承关系,而且对于抽象类implements接口的情况下,它并不是一定要实现这个接口定义的任何方法,因此使用继承的说法更为合理一些。

以上三条规律同时遵守下面这些约束:

  1. 普通类和抽象类都只能最多继承一个普通类,或者最多继承一个抽象类,并且这两种情况是互斥的,也就是说它们要么继承一个普通类,要么继承一个抽象类。
  2. 普通类、抽象类和接口在继承接口时,不受数量的约束,理论上可以继承无限多个接口。当然,对于普通类来说,它必须实现它所继承的所有接口中定义的全部方法。


继承给我们的编程带来的好处就是对原有类的复用(重用)。就像模块的复用一样,类的复用可以提高我们的开发效率,实际上,模块的复用是大量类的复用叠加后的效果。除了继承之外,我们还可以使用组合的方式来复用类。所谓组合就是把原有类定义为新类的一个属性,通过在新类中调用原有类的方法来实现复用。如果新定义的类型与原有类型之间不存在被包含的关系,也就是说,从抽象概念上来讲,新定义类型所代表的事物并不是原有类型所代表事物的一种,比如黄种人是人类的一种,它们之间存在包含与被包含的关系,那么这时组合就是实现复用更好的选择。下面这个例子就是组合方式的一个简单示例:

Java代码 复制代码
  1. publicclassSub{
  2. privateParentp=newParent();
  3. publicvoiddoSomething(){
  4. //复用Parent类的方法
  5. p.method();
  6. //othercode
  7. }
  8. }
  9. classParent{
  10. publicvoidmethod(){
  11. //dosomethinghere
  12. }
  13. }


当然,为了使代码更加有效,我们也可以在需要使用到原有类型(比如Parent p)时,才对它进行初始化。

使用继承和组合复用原有的类,都是一种增量式的开发模式,这种方式带来的好处是不需要修改原有的代码,因此不会给原有代码带来新的BUG,也不用因为对原有代码的修改而重新进行测试,这对我们的开发显然是有益的。因此,如果我们是在维护或者改造一个原有的系统或模块,尤其是对它们的了解不是很透彻的时候,就可以选择增量开发的模式,这不仅可以大大提高我们的开发效率,也可以规避由于对原有代码的修改而带来的风险。

多态

多态是又一个重要的基本概念,上面说到了,它是面向对象的三个基本特征之一。究竟什么是多态呢?我们先看看下面的例子,来帮助理解:

Java代码 复制代码
//汽车接口
  1. interfaceCar{
  2. //汽车名称
  3. StringgetName();
  4. //获得汽车售价
  5. intgetPrice();
  6. }
  7. //宝马
  8. classBMWimplementsCar{
  9. publicStringgetName(){
  10. return"BMW";
  11. }
  12. publicintgetPrice(){
  13. return300000;
  14. }
  15. }
  16. //奇瑞QQ
  17. classCheryQQimplementsCar{
  18. publicStringgetName(){
  19. return"CheryQQ";
  20. }
  21. publicintgetPrice(){
  22. return20000;
  23. }
  24. }
  25. //汽车出售店
  26. publicclassCarShop{
  27. //售车收入
  28. privateintmoney=0;
  29. //卖出一部车
  30. publicvoidsellCar(Carcar){
  31. System.out.println("车型:"+car.getName()+"单价:"+car.getPrice());
  32. //增加卖出车售价的收入
  33. money+=car.getPrice();
  34. }
  35. //售车总收入
  36. publicintgetMoney(){
  37. returnmoney;
  38. }
  39. publicstaticvoidmain(String[]args){
  40. CarShopaShop=newCarShop();
  41. //卖出一辆宝马
  42. aShop.sellCar(newBMW());
  43. //卖出一辆奇瑞QQ
  44. aShop.sellCar(newCheryQQ());
  45. System.out.println("总收入:"+aShop.getMoney());
  46. }
  47. }


运行结果:

  1. 车型:BMW 单价:300000
  2. 车型:CheryQQ 单价:20000
  3. 总收入:320000


继承是多态得以实现的基础。从字面上理解,多态就是一种类型(都是Car类型)表现出多种状态(宝马汽车的名称是BMW,售价是300000;奇瑞汽车的名称是CheryQQ,售价是2000)。将一个方法调用同这个方法所属的主体(也就是对象或类)关联起来叫做绑定,分前期绑定和后期绑定两种。下面解释一下它们的定义:

  1. 前期绑定:在程序运行之前进行绑定,由编译器和连接程序实现,又叫做静态绑定。比如static方法和final方法,注意,这里也包括private方法,因为它是隐式final的。
  2. 后期绑定:在运行时根据对象的类型进行绑定,由方法调用机制实现,因此又叫做动态绑定,或者运行时绑定。除了前期绑定外的所有方法都属于后期绑定。


多态就是在后期绑定这种机制上实现的。多态给我们带来的好处是消除了类之间的耦合关系,使程序更容易扩展。比如在上例中,新增加一种类型汽车的销售,只需要让新定义的类继承Car类并实现它的所有方法,而无需对原有代码做任何修改,CarShop类的sellCar(Car car)方法就可以处理新的车型了。新增代码如下:

Java代码 复制代码
  1. //桑塔纳汽车
  2. classSantanaimplementsCar{
  3. publicStringgetName(){
  4. return"Santana";
  5. }
  6. publicintgetPrice(){
  7. return80000;
  8. }
  9. }



重载和重写

重载和重写都是针对方法的概念,在弄清楚这两个概念之前,我们先来了解一下什么叫方法的型构。型构就是指方法的组成结构,具体包括方法的名称和参数,涵盖参数的数量、类型以及出现的顺序,但是不包括方法的返回值类型,访问权限修饰符,以及abstract、static、final等修饰符。比如下面两个就是具有相同型构的方法:

Java代码 复制代码
  1. publicvoidmethod(inti,Strings){
  2. //dosomething
  3. }
  4. publicStringmethod(inti,Strings){
  5. //dosomething
  6. }


而这两个就是具有不同型构的方法:

Java代码 复制代码
  1. publicvoidmethod(inti,Strings){
  2. //dosomething
  3. }
  4. publicvoidmethod(Strings,inti){
  5. //dosomething
  6. }


了解完型构的概念后我们再来看看重载和重写,请看它们的定义:

  • 重写,英文名是override,是指在继承情况下,子类中定义了与其基类中方法具有相同型构的新方法,就叫做子类把基类的方法重载了。这是实现多态必须的步骤。
  • 重载,英文名是overload,是指在同一个类中定义了一个以上具有相同名称,但是型构不同的方法。在同一个类中,是不允许定义多于一个的具有相同型构的方法的。


我们来考虑一个有趣的问题:构造器可以被重载吗?答案当然是可以的,我们在实际的编程中也经常这么做。实际上构造器也是一个方法,构造器名就是方法名,构造器参数就是方法参数,而它的返回值就是新创建的类的实例。但是构造器却不可以被子类重写,因为子类无法定义与基类具有相同型构的构造器。

分享到:
评论

相关推荐

    Python项目开发实战_银行账户资金交易_编程案例解析实例详解课程教程.pdf

    第一步先创建银行的员工类,第二步由员工类创建员工对象,第三步引用对象中的相关属性,第四步讲述如何销毁由第二步生成的员工对象,第五步由银行的员工类生成继承类,第六步重写继承类中的方法。

    《Java和Android开发实战详解》第6到10章源代码-by 南邮-陈杨

    8.1.3 重写和隐藏父类的方法 145 8.1.4 隐藏父类的成员变量 147 8.1.5 使用父类的构造函数 149 8.2 接口 151 8.2.1 接口简介 151 8.2.2 创建与使用接口 152 8.2.3 在类实现多个接口 154 8.3 接口的...

    java学习笔记-java思维导图流程图表格整理

    Java学习笔记 1 命令行运行 1 Java基础 1 注释 1 单行注释 1 多行注释 1 文档注释 1 标识符和关键字 2 数据类型 4 类型转换 7 ...方法重载 22 ...值传递和引用传递 33 创建和初始化对象 34 ...static详解

    Java 基础核心总结 +经典算法大全.rar

    构造方法 方法重载 方法的重写 初始化 类的初始化 成员初始化 构造器初始化初始化顺序 数组初始化 对象的销毁 对象作用域 this 和 super 访问控制权限继承 多态组合代理 向上转型static final 接口和抽象类接口 ...

    java基础案例与开发详解案例源码全

    5.6.4 方法重载138 5.6.5 自定义方法138 5.6.6 系统提供方法139 5.6.7 方法调用140 5.6.8 方法参数及其传递问题144 5.6.9 理解main方法语法及命令行参数147 5.6.1 0递归算法147 5.7 this关键字148 5.8 JavaBean149 ...

    c++学习经典图书:C++ 编程思想(全).rar

    全书共分十八章,内容涉及对象的演化、数据抽象、隐藏实现、初始化与清除、函数重载与缺省参数、输入输出流介绍、常量、内联函数、命名控制、引用和拷贝构造函数、运算符重载、动态对象创建、继承和组合、多态和虚...

    【04-面向对象(上)】

    •构造器的重载和方法的重载一样,都是方法名相同,形参列表不相同。 •在构造器中可通过this来调用另外一个重载的构造器。 继承的特点 •Java通过关键字extends来实现,实现继承的类称为子类,被继承的...

    疯狂JAVA讲义

    5.7 多态 151 5.7.1 多态性 151 5.7.2 引用变量的强制类型转换 152 5.7.3 instanceof运算符 154 5.8 继承与组合 154 5.8.1 使用继承的注意点 155 5.8.2 利用组合实现复用 156 学生提问:使用组合关系来实现...

    Java基础知识点总结.docx

    方法重载(Overloadjing)与重写(Overriding) 16 数组 17 总结 18 三、 常见关键字 20 四、 面向对象★★★★★ 21 五、 封装(面向对象特征之一)★★★★ 23 六、 继承(面向对象特征之一)★★★★ 25 七、 ...

    java面试题,180多页,绝对良心制作,欢迎点评,涵盖各种知识点,排版优美,阅读舒心

    【基础】重载(Overload)和重写(Override)的区别。重载的方法能否根据返回类型进行区分? 17 【基础】abstract class和interface有什么区别? 18 【基础】静态嵌套类(Static Nested Class)和内部类(Inner Class)...

Global site tag (gtag.js) - Google Analytics