还要理解类编译的class文件中字节码的方法调用指令。
(1)invokestatic:调用静态方法
(2)invokespecial:调用实例构造器方法,私有方法。
(3)invokevirtual:调用所有的虚方法。
(4)invokeinterface:调用接口方法,会在运行时再确定一个实现此接口的对象。
(5)invokedynamic:先在运行时动态解析出调用点限定符所引用的方法,然后再执行该方法。
非虚方法:不能被重写或者说覆盖的方法,指的是构造方法、静态方法、私有方法和final 修饰的方法。
虚方法:则是能被重写的方法,一般指的是实例方法。
为什么静态方法不能隐藏实例方法?
静态方法的调用的是通过在编译器静态绑定的,而实例方法的调用是在运行时动态绑定的,2者的调用的方式不同,所以二者只能存在其一,否则会存在歧义!
为什么静态方法能隐藏静态方法?
因为调用方式一致,不会像上面造成歧义,虽然父类和子类都定义了同样的函数,但是编译器会根据对象的静态类型激活对应的静态方法的引用,造成了重写的假象,实则不是重写!
总结
总体流程就是:编译器将类编译成class文件,其中方法会根据静态类型从而将对应的方法引用写入class中,运行时,JVM会根据INVOKEVIRTUAL 所指向的方法引用在常量池找到该方法的偏移量,再根据this找到引用类型真实指向的对象,访问这个对象类型的方法表,根据偏移量找出存放目标方法引用的位置,取出这个引用,调用这个引用实际指向的方法,完成多态!
原文:https://blog.csdn.net/dawn_after_dark/article/details/74357049
java中静态属性和和静态方法的继承问题 以及多态的实质?
首先结论是:java中静态属性和和静态方法可以被继承,但是没有被重写(overwrite)而是被隐藏。
静态方法和属性是属于类的,调用的时候直接通过类名.方法名完成的,不需继承机制就可以调用如果子类里面定义了静态方法和属性,那么这时候父类的静态方法 或属性称之为“隐藏”,你如果想要调用父类的静态方法和属性,直接通过父类名.方法名或变量名完成,至于是否继承一说,子类是有继承静态方法和属性,但是 跟实例方法和属性不太一样,存在“隐藏”的这种情况。
多态之所以能够实现是依赖于 继承 接口和 重写 、重载(继承和重写最为关键)。有了继承和重写就可以 实现父类的引用可以指向不同子类的对象。重写的功能是:“重写”后子类的优先级要高于父类的优先级,但是“隐藏”是没有这个优先级之分的。
静态属性、静态方法和非静态的属性都可以被 继承 和 隐藏 而不能够被重写,因此不能实现多态,不能实现父类的引用可以指向不同子类的对象。 非静态的方法可以被继承和重写,因此可以实现多态。