java学习之编译时类型(前期绑定)和运行时类型 (后期绑定、动态绑定、运行时绑定)

原创 2013年10月25日 11:49:31

编译时类型和运行时类型:  

  Java的引用变量有两个类型,一个是编译时类型,一个是运行时类型,编译时类型由声明该变量时使用的类型决定,运行时类型由实际赋给该变量的对象决定。如果编译时类型和运行时类型不一致,会出现所谓的多态。因为子类其实是一种特殊的父类,因此java允许把一个子类对象直接赋值给一个父类引用变量,无须任何类型转换,或者被称为向上转型,由系统自动完成。

    引用变量在编译阶段只能调用其编译时类型所具有的方法,但运行时则执行它运行时类型所具有的方法,因此,编写Java代码时,引用变量只能调用声明该变量所用类里包含的方法。与方法不同的是,对象的属性则不具备多态性。通过引用变量来访问其包含的实例属性时,系统总是试图访问它编译时类所定义的属性,而不是它运行时所定义的属性。

 —— 以上摘自《疯狂Java讲义》

前期绑定和后期绑定(动态绑定、运行时绑定):

绑定:将一个方法调用同方法主体关联起来叫做绑定

前期绑定:在程序执行之前进行绑定(如果有的话,由编译器和连接器完成),前期绑定是面向过程程序设计语言中默认的绑定方式,例如,C语言只有一种方法调用,那就是前期绑定。

后期绑定:就是在程序运行时根据对象的类型进行绑定,也叫作动态绑定或运行时绑定。

注意:Java中除了static和final方法(private方法属于final方法,因为类中的private方法被隐式指定为final方法)之外,其他方法都是后期绑定。这意味着通常不必判定是否该进行后期绑定,因为它是自动发生的。

下面举例说明:

1.子类方法覆盖父类方法( 子类重写父类中的方法,调用子类中的方法

class Father{
      public void method(){
          System.out.println("父类方法:"+this.getClass());
      }
 }
 public class Son extends Father{
	public void method(){
          System.out.println("子类方法:"+this.getClass());
      }
	public static void main(String[] args){
		Father instance = new Son();
		instance.method();
	}
}
运行结果:
子类方法:class Son

 2. 子类没有重写父类中的方法,所以到父类中寻找相应的方法
class Father{
      public void method(){
          System.out.println("父类方法:"+this.getClass());
      }
 }
 public class Son extends Father{
	public static void main(String[] args){
		Father instance = new Son();
		instance.method();
	}
}
运行结果: 父类方法: class Son

3.动态绑定只是针对对象的方法,对于属性无效。因为属性不能被重写

class Father{
     public String name = "Father'name";
      
 }

 public class Son extends Father{
	public String name = "Son'name";

	public static void main(String[] args){
		Father instance = new Son();
		System.out.println(instance.name);
	}
}
运行结果:
Father'name
这里还可以从另外一个方面来说明:如果将Father类的
 public String name = "Father'name";
修改为

 private String name = "Father'name";
那么编译器将报错:  错误: name可以在Father中访问private
		System.out.println(instance.name);
说明在
               System.out.println(instance.name);这行代码执行时,访问的是父类的 name属性,而该属性被声明为private,所以无法访问,因而报错!

下面在分析一个例子:
class A
{
int count = 20;
}

class B extends A
{
int count = 200;
}

public class Test 
{
public static void main(String[] args) 
{
A a = new A();
System.out.println(a.count);
B b = new B();
System.out.println(b.count);
A ab = b;   //向上转型
System.out.println(ab.count);
}
}

运行结果  :20
            200
            20  
结果分析:

 前两行的输出毫无疑问,问题在
        A ab = b;    
        System.out.println(ab.count);
的输出是20,而不是200;在这之间我们可以用
        System.out.println(ab == b);   
来进行简单的判断,结果输出为 true ,说明 ab 和 b 两个引用变量指向同一个实例,既然 ab 和 b 指向同一个实例,为什么输出的是20不是200呢?原因在于:
  1.对于 class A 和class B来说,class B是class A的子类,由于子类的变量并不会覆盖父类的变量,所以实际上在class B中是存在来两个count,在这分别记作 A.count 和B.count ;
        2.虽然在 class B中存在A.count 和B.count ,但是究竟输出那一个 count ,取决于该引用变量的声明时类型(本文开头红色文字部分已经说明),此处 声明时类型 是 class A,所以输出 20 即A.count ,同理若改为 B ab = b ;则输出 200 即 B.count ;


相关文章推荐

java运行时多态,后期绑定的初始化顺序

package polymorphism.rodent; import static net.mindview.util.Print.*; class Characteristic { p...

C++中的动态类型与动态绑定、虚函数、运行时多态的实现

C++中的动态类型与动态绑定、虚函数、运行时多态的实现,认真捉摸。

Java运行时绑定探讨

运行时绑定也叫动态绑定,它是一种调用对象方法的机制。Java调用对象方法时,一般采用运行时绑定机制。   1.Java的方法调用过程 编译器查看对象的声明类型和方法名(对象变量的声明类型...

Java运行时绑定探讨之向上转型

运行时绑定也叫动态绑定,它是一种调用对象方法的机制。Java调用对象方法时,一般采用运行时绑定机制。1.Java的方法调用过程编译器查看对象的声明类型和方法名(对象变量的声明类型)。通过声明类型找到方...

Flex 运行时绑定

运行时绑定
  • dongnhk
  • dongnhk
  • 2012年02月16日 21:04
  • 386

使用最新的AIR3 Beta2的运行时绑定功能

Adobe于昨日发布AIR3 Beta2及SDK,这个测试版本最新加入了Captive Runtime(运行时绑定)的功能。接下来,我第一时间跟大家分享一下如何使用这个功能。 首先,什么是运行时绑定...
  • wkyb608
  • wkyb608
  • 2011年09月27日 13:10
  • 390

FlashBuilder导出选项 导出具有运行时绑定的应用程序 使用共享运行时导出应用程序

这两天在搞ADT导出。 遇到了这样一个错误 the air sdk no longer supports packaging android applications for use th...

Java前期(静态)绑定和后期(动态)绑定

转自:http://www.cnblogs.com/jstarseven/articles/4631586.html 程序绑定的概念: 绑定指的是一个方法的调用与方法所在的类(方法主体...

objective-c学习笔记第六章《objective-c 多态,动态类型,动态绑定与java中的区别》

书接上文。 所谓多态,其实是oop概念里面一个比较重要的模型。多态所解决的就是在一个程序中调用不同类,这些个好多类中方法名相同的问题。比如说我在ClassA和ClassB中分别定义了两个print方...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:java学习之编译时类型(前期绑定)和运行时类型 (后期绑定、动态绑定、运行时绑定)
举报原因:
原因补充:

(最多只允许输入30个字)