1. 方法
程序绑定的概念:
绑定指的是一个方法的调用与方法所在的类(方法主体)关联起来。对java来说,绑定分为静态绑定和动态绑定;或者叫做前期绑定和后期绑定.
静态绑定:
在程序执行前方法已经被绑定(也就是说在编译过程中就已经知道这个方法到底是哪个类中的方法),此时由编译器或其它连接程序实现。例如:C。
针对java简单的可以理解为程序编译期的绑定;这里特别说明一点,java当中的方法只有final,static,private和构造方法是前期绑定
动态绑定:
后期绑定:在运行时根据具体对象的类型进行绑定。
若一种语言实现了后期绑定,同时必须提供一些机制,可在运行期间判断对象的类型,并分别调用适当的方法。也就是说,编译器此时依然不知道对象的类型,但方法调用机制能自己去调查,找到正确的方法主体。不同的语言对后期绑定的实现方法是有所区别的。但我们至少可以这样认为:它们都要在对象中安插某些特殊类型的信息。
动态绑定的过程:
- 虚拟机提取对象的实际类型的方法表;
- 虚拟机搜索方法签名;
- 调用方法。
对于private的方法, 首先一点它不能被继承,既然不能被继承那么就没办法通过它子类的对象来调用,而只能通过这个类自身的对象来调用。因此就可以说private方法和定义这个方法的类绑定在了一起。
final方法 虽然可以被继承,但不能被重写(覆盖),虽然子类对象可以调用,但是调用的都是父类中所定义的那个 final 方法,(由此我们可以知道将方法声明为final类型,一是为了防止方法被覆盖,二是为了有效地关闭java中的动态绑定)。
构造方法也是不能被继承的(网上也有说子类无条件地继承父类的无参数构造函数作为自己的构造函数,不过个人认为这个说法不太恰当, 因为我们知道子类是通过 super() 来调用父类的无参构造方法,来完成对父类的初始化 , 只是有些时候不必显式的调用 super() 如果父类存在无参数的构造方法 , 而我们使用从父类继承过来的方法是不用这样做的,因此不应该说子类继承了父类的构造方法),因此编译时也可以知道这个构造方法到底是属于哪个类。
对于static方法,具体的原理我也说不太清。不过根据网上的资料和我自己做的实验可以得出结论: static 方法可以被子类继承,但是不能被子类重写 (覆盖),但是可以被子类隐藏。(这里意思是说如果父类里有一个static方法,它的子类里如果没有对应的方法,那么当子类对象调用这个方法时就会使用 父类中的方法。而如果子类中定义了相同的方法,则会调用子类的中定义的方法。 唯一的不同就是,当子类对象上转型为父类对象时,不论子类中有没有定义这个静 态方法,该对象都会使用父类中的静态方法。因此这里说静态方法可以被隐藏而不能被覆盖。 这与子类隐藏父类中的成员变量是一样的。隐藏和覆盖的区别在于,子类对象转换成父类对象后,能够访问父类被隐藏的变量和方法,而不能访问父类被覆盖的方法)
2.成员变量
与方法不同,在处理java类中的成员变量(实例变量和类变量)时,并不是采用运行时绑定,而是一般意义上的静态绑定。所以在向上转型的情况下,对象的方法可以找到子类,而对象的属性(成员变量)还是父类的属性(子类对父类成员变量的隐藏)。
java 因为什么对属性要采取静态的绑定方法。这是因为静态绑定是有很多的好处,它可以让我们在编译期就发现程序中的错误,而不是在运行期。这样就可以提高程序的运行效率!而对方法采取动态绑定是为了实现多态,多态是java的一大特色。多态也是面向对象的关键技术之一,所以java是以效率为代价来实现多态这是 很值得的。
3. 代码示例
父类:
public class Parents {
private int age;
public String name;
public Parents(int age, String name) {
this.age = age;
this.name = name;
}
public void test() {
System.out.println("parents test: name=" + name);
}
//子类不能被覆盖,但是可以继承
public final void testFinal(){
System.out.println("parents testFinal");
}
public static void testStatic(){
System.out.println("parents testStatic");
}
}
子类:
public class Childs extends Parents {
private int points;
private String name;
public Childs(int points, String name) {
super(10, "panpan");
this.name = name;
this.points = points;
}
public static void main(String[] args) {
Parents person1 = new Childs(100, "binbin");
// 此时报错,因为成员变量属于静态绑定,编译期间决定,person1属于父类引用变量,
// 所以会去父类中查找是否存在points成员变量,此时不存在报错
// System.out.println(person1.points);
// 打印panpan,因为是person1父类引用变量
System.out.println(person1.name);
Childs person2 = new Childs(100, "binbin");
// 打印binbin,因为是person2子类引用变量
System.out.println(person2.name);
// 以下结果都打印childs test: name=binbin,因为此时方法属于动态绑定,运行期间决定,所以根据实际对象类型来决定
// person1和peron2都是childs类型对象
person1.test();
person2.test();
// 以下两行均打印parents testFinal
person1.testFinal();
person2.testFinal();
// 打印parents testStatic,因为person1父类引用变量
person1.testStatic();// 此时如果Parents没定义testStatic静态方法,编译出错
// 打印childs testStatic,因为person2子类引用变量
person2.testStatic();// 此时如果Childs没定义testStatic静态方法,编译出错
}
public void test() {
System.out.println("childs test: name=" + name);
}
/*
* 编译错误,不能覆盖final方法 public final void testFinal(){
* System.out.println("childs testFinal"); }
*/
public static void testStatic() {
System.out.println("childs testStatic");
}
}