今天面试遇到的一道题,啊还是答错了。回来敲一次代码,加深记忆。
情景(一)父类与子类里面的方法都是public void
A.java
public class A{ public void a(){System.out.println(getClass().getSimpleName()+":a()");} public void b(){System.out.println(getClass().getSimpleName()+":b()");} public void c(){System.out.println(getClass().getSimpleName()+":c()");} }
B.java
class B extends A{ public void a(){System.out.println(getClass().getSimpleName()+":a()");} public void b(){System.out.println(getClass().getSimpleName()+":b()");} public void d(){System.out.println(getClass().getSimpleName()+":d()");} public static void main(String[] args){ A a = new B(); a.a(); a.b(); //a.d();编译会报错,因为A类中并没有d()方法 System.out.println(); B b = (B)a; b.a(); b.b(); b.c(); } }
输出:编译看声明类型,运行看引用类型。(编译看左边,运行看右边。)
B:a() B:b() B:a() B:b() B:c()//虽然B类里面没有重写c()方法,但是运行的时候依然输出B:c()而不是A:c()
情景(二)修改方法b()为static修饰
A.java(其中b()方法的输出也进行修改,因为getClass()方法不是静态的,静态方法不能调用非静态方法)
public class A{ public void a(){System.out.println(getClass().getSimpleName()+":a()");} public static void b(){System.out.println(Thread.currentThread().getStackTrace()[1].getClassName()+":b()");} public void c(){System.out.println(getClass().getSimpleName()+":c()");} }
B.java
class B extends A{ public void a(){System.out.println(getClass().getSimpleName()+":a()");} public static void b(){System.out.println(Thread.currentThread().getStackTrace()[1].getClassName()+":b()");} public void d(){System.out.println(getClass().getSimpleName()+":d()");} public static void main(String[] args){ A a = new B(); a.a(); a.b();//a.d(); System.out.println(); B b = (B)a; b.a(); b.b();//实际上并不是重写了A里面的b方法,而是一个新的静态方法 b.c(); } }
输出:可以看到第二行输出 发生了变化,A a=new B(); 输出了A中的b()方法。 B b=(B)a;输出B中的b()方法。
值得注意的是,这个b()方法并不是重写了A中的b().
B:a() test.A:b() B:a() test.B:b() B:c()
情景(三)增加一些变量
A.java
public class A{ int a=1; static int a_static = 10; final static int a_final =100; public void a(){System.out.println(getClass().getSimpleName()+":a()");} public static void b(){System.out.println(Thread.currentThread().getStackTrace()[1].getClassName()+":b()");} public void c(){System.out.println(getClass().getSimpleName()+":c()");} }
B.java
class B extends A{ int a =2; static int a_static =20; final static int a_final =200; public void a(){System.out.println(getClass().getSimpleName()+":a()");} public static void b(){System.out.println(Thread.currentThread().getStackTrace()[1].getClassName()+":b()");} public void d(){System.out.println(getClass().getSimpleName()+":d()");} public static void main(String[] args){ A a = new B(); a.a(); a.b();//a.d(); System.out.println("a = "+a.a); System.out.println("a_static = "+a.a_static); System.out.println("a_final = "+a.a_final); System.out.println(); B b = (B)a; b.a(); b.b();//实际上并不是重写了A里面的b方法,而是一个静态方法 b.c(); System.out.println("a = "+b.a); System.out.println("a_static = "+b.a_static); System.out.println("a_final = "+b.a_final); } }
输出:变量不管用什么修饰符,值都看声明类型,即左边。
B:a() test.A:b() a = 1 a_static = 10 a_final = 100 B:a() test.B:b() B:c() a = 2 a_static = 20 a_final = 200
非静态方法,编译看左边,运行看右边。
静态方法,编译看左边,运行看左边。(特殊地,如果子类中声明了一个一模一样的静态方法,则执行子类中的)
成员变量,编译看左边,运行看左边。