java的动态绑定和静态绑定

背景

1.当子类和父类存在同一个方法,子类重写了父类的方法,程序在运行时调用的是父类的方法还是子类的重写方法呢(尤其是存在向上类型转换的情况)?

2.当一个类中存在方法名相同但参数不同(重载)的方法,程序在执行的时候该如何辨别区分使用哪个方法呢?

在java中存在绑定的机制解决以上疑问。

 

绑定

绑定:将一个方法的调用与方法所在的类(方法主体)关联起来。即决定调用哪个方法和变量。

在java中,绑定分为静态绑定和动态绑定。也叫作前期绑定和后期绑定。

 

静态绑定

在程序执行以前已经被绑定(即在编译过程中就已经知道这个方法到底是哪个类中的方法)。

java当中的方法只有final、static、private修饰的的方法和构造方法是静态绑定的。

private修饰的方法:private修饰的方法是不能被继承的,因此子类无法访问父类中private修饰的方法。所以只能通过父类对象来调用该方法体。因此可以说private方法和定义这个方法的类绑定在了一起。

final修饰的方法:可以被子类继承,但是不能被子类重写(覆盖),所以在子类中调用的实际是父类中定义的final方法。(使用final修饰方法的两个好处:(1)防止方法被覆盖;(2)关闭java中的动态绑定)。

static修饰的方法:可以被子类继承,但是不能被子类重写(覆盖),但是可以被子类隐藏。(这里意思是数哦如果父类里有一个static方法,它的子类里如果没有对应的方法,那么当子类对象调用这个方法时就会使用父类中的方法,而如果子类中定义了相同的方法,则会调用子类中定义的方法,唯一的不同就是:当子类对象向上类型转换为父类对象时,不论子类中有没有定义这个静态方法,该对象都会使用父类中的静态方法,因此这里说静态方法可以被隐藏而不能被覆盖。这与子类隐藏父类中的成员变量是一样的。隐藏和覆盖的区别在于,子类对象转换成父类对象后,能够访问父类被隐藏的变量和方法,而不能访问父类被覆盖的方法)。

构造方法:构造方法也是不能被继承的(因为子类是通过super方法调用父类的构造函数,或者是jvm自动调用父类的默认构造方法),因此编译时也可以知道这个构造方法方法到底是属于哪个类的。

因此,一个方法被继承,或者是被继承后不能被覆盖,那么这个方法就采用静态绑定

 

动态绑定

在运行时期根据具体对象的类型进行绑定。

若一种语言实现了后期绑定,同时必须提供一些机制,可在运行期间判断对象的类型,并分别调用适当的方法。也就是说,编译器此时依然不知道对象的类型,但方法调用机制能自己去调查,找到正确的方法主体。不同的语言对后期绑定的实现方法是有所区别的,但我们至少可以这样认为:它们都要在对象中安插某些特殊类型的信息。

动态绑定的过程:

1.虚拟机提取对象实际类型的方法表

2.虚拟机搜索方法签名

3.调用方法

 

java中重载的方法使用静态绑定,重写的方法使用动态绑定。

 

实验

package practice;

public class Bind{
	public static void main(String[] args) {
		Child c = new Child();
		Parent p = c;
		
		System.out.println(p.getPristr());
		System.out.println(c.pristr);
		c.print();
		p.print();
		
		c.print1();
		p.print1();
		
		c.print2();
		p.print2();
		
	}
}

class Parent{
	private String pristr = "parent private string";
	String pubstr = "public string";
	
	public String getPristr() {
		return pristr;
	}

	public void setPristr(String pristr) {
		this.pristr = pristr;
	}

	public Parent() {
		System.out.println("parent构造函数");
	}
	
    final public void print() {
    	System.out.println("parent的print");
    }
    
    public static void print1() {
    	System.out.println("parent的print1");
    }
    
    public void print2() {
    	System.out.println("parent的print2");
    }
    
}

class Child extends Parent{
	String pristr = "child private string";
	String pubstr = "public string";
	
	public Child() {
		System.out.println("child构造函数");
	}
	
	public static void print1() {
		System.out.println("child的print1");
    }
	
	public void print2() {
    	System.out.println("child的print2");
    }
}

结果

parent构造函数
child构造函数
parent private string
child private string
parent的print
parent的print
child的print1
parent的print1
child的print2
child的print2

 

你可能感兴趣的:(Java)