多个类中存在相同属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属性和行为,只要继承那一个类即可。其中如图中所示,食草动物、食肉动物、兔子、羊、狮子、豹都可以称为子类,动物类称为父类、超类(superclass)或者基类。

继承描述的是事物之间的所属关系。例如,图中兔子属于食草动物,食草动物属于动物。可见,父类更通用,子类更具体。我们通过继承,可以使多种事物之间形成一种关系体系。

继承:就是子类继承父类的属性和行为,使得子类对象具有与父类相同的属性、相同的行为。子类可以直接访问父类中的非私有的属性和行为。

继承的格式

通过 extends 关键字,可以声明一个子类继承另外一个父类,定义格式如下:

修饰符 class 父类 {
  ...
}

修饰符 class 子类 extends 父类 {
  ...
}

来一个例子:

  1. // 父类
  2. public class Animal {
  3. String name;
  4. public void eat(){
  5. System.out.println("肚子饿了都要吃东西");
  6. }
  7. }
  8.  
  9. // 子类继承父类
  10. public class Lion extends Animal{
  11. public void printName(){
  12. System.out.println("名字:"+ name);
  13. }
  14. }
  15.  
  16. // 测试类
  17. public class Demo {
  18. public static void main(String[] args) {
  19. // 创建一个狮子类对象
  20. Lion lion = new Lion();
  21. // 为该Lion类的name属性进行赋值
  22. lion.name = "狮子王";
  23. // 调用该Lion类的printName()方法
  24. lion.printName(); // 名字:狮子王
  25. // 调用从Animal类继承来的eat()方法
  26. lion.eat(); // 肚子饿了都要吃东西
  27. }
  28. }

通过上面代码我们可以看出用继承的方式可以 提高代码的复用性。同时类与类之间产生了关系,是多态的前提,这个在接下来的章节说明。

继承后的特点

当类之间产生了关系后,其中各类中的成员变量和成员方法会产生了哪些影响呢?

一、成员变量

1、成员变量不重名

如果子类父类中出现不重名的成员变量,这时的访问是没有影响的。

  1. public class Fu {
  2. // Fu中的成员变量。
  3. int numFu = 10;
  4. }
  5.  
  6. public class Zi extends Fu {
  7. // Zi中的成员变量
  8. int numZi = 20;
  9. // Zi中的成员方法
  10. public void show() {
  11.   // 访问父类中的numFu,继承而来可以直接访问
  12.   System.out.println("Fu numFu="+numFu);
  13.   // 访问子类中的numZi
  14.   System.out.println("Zi numZi="+numZi);
      }
  15. }
  16.  
  17. public class Demo {
  18. public static void main(String[] args) {
  19. // 创建子类对象
  20. Zi z = new Zi();
  21. // 调用子类中的show方法
  22. z.show();
  23. }
  24. }
  25.  
  26. 演示结果:
  27. Fu num1 = 10
  28. Zi num2 = 20

2、成员变量重名

  1. public class Fu {
  2.  
  3. int numFu = 10;
  4.  
  5. int num = 100;
  6.  
  7. public void methodFu() {
  8. System.out.println(num);
  9. }
  10.  
  11. }
  12.  
  13. public class Zi extends Fu {
  14.  
  15. int numZi = 20;
  16.  
  17. int num = 200;
  18.  
  19. public void methodZi() {
  20. System.out.println(num);
  21. }
  22.  
  23. }
  24.  
  25. public class Demo {
  26.  
  27. public static void main(String[] args) {
  28. Fu fu = new Fu();
  29. System.out.println(fu.numFu); //
  30.  
  31. Zi zi = new Zi();
  32. System.out.println(zi.numFu); //
  33. System.out.println(zi.numZi); // 20
  34.  
  35. // 直接通过子类对象访问成员变量:等号左边是谁,就优先用谁,没有则向上找。
  36. System.out.println(zi.num); // 优先子类,200
  37. System.out.println(zi.abc); // 到处都没有,编译报错!
  38.  
  39. //间接通过成员方法访问成员变量:该方法属于谁,就优先用谁,没有则向上找。
  40. zi.methodZi(); // 200, 这个方法是子类的,优先用子类的没有再向上找
  41. zi.methodFu(); // 100,这个方法是在父类当中定义的,
  42. }
  43.  
  44. }

