老生常谈 Java中的继承(必看)

Java作为一面向对象的语言,具备面向对象的三大特征——继承,多态,封装。

继承顾名思义,继任,承接,传承的意思。面向对象的语言有一个好处,就是可以用生活中的例子来说明面向对象的特性。那么我们先来看看生活中的继承关系有哪些?最常见的:父母子女;汽车,电动车,自行车和车。无论哪种车,都有具备车的特性。再比如说:家里面的电饭锅,电磁炉,电冰箱。他们都属于电器类,都具有名字这个属性,也都需要用电这个方法。如果在程序中我们一个个类去把这些重复的代码都写上去,那不是浪费时间和精力吗?联系之前的知识,我们能够从一个个对象中抽象出来一个类。那么我们也应该能够从具有包含关系的一个个类中抽象出一个具有共同属性和方法的类,也就是父类。比如说无论是三角形,矩形还是圆形,他们都有求边长的方法,那么就可以抽象出一个父类图形类,类中有一个求边长的方法。Java中的继承使用的是extends关键字,继承的类叫做子类(派生类或者超类),被继承的类叫做父类(或者基类)。凡是这种可以有包含关系的类都能实现继承关系。

下面是继承的格式:

public class 子类类名 extends 父类类名{
}

来看一个继承的简单例子:

//父类
public class Person {

  private String name;
  public int age;
  protected char sex;
  String country;

  public String getName(){
    return name;
  }
  public void setName(String name){
    this.name=name;
  }

  public void speak(){
    System.out.println(name+"正在说话!");
  }
}
//子类
public class Teacher extends Person {

}
//测试类
public class Test {
  public static void main(String args[]){
    Teacher t=new Teacher();//实例化子类
//    t.name="张三";//编译器报错
    t.sex='男';
    t.age=10;
    t.country="中国";
    t.setName("张三");
    t.getName();
    t.speak();
  }
}

运行后发现可以正常输出,也就是说子类里面在没有定义任何属性和方法的情况下,可以使用由父类继承来的属性和方法,这也就说明了继承的实现。那么子类都从父类继承了那些内容呢?答案是:子类类可以继承父类所有的属性和方法。这里可能就会有人疑惑了,那测试类中的name属性不是报错了吗?那么私有的属性是不是不能够被继承呢?如果说,子类没有继承父类的name这个属性,那么子类中就不存在name这个属性。既然不存在,那么子类中的setName、getName、speak这三个方法应该都会报错才对,但事实是并没有报错,所以Java中子类是可以继承父类中的所有的方法和属性值的。Java中的访问修饰符是用于限制类中的属性或者方法的访问权限的,与是否被继承并没有直接关系。这才是name属性值报错的原因。

当然子类是可以定义自己特有属性和方法的,这个并没有任何问题。弄清楚了子类能够继承父类那些东西之后,我们给父类加上这么一段代码:

public Person(String name){
  this.name=name;
}

给父类加上了这个构造器之后发现,子类报错了!这是为什么呢?原来Java在实例化子类对象的时候会通过子类的无参构造器调用父类的无参构造器,当给父类提供了一个有参构造器,JVM不会再为父类提供默认的无参构造器,子类实例化对象找不到父类无参构造器编译器自然会报错了。

下面我们来验证一下:

给父类加上无参构造器:

public Person(){
   System.out.println("父类构造器被调用了");
}

控制台输出结果如下:

这也就证明了子类会调用父类的无参构造器,也就是说子类在实例化的时候是产生了两个对象(这里不考虑Object),一个子类对象,一个父类对象。

好到这里相信读者对类的继承已经基本清楚了。我们返回我们刚才使用的例子,我们定义了人这个类,类里面有说话(speak)的方法。试想定义几个类:中国人,美国人,俄罗斯人,他们都继承人这个类,都有说话的方法,但是他们说话的方法一样吗?可以直接使用父类的说话方法吗?中国人说话用中文,美国人用英语,俄罗斯用俄语,显然不能用同一个方法。也就是说当父类的方法不满足子类的需求的时候,那怎么办?这里就可以用到方法的重写。

先来看看方法重写的条件:

1.必须要有继承关系;

2.重写方法时子类方法的访问修饰符必须要大于或者等于父类方法的访问修饰符;

3.重写方法时子类方法的返回值类型,方法名,参数都必须要和父类的一致。

满足了这些条件就叫做方法的重写。来看例子:

public class Chinese extends Person{
  public viod speak(){
    System.out.println(name+"是中国人说的是中文,说话方式不同了。");
  }
}

这样就完成了方法的重写,在测试类中实例化Chinese对象,调用speak方法就会输出:xxx是中国人说的是中文,说话方式不同了。要注意的是:方法发生重写后,使用子类对象调用的speak方法是子类重写后的方法,而不再是父类的方法。方法的调用取决于new关键字后面的类,如果是父类,那就是调用父类的方法,如果是子类,那就调用子类重写后的方法。如果这时仍然想调用父类的方法,可以使用super关键字进行调用。把代码改成下面这样:

public class Chinese extends Person{
  public viod speak(){
    super.speak();
    System.out.println(name+"是中国人说的是中文,说话方式不同了。");
  }
}

这时就会输出两句话:xxx正在说话

xxx是中国人说的是中文,说话方式不同了。

这里需要注意一下重写跟重载的区别主要是条件:

重写:上面已经列过条件,这里就不在赘述。重写的重要用途是拓展父类的方法,以满足子类自己的需求。

重载:条件是:1.同类或者继承关系的类中;2.方法名相同,但是方法的参数必须不同。方法的重载重要是为了处理不同类型的数据。

另外类的继承还有一个优势就是——Java的自动转型。在Java中当小范围的数据向大范围的数据转换时,就会发生自动转型。自动转型的优势就在于我可以在一个类中定义一个方法,方法的参数是父类类型,这样无论有多少个子类,那就都能调用这个方法,这样就极大的提高了程序的扩展性。比如说,还是我们一直用着的这个例子,假如现在有一个外星来客,要教会人类一种很牛逼的技术,如果只有一两种人,那可以在外星人类中写两个教的方法,但是如果有100种1000种人呢?不可能写个1000种方法吧?

那么就可以使用下面这个方法:

public class Alien{
  public void teach(Person p){
    p.study();
  }
}

这样无论有多少种人,我们都可以直接传进去不同的子类对象,通过自动转型调用各自的学习方法,这样岂不是美滋滋。当然这里也会有问题,如果使用自动转型调用子类特有的方法时会出错,但是这是由Java的编译机制所产生的问题,自动转型有其优势,我们需扬长避短就好。

总结:有包含关系的类都可以使用继承,子类可以继承父类的所有属性和方法,继承可以提高代码的重用性和程序的拓展性。重写可以拓展父类的方法,更好的适应子类的需要,Java的自动转型能够大量简化代码,却也存在问题(当然不影响我们使用)。

