这里将自己学习java及其应用的一些笔记、积累分享一下,如果涉及到了文章、文字侵权,请联系我删除或调整。
一、编译期绑定和运行时绑定
1.1 概述
通常,我们所说的绑定,是指将一个方法调用同方法主体关联起来。在了解“编译期绑定”、“运行期绑定”前,我们还需要理解一个概念,即“编译时类型和运行时类型”。
- 编译时类型和运行时类型
Java的引用变量有两个类型,一个是编译时类型,一个是运行时类型。
编译时类型由声明该变量时使用的类型决定,运行时类型由实际赋给该变量的对象决定。如果编译时类型和运行时类型不一致,会出现所谓的多态。因为子类其实是一种特殊的父类,因此java允许把一个子类对象直接赋值给一个父类引用变量,无须任何类型转换,被称为向上转型,由系统自动完成。
引用变量在编译阶段只能调用其编译时类型所具有的方法,但运行时则执行它运行时类型所具有的方法,因此,编写Java代码时,引用变量只能调用声明该变量所用类里包含的方法。
与方法不同的是,对象的属性不具备多态性。通过引用变量来访问其包含的实例属性时,系统总是试图访问它编译时类所定义的属性,而不是它运行时所定义的属性。
1.2 编译期绑定、运行期绑定
-
编译期绑定
在程序执行之前进行绑定(由编译器和连接器完成),编译期绑定是面向过程程序设计语言中默认的绑定方式,例如,C语言只有一种方法调用。
-
运行期绑定
就是在程序运行时根据对象的类型进行绑定,也叫作动态绑定。
注意:Java中除了static和final方法(private方法本质也属于final方法,因为类中的private方法被隐式指定为final方法)之外,其他方法都是运行期绑定。这意味着通常不必判定是否该进行后期绑定,因为它是自动发生的。同时,也可以明确,将方法声明为final类型的一是为了防止方法被覆盖,二是为了有效的关闭java中的动态绑定。
1.3 练习:验证编译期绑定与运行期绑定
package 编译期和运行期绑定;
public class Test1 {
public static void main(String[] args) {
B b = new B();// 真实类型
A a = new B();// 向上转型至父类型
// 运行期,查找并绑定到子类方法
b.g();
a.g();// 二者都是调用子类B的g()
/* 静态成员在编译器绑定
* 编译器查找,编译器根据变量a的类型定义在这个类中
* 查找f方法并绑定
*/
b.f();// B b = ... 相当于B.f()
a.f();// A a = ... 相当于A.f()
/*静态方法更提倡通过类名调用,A.f()*/
System.out.println(b.v1);
System.out.println(a.v1);
}
}
class A {
int v1 = 1;
static void f() {
System.out.println("A.f()");
}
void g() {
System.out.println("A.g()");
}
}
class B extends A {
int v1 = 2;
static void f() {
System.out.println("B.f()");
}
void g() {
System.out.println("B.g()");
}
}