- Java中的继承作为Java面向对象三大特性之一,知识点还是挺多的,对它进行了如下几点总结:
一、继承的概念及其限制
1. 概念
继承是从已有的类中派生出已有的类,继承来的类能吸收已有类的非私有属性和方法(行为),并能扩展新的属性和方法。通俗来说,继承就是分为了父类和子类,父类有时也成为超类(super class),子类又成为派生类;子类继承了父类的属性和方法,并且具有了父类所没有的。
public class Person {
public static void prt(String s) {
System.out.println(s);
}
Person() {
System.out.println("A Person.");
}
Person(String name) {
System.out.println("A person name is:" + name);
}
}
2. 限制
(1)Java的继承只能是单继承,一个子类只能继承一个父类;
(2)Java的继承允许多层继承,即子类又是另一个类的父类(可以理解 为爷爷、爸爸与儿子的关系);
二、继承中初始化的顺序
属性、方法、构造方法和自由块都是类中的成员,在创建类的对象时,类中各成员的执行顺序:
1.父类静态成员和静态初始化快,按在代码中出现的顺序依次执行。
2.子类静态成员和静态初始化块,按在代码中出现的顺序依次执行。
3. 父类的实例成员和实例初始化块,按在代码中出现的顺序依次执行。
4.执行父类的构造方法。
5.子类实例成员和实例初始化块,按在代码中出现的顺序依次执行。
6.执行子类的构造方法。
public class Test {
public static void main(String[] args) {
Son s = new Son();
}
}
class Parent{
{
System.out.println("parent中的初始化块");
}
static{
System.out.println("parent中static初始化块");
}
public Parent(){
System.out.println("parent构造方法");
}
}
class Son extends Parent{
{
System.out.println("son中的初始化块");
}
static{
System.out.println("son中的static初始化块");
}
public Son(){
System.out.println("son构造方法");
}
}
输出结果为:
parent中static初始化块
son中的static初始化块
parent中的初始化块
parent构造方法
son中的初始化块
son构造方法
三、子父类出现后,类中的成员都有了哪些特点:
1.成员变量:
当子父类中出现一样的属性时,子类类型的对象,调用该属性,值是子类的属性值。
如果想要调用父类中的属性值,需要使用一个关键字:super
This:代表是本类类型的对象引用。
Super:代表是子类所属的父类中的内存空间引用。
注意:子父类中通常是不会出现同名成员变量的,因为父类中只要定义了,子类就不用在定义了,直接继承过来用就可以了。
2. 成员函数:
当子父类中出现了一模一样的方法时,建立子类对象会运行子类中的方法。好像父类中的方法被覆盖掉一样。所以这种情况,是函数的另一个特性:覆盖(复写,重写)。
什么时候使用覆盖呢?
当一个类的功能内容需要修改时,可以通过覆盖来实现。
3. 构造函数:
发现子类构造函数运行时,先运行了父类的构造函数。为什么呢?
原因:子类的所有构造函数中的第一行,其实都有一条隐身的语句super();
super(): 表示父类的构造函数,并会调用于参数相对应的父类中的构造函数。而super():是在调用父类中空参数的构造函数。
为什么子类对象初始化时,都需要调用父类中的函数?(为什么要在子类构造函数的第一行加入这个super()?)
因为子类继承父类,会继承到父类中的数据,必须要看父类是如何对自己的数据进行初始化的。
所以子类在进行对象初始化时,先调用父类的构造函数,这就是子类的实例化过程。
注意:
子类中所有的构造函数都会默认访问父类中的空参数的构造函数,因为每一个子类构造内第一行都有默认的语句super();
如果父类中没有空参数的构造函数,那么子类的构造函数内,必须通过super语句指定要访问的父类中的构造函数。
如果子类构造函数中用this来指定调用子类自己的构造函数,那么被调用的构造函数也一样会访问父类中的构造函数。
问题:
super()和this()是否可以同时出现的构造函数中?
两个语句只能有一个定义在第一行,所以只能出现其中一个。
四.面试题
1. 腾讯面试题
public class A {
public int a = 0;
public void fun(){
System.out.println("-----A-----");
}
}
public class B extends A{
public int a = 1;
public void fun(){
System.out.println("-----B-----");
}
public static void main(String[] args){
A classA = new B();
System.out.println(classA.a);
classA.fun();
}
}
输出结果为:
0
-----B-----
解析:
java中变量不能重写,可以按如下口诀记忆
变量多态看左边,
方法多态看右边,
静态多态看左边。
2. 经典笔试题
class Father {
int x = 1;
public Father() {
System.out.println("father");
}
public Father(String s) {
System.out.println("father:" + s);
}
}
class Son extends Father {
int x = 2;
public Son() {
System.out.println("son");
}
public Son(String s) {
System.out.println("son:" + s);
}
}
public class Test {
public static void main(String[] args) {
// 都是new的无参构造方法
// 第一种情况,输出结果的顺序
Son f = new Son();
System.out.println(f.x);
/*
father
son
2
说明:没有使用多态,父类的引用指向子类的对象,还是会执行父类的无参构造方法。*/
// 第二种情况,输出结果的顺序
Father f = new Son();
System.out.println(f.x);
/*
father
son
1
说明:使用了多态,注意这里输出的是1。
*/
// 使用new的有参构造方法
Son f = new Son("hello");
System.out.println(f.x);
/*
father
son:hello
2
*/
Father f = new Son("hello");
System.out.println(f.x);
/*
father
son:hello
1
*/
}
}
解析:只要是new子类,无论是调用子类的有参构造还是无参构造都会执行父类的无参构造。当子类和父类有同一个变量名字的变量时,如果是使用了多态则调用的是父类的变量。否则调用的是子类的变量。
补充:执行顺序,静态块>构造语句块>函数块