从上面代码来看,想用父类的num值又怎么办呢?

子父类中出现了同名的成员变量时,在子类中需要访问父类中非私有成员变量时,需要使用 super 关键字,修饰父类成员变量,类似于之前学过的 this 。

使用格式:super.父类成员变量名

因此子类的代码可以写为:

  1. public class Zi extends Fu {// Zi中的成员方法
  2. public void show() {
  3. System.out.println("Fu num="+super.num);
  4. System.out.println("Zi num="+num);
  5. }
  6. }
  7.  
  8. 演示结果:
  9. Fu num = 100
  10. Zi num = 200

注意事项:

  (1)Fu 类中的成员变量是非私有的,子类中可以直接访问。若Fu 类中的成员变量私有了,子类是不能直接访问的。通常编码时,我们遵循封装的原则,使用private修饰成员变量,那么如何访问父类的私有成员变量呢?

  答案是可以在父类中提供公共的getter方法和setter方法。

  (2)局部变量: 直接写成员变量名

  (3)本类的成员变量: this.成员变量名

  (4)父类的成员变量: super.成员变量名

因此子类的代码又可以修改为:

  1. public class Zi extends Fu {// Zi中的成员方法
  2. public void show() {
  3. int innerNum = 10; // 局部变量
  4. System.out.println("Fu num="+super.num);
  5. System.out.println("Zi num="+this.num);
  6. System.out.println("innerNum="+innerNum);
  7. }
  8. }

二、成员方法

1、成员方法不重名

  1. public class Fu {
  2.  
  3. public void methodFu() {
  4. System.out.println("父类方法执行!");
  5. }
  6.  
  7. }
  8.  
  9. public class Zi extends Fu {
  10.  
  11. public void methodZi() {
  12. System.out.println("子类方法执行!");
  13. }
  14.  
  15. }
  16.  
  17. public class Demo {
  18. public static void main(String[] args) {
  19.  
  20. Zi zi = new Zi();
  21.  
  22. zi.methodFu();
  23. zi.methodZi();
  24. }
  25. }
  26.  
  27. 运行结果:
  28. 父类方法执行!
  29. 子类方法执行!

如果子类父类中出现不重名的成员方法,这时的调用是没有影响的。对象调用方法时,会先在子类中查找有没有对应的方法,若子类中存在就会执行子类中的方法,若子类中不存在就会执行父类中相应的方法。

2、成员方法重名

先来一波代码:

  1. public class Fu {
  2.  
  3. public void methodFu() {
  4. System.out.println("父类方法执行!");
  5. }
  6.  
  7. public void method() {
  8. System.out.println("父类:重名方法执行!");
  9. }
  10.  
  11. }
  12.  
  13. public class Zi extends Fu {
  14.  
  15. public void methodZi() {
  16. System.out.println("子类方法执行!");
  17. }
  18.  
  19. public void method() {
  20. System.out.println("子类:重名方法执行!");
  21. }
  22.  
  23. }
  24.  
  25. public class Demo01ExtendsMethod {
  26.  
  27. public static void main(String[] args) {
  28. Zi zi = new Zi();
  29. zi.method(); // 子类:重名方法执行!
  30. }
  31.  
  32. }

上述代码中因为子类又method成员方法,所以执行了子类的method方法。其实跟之前的重名的成员变量类似:看子类有没有,没有就向父类查找。

重写

其实如果子类父类中出现重名的成员方法,这时的访问是一种特殊情况,叫做方法重写 (Override)

方法重写 :子类中出现与父类一模一样的方法时(返回值类型、方法名、参数列表都相同)会出现覆盖效果,也称为重写或者复写。声明不变,重新实现。

