• 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子类,无论是调用子类的有参构造还是无参构造都会执行父类的无参构造。当子类和父类有同一个变量名字的变量时,如果是使用了多态则调用的是父类的变量。否则调用的是子类的变量。
补充:执行顺序,静态块>构造语句块>函数块