欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Java-静态绑定和动态绑定

程序员文章站 2024-03-21 12:01:28
...

Java-静态绑定和动态绑定

绑定的概念

  1. 绑定指的是一个方法的调用与方法所在的类(方法主体)关联起来。
  2. 对java来说,绑定分为静态绑定和动态绑定;或者叫做前期绑定和后期绑定.

静态绑定

  1. 概念:在程序执行前方法已经被绑定(也就是说在编译过程中就已经知道这个方法到底是哪个类中的方法),此时由编译器或其它连接程序实现。
  2. 针对java简单的可以理解为程序编译期的绑定;这里特别说明一点,java当中的方法只有final,static,private和构造方法是前期绑定
    private: 不能被继承,则不能通过子类对象调用,而只能通过类本身的对象进行调用,所以可以说private方法和方法所属的类绑定;
    final: final方法虽然可以被继承,但是不能被重写(覆盖),虽然子类对象可以调用,但是调用的都是父类中的final方法(因此可以看出当类中的方法声明为final的时候,一是为了防止方法被覆盖,而是为了有效关闭java的动态绑定);
    static: static方法可以被子类继承,但是不能被子类重写(覆盖),但是可以被子类隐藏。(这里意思是说如果父类里有一个static方法,它的子类里如果没有对应的方法,那么当子类对象调用这个方法时就会使用父类中的方法。而如果子类中定义了相同的方法,则会调用子类的中定义的方法。唯一的不同就是,当子类对象上转型为父类对象时,不论子类中有没有定义这个静态方法,该对象都会使用父类中的静态方法。因此这里说静态方法可以被隐藏而不能被覆盖。这与子类隐藏父类中的成员变量是一样的。隐藏和覆盖的区别在于,子类对象转换成父类对象后,能够访问父类被隐藏的变量和方法,而不能访问父类被覆盖的方法)。
    构造方法: 也是不能被继承的。子类是通过super()来调用父类的无参构造方法,来完成对父类的初始化, 而我们使用从父类继承过来的方法是不用这样做的,因此不应该说子类继承了父类的构造方法,因此编译时也可以知道这个构造方法到底是属于哪个类。

动态绑定

  1. 概念:在运行时根据具体对象的类型进行绑定。若一种语言实现了动态绑定,同时必须提供一些机制,可在运行期间判断对象的类型,并分别调用适当的方法。也就是说,编译器此时依然不知道对象的类型,但方法调用机制能自己去调查,找到正确的方法主体。不同的语言对动态绑定的实现方法是有所区别的。但我们至少可以这样认为:它们都要在对象中安插某些特殊类型的信息。
  2. 在Java中:调用的方法依赖于隐式参数的实际类型,并且在运行时实现动态绑定。
  3. 动态绑定的过程:
    a. 编译器查看对象的声明类型和方法名;
    b. 编译器查看调用方法时提供的参数类型。例如a.f(“hello”),编译器将会挑选f(String),而不是f(int),由于存在类型转换(int转换为double),所以可能会更复杂。如果编译器没找到参数类型匹配的方法,或者发现有多个方法与之匹配,就会报告一个错误。
    c. 采用动态绑定调用方法的时候,一定调用与所引用对象的实际类型最合适的类的方法。如果a的实际类型是D,它是C类的子类,如果D定义了一个方法f(String),就直接调用它,否则将在D类的超类中寻找f(String),以此类推。
  4. 动态绑定特点: 每次调用方法都要进行搜索,时间开销太大,所以虚拟机预先为每个类创建一个方法表(method table),其中列出了所有方法的签名和实际调用的方法。这样在调用方法的时候,只需要查找这个表即可。

举例

  1. 代码
//父类
public class People {
   public String name="人类";

   public String getName(){
       return this.name;
   }
}

//子类
public class Student extends People{
     public String name="学生";
     public int age=15;
     public String getName(){
         return this.name;
     }
     public int getAge(){
         return this.age;
     }
}

//main方法
public class Test1 {
	public static void main(String[] args) {
		People people=new Student();
		System.err.println("静态绑定结果");
		System.out.println(people.name);
		
		People a =new Student();
		System.err.println("动态绑定结果");
		System.out.println(a.getName());
	}
}

  1. 结果
静态绑定结果:
人类
动态绑定结果:
学生