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

来源:互联网 发布:iframe 端口不同 跨域 编辑:程序博客网 时间:2024/06/05 09:37
  • 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构造方法");        }    }   

输出结果为:

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