注意区分重写(Override)和重载(Overload):

重写(Override):方法的名称一样,参数列表【也一样】。
重载(Overload):方法的名称一样,参数列表【不一样】。
更优雅的覆盖重写的方式:
  1. public class Fu {
  2.  
  3. public void methodFu() {
  4. System.out.println("父类方法执行!");
  5. }
  6.  
  7. public void method() {
  8. System.out.println("父类:重名方法执行!");
  9. }
  10.  
  11. }
  12.  
  13. public class Zi extends Fu {
  14.  
  15. public void methodZi() {
  16. System.out.println("子类方法执行!");
  17. }
  18.  
  19. @Override
  20. public void method() {
  21. System.out.println("子类:重名方法执行!");
  22. }
  23.  
  24. }
方法覆盖重写的注意事项:
(1) 必须保证父子类之间方法的名称相同,参数列表也相同。
@Override:写在方法前面,用来检测是不是有效的正确覆盖重写。这个注解就算不写,只要满足要求也是正确的方法覆盖重写。
(2)子类方法的返回值必须【小于等于】父类方法的返回值范围。
(3)子类方法的权限必须【大于等于】父类方法的权限修饰符。
public > protected > (default) > private,实例代码如下:
  1. public class Fu {
  2. // 没写权限修饰符,就是default
  3. void method(){
  4. System.out.println("父类成员方法!");
  5. }
  6.  
  7. }
  8.  
  9. public class Zi extends Fu{
  10.  
  11. @Override
  12. public void method(){
  13. System.out.println("子类成员方法!");
  14. }
  15.  
  16. }
(4)用到 super.父类成员方法(),表示调用父类的成员方法,如:
  1. public class Zi extends Fu {
  2.  
  3. public void methodZi() {
  4. System.out.println("子类方法执行!");
  5. }
  6.  
  7. public void method() {
  8. super.method();
  9. }
  10.  
  11. }

 三、构造方法

当类之间产生了关系,其中各类中的构造方法,又产生了哪些影响呢?

构造方法的作用是初始化成员变量的。所以子类的初始化过程中,必须先执行父类的初始化动作。子类的构造方法中默认有一个 super() ,表示调用父类的构造方法,父类成员变量初始化后,才可以给子类使用。代码如下:

  1. public class Fu {
  2.  
  3. public Fu() {
  4. System.out.println("父类无参构造方法!");
  5. }
  6.  
  7. public Fu(int num) {
  8. System.out.println("父类有参构造方法!");
  9. }
  10.  
  11. }
  12.  
  13. public class Zi extends Fu {
  14.  
  15. public Zi() {
  16. super(); // 在调用父类无参构造方法
  17. System.out.println("子类无参构造方法!");
  18. }
  19.  
  20. }
  21.  
  22. public class Demo {
  23.  
  24. public static void main(String[] args) {
  25. new Zi();
  26. }
  27.  
  28. }
  29.  
  30. 运行结果:
  31. 父类无参构造方法!
  32. 子类无参构造方法!
继承关系中,父子类构造方法的访问特点:
(1)子类构造方法当中有一个默认隐含的“super()”调用,所以一定是先调用的父类构造后执行的子类构造。看代码:
  1. public class Fu {
  2. public Fu(){
  3. System.out.println("父类无参构造方法!");
  4. }
  5. public Fu(int num){
  6. System.out.println("父类有参构造方法!");
  7. }
  8. }
  9.  
  10. public class Zi extends Fu{
  11. public Zi(){
  12. System.out.println("子类无参构造方法!");
  13. }
  14. }
  15.  
  16. public class Demo {
  17. public static void main(String[] args) {
  18. new Zi();
  19. }
  20. }
  21.  
  22. 运行结果:
  1. 父类无参构造方法!
    子类无参构造方法!
