Java面试题之多态

面向对象编程有三大特性:封装、继承、多态。
1. 封装隐藏了类的内部实现机制,可以在不影响使用的情况下改变类的内部结构,同时也保护了数据。对外界而已它的内部细节是隐藏的,暴露给外界的只是它的访问方法。
2. 继承是为了重用父类代码。两个类若存在IS-A的关系就可以使用继承,同时继承也为实现多态做了铺垫。
3. 所谓多态就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定。因为在程序运行时才确定具体的类,这样,不用修改源程序代码,就可以让引用变量绑定到各种不同的类实现上,从而导致该引用调用的具体方法随之改变,即不修改程序代码就可以改变程序运行时所绑定的具体代码,让程序可以选择多个运行状态,这就是多态性。

    多态性总结:指向子类的父类引用由于向上转型了,它只能访问父类中拥有的方法和属性,而对于子类中存在而父类中不存在的方法,该引用是不能使用的,尽管是重载该方法。若子类重写了父类中的某些方法,在调用该些方法的时候,必定是使用子类中定义的这些方法(动态连接、动态调用)。


例子:

Java面试题之多态_第1张图片

运行结果为:22,34,17

分析: 

   Step 1:  new B()构造一个B类的实例

               super(5) => 显示调用父类A带参的构造函数,该构造函数调用setValue(v),由于多态性,这里调用的实际是B类的setValue方法,于是调用B类中的setValue方法 ,将B实例的value值设置为2 x 5 = 10

               setValue(getValue() - 3) =>先执行getValue方法,B类中没有重写getValue方法,因此调用父类A的getValue方法。

                1.调用getValue方法之前,B的成员变量value值为10;

                2.value++ 执行后, B的成员变量value值为11,此时开始执行到return语句,将11这个值作为getValue方法的返回值返回出去;

                3.但是由于getValue块被try finally块包围,因此finally中的语句无论如何都将被执行,所以步骤2中11这个返回值会先暂存起来,到finally语句块执行完毕后再真正返回出去;

                4.finally语句块中 this.setValue(value)方法调用的是B类的setValue方法,将B的value值设置成为了2 * 11 = 22,第一个打印出来的结果为22( System.out.println(value) 的结果)

Step2:  new B().getValue()

          B类中没有独有的getValue方法,此处调用A的getValue方法。同Step 1,

            1.调用getValue方法之前,B的成员变量value值为16

            2.value++ 执行后, B的成员变量value值为17,此时执行到return语句,会将17这个值作为getValue方法的返回值返回出去;

            3.但是由于getValue块被try finally块包围而finally中的语句无论如何都一定会被执行,所以步骤2中17这个返回值会先暂存起来,到finally语句块执行完毕后再真正返回出去;

           4.finally语句块中继续和上面说的一样: this.setValue(value)方法调用的是B类的setValue()方法将B的value值设置成为了2 * 17 = 34,第二个打印出来的结果为34( System.out.println(value) 的结果)

           5.finally语句执行完毕 会把刚刚暂存起来的17返回出去,因此new B().getValue()最终的返回值是17(System.out.println(new B().getValue())的结果)

 

你可能感兴趣的:(Java面试题)