以上这篇老生常谈 Java中的继承(必看)就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • C++/java 继承类的多态详解及实例代码

    C++/java 继承类的多态详解 学过C++和Java的人都知道,他们二者由于都可以进行面向对象编程,而面向对象编程的三大特性就是封装.继承.多态,所有今天我们就来简单了解一下C++和Java在多态这方面的不同. 首先我们各看一个案例. C++ //测试继承与多态 class Animal { public: char name[128]; char behavior[128]; void outPut() { cout << "Animal" << endl

  • 详解java中继承关系类加载顺序问题

    详解java中继承关系类加载顺序问题 实例代码: /** * Created by fei on 2017/5/31. */ public class SonClass extends ParentClass{ public SonClass(){ System.out.println("SonClass's constructor"); } { System.out.println("SonClass's block");} static { System.out

  • 详解 Java继承关系下的构造方法调用

    详解 Java继承关系下的构造方法调用 在Java中创建一个类的对象时,如果该类存在父类,则先调用父类的构造方法,然后再调用子类的构造方法.如果父类没有定义构造方法,则调用编译器自动创建的不带参数的默认构造方法.如果父类定义了public的无参的构造方法,则在调用子类的构造方法前会自动先调用该无参的构造方法.如果父类只有有参的构造方法,没有无参的构造方法,则子类必须在构造方法中必须显式调用super(参数列表)来指定某个有参的构造方法.如果父类定义有无参的构造方法,但无参的构造方法声明为priv

  • java 中继承和多态详细介绍

    继承和多态 一.this super关键字 1.this: 可以在构造器中的第一代码中调用本类中的其他构造器.this(参数) 非类方法参数中隐式传入的参数,表示调用当前方法的对象. 2.super: 可以在构造器的第一句代码调用父类的构造器.super(参数). 非静态方法中表示继承的父类对象,可以调用父类方法和属性. 二.方法的覆写: 子类重新实现了和父类一样的方法.访问修饰和异常都必须至少和父类的相同或者更大的范围. 三.方法的重载: 相同的方法的名字不同的参数列表. 四.多态: java

  • Java IO流体系继承结构图_动力节点Java学院整理

    Java IO体系结构看似庞大复杂,其实有规律可循,要弄清楚其结构,需要明白两点: 1. 其对称性质:InputStream 与 OutputStream, Reader 与 Writer,他们分别是一套字节输入-输出,字符输入-输出体系 2. 原始处理器(适配器)与链接流处理器(装饰器) 其结构图如下: Reader-Writer体系 1. 基类 InputStream与OutputStream是所有字节型输入输出流的基抽象类,同时也是适配器(原始流处理器)需要适配的对象,也是装饰器(链接流处

  • Java 继承与多态的深入理解

    Java 继承与多态的深入理解 1.  什么是继承,继承的特点? 子类继承父类的特征和行为,使得子类具有父类的各种属性和方法.或子类从父类继承方法,使得子类具有父类相同的行为. 特点:在继承关系中,父类更通用.子类更具体.父类具有更一般的特征和行为,而子类除了具有父类的特征和行为,还具有一些自己特殊的特征和行为. 在继承关系中.父类和子类需要满足is-a的关系.子类是父类. 表示父类和子类的术语:父类和子类.超类和子类.基类和派生类,他们表示的是同一个意思. 2.  为什么需要继承?什么时候应该

  • 详解Java中的封装、继承、多态

    封装 在如何理解面向对象这篇文章中,提到所谓的封装就是"功能都给你做好了,你不必去理解它是怎么写出来的,直接使用即可.".但你得清楚一点,那就是这句话是相对于使用者来说的,而作为开发者,封装就得我们自己来干. 那么作为开发者,我们应该如何去封装呢?其实你应该反过来问,他们应该如何去使用,这样一想会简单很多,作为使用者,自然是希望越简单越好,也就是说,一些复杂的东西,我们不应该让使用者去操作,那也就是说我们应该把复杂的,以及不必要的参数给它封死,不让使用者去操作. 为什么不让使用者去操作

  • Java中继承thread类与实现Runnable接口的比较

    Java中线程的创建有两种方式:  1.  通过继承Thread类,重写Thread的run()方法,将线程运行的逻辑放在其中 2.  通过实现Runnable接口,实例化Thread类 在实际应用中,我们经常用到多线程,如车站的售票系统,车站的各个售票口相当于各个线程.当我们做这个系统的时候可能会想到两种方式来实现,继承Thread类或实现Runnable接口,现在看一下这两种方式实现的两种结果. package com.threadtest; class MyThread extends T

  • java继承中的构造方法实例解析

    本文实例讲述了java继承中的构造方法.分享给大家供大家参考.具体如下: 继承中的构造方法: 1.子类的构造过程中必须调用其基类的构造方法. 2.子类可以在自己的构造方法中使用super(argument_list)调用基类的构造方法. 2.1.使用this(argument_list)调用本类的另外构造方法.   2.2.如果调用super,必须写在子类构造方法的第一行. 3.如果子类的构造方法中没有显示的调用基类的构造方法,则系统默认调用基类的无参数构造方法. 4.如果子类构造方法中既没有显

  • 老生常谈 Java中的继承(必看)

    Java作为一面向对象的语言,具备面向对象的三大特征--继承,多态,封装. 继承顾名思义,继任,承接,传承的意思.面向对象的语言有一个好处,就是可以用生活中的例子来说明面向对象的特性.那么我们先来看看生活中的继承关系有哪些?最常见的:父母子女:汽车,电动车,自行车和车.无论哪种车,都有具备车的特性.再比如说:家里面的电饭锅,电磁炉,电冰箱.他们都属于电器类,都具有名字这个属性,也都需要用电这个方法.如果在程序中我们一个个类去把这些重复的代码都写上去,那不是浪费时间和精力吗?联系之前的知识,我们能

  • 老生常谈Eclipse中的BuildPath(必看篇)

    什么是Build Path? Build Path是指定Java工程所包含的资源属性集合. 在一个成熟的Java工程中,不仅仅有自己编写的源代码,还需要引用系统运行库(JRE).第三方的功能扩展库.工作空间中的其他工程,甚至外部的类文件,所有这些资源都是被这个工程所依赖的,并且只有被引用后,才能够将该工程编译成功,而Build Path就是用来配置和管理对这些资源的引用的. Build Path一般包括: JRE运行库 第三方的功能扩展库(*.jar格式文件) 其他的工程 其他的源代码或Clas

  • 全面理解JavaScript中的继承(必看)

    JavaScript中我们可以借助原型实现继承. 例如 function baz(){ this.oo=""; } function foo(){ } foo.prototype=new baz(); var myFoo=new foo(); myFoo.oo; 这样我们就可以访问到baz里的属性oo啦.在实际使用中这个样不行滴,由于原型的共享特点(数据保存在了堆上), 所有实例都使用一个原型,一但baz的属性有引用类型就悲剧了,一个实例修改了其他实例也都跟着变了...wuwuwu 自

  • 老生常谈java垃圾回收算法(必看篇)

    1.引用计数法(Reference Counting Collector) 1.1算法分析 引用计数是垃圾收集器中的早期策略.在这种方法中,堆中每个对象实例都有一个引用计数.当一个对象被创建时,且将该对象实例分配给一个变量,该变量计数设置为1.当任何其它变量被赋值为这个对象的引用时,计数加1(a = b,则b引用的对象实例的计数器+1),但当一个对象实例的某个引用超过了生命周期或者被设置为一个新值时,对象实例的引用计数器减1.任何引用计数器为0的对象实例可以被当作垃圾收集.当一个对象实例被垃圾收

  • 一个牛人给Java初学者的建议(必看篇)

    给初学者之一:浅谈Java及应用学java 从不知java为何物到现在一个小小的j2ee项目经理虽说不上此道高手,大概也算有点斤两了吧每次上网,泡bbs逛论坛,没少去java相关的版 面总体感觉初学者多,高手少,精通的更少由于我国高等教育制度教材陈旧,加上java自身发展不过十年左右的时间还有一个很重要的原因就是java这门语 言更适合商业应用所以高校里大部分博士老师们对此语言的了解甚至不比本科生多在这种环境下,很多人对java感到茫然,不知所措,不懂java能做什么即 便知道了java很有用,

  • 老生常谈C++ 中的继承

    继承 1 什么是继承 1.1 继承的概念 继承机制是面向对象程序设计使代码可以复用的最重要的手段,这个机制允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称为派生类.继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程.以前解除的都是函数复用,继承是类设计层次的复用. 代码演示如下 #include <iostream> #include <string> using namespace std; class Person { public: v

  • 老生常谈PHP数组函数array_merge(必看篇)

    很久之前就用到过这个函数,只不不过是简单的用用而已并没有做太深入的研究 今天在翻阅别人博客时看到了对array_merge的一些使用心得,故此自己来进行一次总结. array_merge是将一个或者多个数组进行合并. 这个函数多用于在从数据库中取出的结果集的合并操作. 参数配置也很简单array_merge(arr1,arr2,arrN) 注意此处的参数必须为数组,否则会报错. 虽然,看起来很简单,但是其中也有不少的坑. 我们将从单数组和多数组的方向来进行分析. 1.多个数组进行合并操作(自定义

  • RSA加密算法java简单实现方法(必看)

    简单完整的代码,通过这个代码你将对RSA加密算法在Java中的实现方法有一个初步的了解,这个类,你可以直接使用,水平高的,就自己修改完善下代码. package security; import java.security.*; import java.security.spec.*; import java.security.interfaces.*; import javax.crypto.spec.*; import javax.crypto.interfaces.*; import ja

  • 老生常谈java中的fail-fast机制

    在JDK的Collection中我们时常会看到类似于这样的话: 例如,ArrayList: 注意,迭代器的快速失败行为无法得到保证,因为一般来说,不可能对是否出现不同步并发修改做出任何硬性保证.快速失败迭代器会尽最大努力抛出 ConcurrentModificationException.因此,为提高这类迭代器的正确性而编写一个依赖于此异常的程序是错误的做法:迭代器的快速失败行为应该仅用于检测 bug. HashMap中: 注意,迭代器的快速失败行为不能得到保证,一般来说,存在非同步的并发修改时

  • 老生常谈java中的Future模式

    jdk1.7.0_79 本文实际上是对上文<简单谈谈ThreadPoolExecutor线程池之submit方法>的一个延续或者一个补充.在上文中提到的submit方法里出现了FutureTask,这不得不停止脚步将方向转向Java的Future模式. Future是并发编程中的一种设计模式,对于多线程来说,线程A需要等待线程B的结果,它没必要一直等待B,可以先拿到一个未来的Future,等B有了结果后再取真实的结果. ExecutorService executor = Executors.

随机推荐