(2)子类构造可以通过super关键字来调用父类重载构造。
(3)super的父类构造调用,必须是子类构造方法的第一个语句。不能一个子类构造调用多次super构造。代码如下:
  1. public class Zi extends Fu{
  2. public Zi(){
  3. System.out.println("子类无参构造方法!");
  4. }
  5. public void method(){
  6. // 错误写法,写在了子类成员方法中
  7. super(200)
  8. System.out.println("子类成员方法!");
  9. }
  10. }
  11.  
  12. 或者
  13.  
  14. public class Zi extends Fu{
  15. public Zi(){
  16. // 错误写法,存在了两个super
  17. super();
  18. super(200);
  19. System.out.println("子类无参构造方法!");
  20. }
  21. }
  22.  
  23. 或者
  24.  
  25. public class Zi extends Fu{
  26. public Zi(){
  27. System.out.println("子类无参构造方法!");
  28. // 错误写法,super应该写在第一个语句
  29. super(200);
  30. }
  31. }
因此,子类必须调用父类构造方法,不写则赠送super();写了则用写的指定的super调用,super只能有一个,还必须是第一个。

四、super和this

父类空间优先于子类对象产生

在每次创建子类对象时,先初始化父类空间,再创建其子类对象本身。目的在于子类对象中包含了其对应的父类空间,便可以包含其父类的成员,如果父类成员非private修饰,则子类可以随意使用父类成员。代码体现在子类的构造方法调用时,一定先调用父类的构造方法。理解图解如下:

1、super和this的含义

super :代表父类的存储空间标识(可以理解为父亲的引用)。
this :代表当前对象的引用(谁调用就代表谁)。

2、super和this的用法

(1)访问成员

this.成员变量 ‐‐ 本类的

super.成员变量 ‐‐ 父类的

this.成员方法名() ‐‐ 本类的

super.成员方法名()  --  父类的

(2)访问构造方法

this(...) ‐‐ 本类的构造方法
super(...) ‐‐ 父类的构造方法

五、继承的特点

1、Java只支持单继承,不支持多继承

//一个类只能有一个父类,不可以有多个父类。
class C extends A{} //ok
class C extends A,B... //error

2、Java支持多层继承(继承体系)

class A{}
class B extends A{}
class C extends B{}

顶层父类是Object类。所有的类默认继承Object,作为父类。并且子类和父类是一种相对的概念。

