您的位置:首页 > 职场人生

Java继承知识点总结(加面试题实例)

2017-08-16 21:46 477 查看
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子类,无论是调用子类的有参构造还是无参构造都会执行父类的无参构造。当子类和父类有同一个变量名字的变量时,如果是使用了多态则调用的是父类的变量。否则调用的是子类的变量。

补充:执行顺序,静态块>构造语句块>函数块
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息