Java基础——面向对象(二)继承

版权声明:Dirichlet_zju https://blog.csdn.net/Dirichlet_zju/article/details/85617551

一、继承概述

1.使用extends关键字

继承后子类可以使用父类成员

2.继承的好处

  • 提高代码的复用性
  • 让类与类之间产生了关系,给第三个特征多态提供了前提

3.Java中继承的特点 

Java中支持单机城。不直接支持多继承,但对C++中的多继承机制进行改良。

思考:为什么Java要对多继承机制进行改良?

这个问题也就是多继承机制有哪些缺点?简单来看,继承过程中产生的多个父类中如果具有相同的成员,会产生调用不确定性。因此在Java中是通过“多实现”的方式来体现,即多重继承。

当要使用一个继承体系时:

1.查看该体系中的顶层类,了解该体系的基本功能。

2.创建体系中的最子类对象,完成功能的使用。

 二、定义继承

1.什么时候定义继承?

当类与类之间存在着所属关系的时候,就定义继承。如果xxx是yyy中的一种->xxx extends yyy

2.子父类中成员变量的特点

aaa) 成员变量

  • 子类中有父类的相同名称的变量,那么直接调用时调用的是字类中的变量。当需要区分时,用super关键字区分父类。
  • 子类不能直接访问父类中的私有内容(private所修饰),但可以通过get(); set();实现。 

super关键字:super.父类成员,用法和this很相像。

和this区别:

  • this代表一个本类对象的引用。
  • super代表一个父类空间。

bbb) 成员函数

当子父类中成员函数一模一样时,会运行子类的函数。这种现象称为覆盖操作。这是函数在子父类中的特性。当然也可以super

覆盖:

函数有两个特性:

  1. 重载,同一个类中
  2. 覆盖,自类中。覆盖也称重写,覆写,override。

覆盖的注意事项:

  1. 子类方法覆盖父类方法时,子类权限必须大于等于父类权限——权限只能放大
  2. 静态只能覆盖静态,或被静态覆盖——同非同静

思考:什么时候使用覆盖操作?

当对一个类进行子类的扩展时,子类需要保留父类的功能声明,但是要定义子类中该功能的特有内容时,就使用覆盖操作完成。

ccc) 构造函数

在子类构造对象时,发现,访问子类构造函数时,父类也运行了,为什么?

在子类的构造函数中第一行有一个默认的隐式语句--->super(); 


3.子类的实例化过程:

自类中所有的构造函数默认都会访问父类中的空参数构造函数。

为什么子类实例化的时候要访问父类中的构造函数呢?

  • 因为子类继承了父类,获取到了父类中的内容(属性),所以在使用父类之前,要先看弗雷是如何对自己的内容进行初始化的。
  • 如果父类中没有定义空参数构造函数,那么子类构造函数必须用super明确要调用父类哪个构造函数。

注意:super语句必须要定义在子类构造函数的第一行。因为父类的初始化动作要先完成。但是,如果有了this()调用本类构造函数,默认的super就没有了,但是一定有一处默认调用super()。

一个对象实例化过程:

Person p = new Person();

  1. JVM会读取指定的路径下的Person.class文件,并加载进内存,并会先加在Person的父类(如果有直接父类的情况下)。
  2. 在堆内存中开辟空间,分配地址。
  3. 并在对象空间中,对对象中的属性进行默认初始化。
  4. 调用对应的构造函数进行初始化。
  5. 在构造函数中,第一行会先调用父类中构造函数进行初始化。
  6. 父类初始化完毕后,再对子类的属性进行显示初始化。
  7. 在进行子类构造函数的特定初始化。
  8. 初始化完毕后,将地址赋值给引用变量

 三、final关键字

  1. final是一个修饰符,可以修饰类、方法、变量
  2. final修饰的类不可以被继承。
  3. final修饰的方法不可以被覆盖。
  4. final修饰的变量是一个常量,只能赋值一次。

思考:为什么要用final修饰变量?

其实在程序中如果一个数据是固定的,那么直接使用这个数据就可以了,但是这样阅读性差,所以给该数据起个名称。而且这个变量名称的值不能变化,所以加上final固定。

写法规范:常量所有字母都大写,多个单词,中间用_连接。 

猜你喜欢

转载自blog.csdn.net/Dirichlet_zju/article/details/85617551