【Java】面向对象之继承的更多相关文章

  1. Java面向对象之继承(一)

    目录 Java面向对象之继承 引言 继承的特点 语法格式 父子类的关系 继承要点 重写父类方法 继承中的构造器 继承中的super关键字 ... Java面向对象之继承 继承是面向对象的第二大特征,是 ...

  2. Java面向对象的继承

    继承也是面向对象的又一重要特性,继承是类于类的一种关系,通俗来说狗属于动物类,那么狗这个类就继承了动物类 java中的继承是单继承的,一个类只能继承与一个父类 子类继承父类之后,子类就拥有了父类的所有 ...

  3. JAVA面向对象之继承

    继承: 子类继承父类的特征和行为,使得子类对象(实例)具有父类的实例域和方法 class 子类 extends 父类 { } 继承的特性: 子类拥有父类非private的属性,方法. 子类可以拥有自己 ...

  4. Java:[面向对象:继承,多态]

    本文内容: 继承 多态 首发时期:2018-03-23 继承: 介绍: 如果多个类中存在相同的属性和行为,可以将这些内容抽取到单独一个类中,那么多个类(子类)无需再定义这些属性和行为,只要继承那个类( ...

  5. Java面向对象之继承extends 入门实例

    一.基础概念 (一)继承的好处: 1.继承:提高了代码的复用性. 2.让类与类之间产生了关系,给多态这个特征提供了前提. (二)继承的种类: 1.单继承:一个类只能有一个父类. 2.多继承:一个类可以 ...

  6. Java面向对象_继承——基本概念以及管理化妆品实例分析

    一.继承的基本概念: 1.继承是面向对象三大特征之一 2.被继承的类成为父类(超类),继承父类的类成为子类(派生类) 3.继承是指一个对象直接使用另一个对象的属性和方法 4.通过继承可以实现代码重用 ...

  7. Java面向对象(继承、抽象类)

    面向对象 今日内容介绍 u 继承 u 抽象类 第1章 继承 1.1 继承的概念 在现实生活中,继承一般指的是子女继承父辈的财产.在程序中,继承描述的是事物之间的所属关系,通过继承可以使多种事物之间形成 ...

  8. java面向对象之 继承 Inheritance

    对象的一个新类可以从现有的类中派生,这个过程称为类继承.新类继承了原始类的特性,新类称为原始类的派生类(子类),而原始类称为新类的基类(父类).派生类可以从它的基类那里继承方法和实例变量,并且类可以修 ...

  9. Java 面向对象之继承和重写OverWrite,重写和重载的区别,抽象类

    01继承的概述 A:继承的概念 a:继承描述的是事物之间的所属关系,通过继承可以使多种事物之间形成一种关系体系 b:在Java中,类的继承是指在一个现有类的基础上去构建一个新的类,构建出来的新类被称作 ...

  10. Java面向对象之继承,方法重写,super关键字,员工类系列继承题

    在程序中,如果想声明一个类继承另一个类,需要使用extends关键字. 格式: class 子类 extends 父类 {} 继承的好处 1.继承的出现提高了代码的复用性,提高软件开发效率. 2.继承 ...

随机推荐

  1. SpringBoot系列:Spring Boot集成Spring Cache,使用RedisCache

    前面的章节,讲解了Spring Boot集成Spring Cache,Spring Cache已经完成了多种Cache的实现,包括EhCache.RedisCache.ConcurrentMapCac ...

  2. xss姿势利用

    1.定位页面可以出现xss的位置 可能会出现联合点利用 一个页面多个存储位置或者一个页面多个参数联合利用 例如输入xss 查看页面源码页面里有多个xss 或者多个参数显示 可以利用 需要注意的是有的是 ...

  3. Maven -- 使用Myeclipse创建Maven项目

    使用Myeclipse创建Maven项目有如下几种方式: 1.创建Maven Java项目 1.1 选择新建Maven项目 1.2.选择创建简单项目 1.3.填写项目信息 1.4.创建成功后项目目录结 ...

  4. 05 Node.js学习笔记之发送文件数据

    这章学习在NodeJs中如何将Html文件发送到客户端上,以及定义Content-Type内容类型 //1.载入http和fs模块 var http=require("http") ...

  5. HTTP Catcher

    HTTP Catcher HTTP Catcher 是一个 Web 调试工具.它可以拦截.查看.修改和重放来自 iOS 系统的 HTTP 请求. 你不需要连接电脑,HTTP Catcher 可以在后台 ...

  6. 使用JRebel插件实现SpringBoot应用代码热加载

    前言 在实际的开发过程中,我们经常修改代码之后,手动的重启项目,查看修改效果.那么有没有一种方式能够快速的.自动的帮我们将修改代码自动更新,避免手动重启,从而提高开发效率呢?是有的,在我之前的文章里面 ...

  7. Java网络编程(一)Socket套接字

    一.基础知识 1.TCP:传输控制协议. 2.UDP:用户数据报协议. 二.IP地址封装 1.InetAddress类的常用方法 getLocalHost() 返回本地主机的InetAddress对象 ...

  8. TCP UDP基本编程(一)

    tcp udp均可以用来网络通信,在使用之前建议先搜索一下相关网络连接的基本知识,可以更好的理解和使用,tcp建议看下如下文章:https://blog.csdn.net/chuangsun/arti ...

  9. day3,用户交互,input的应用

    1.与用户交互 输入:input() python2.x版本 input后面家的东西要声明输入的类型       >>> input(">>:")   ...

  10. 有Bug?你的代码神兽选对了吗

    传说每一个优秀的程序员都有自己专属的镇码神兽 通过 工具网址 http://www.makepic.net/Tool/Image2ascii.html 将自己喜欢的神兽图片转成文本, 可以选择不同的分 ...