本系列文章经补充和完善,已修订整理成书《Java编程的逻辑》,由机械工业出版社华章分社出版,于2018年1月上市热销,读者好评如潮!各大网店和书店有售,欢迎购买,京东自营连接:http://item.jd.com/12299018.htmlhtml
第15节咱们介绍了继承和多态的基本概念,而上节咱们进一步介绍了继承的一些细节,本节咱们经过一个例子,来介绍继承实现的基本原理。须要说明的是,本节主要从概念上来介绍原理,实际实现细节可能与此不一样。编程
例子
swift
这是基类代码:微信
public class Base { public static int s; private int a; static { System.out.println("基类静态代码块, s: "+s); s = 1; } { System.out.println("基类实例代码块, a: "+a); a = 1; } public Base(){ System.out.println("基类构造方法, a: "+a); a = 2; } protected void step(){ System.out.println("base s: " + s +", a: "+a); } public void action(){ System.out.println("start"); step(); System.out.println("end"); } }
Base包括一个静态变量s,一个实例变量a,一段静态初始化代码块,一段实例初始化代码块,一个构造方法,两个方法step和action。函数
这是子类代码:布局
public class Child extends Base { public static int s; private int a; static { System.out.println("子类静态代码块, s: "+s); s = 10; } { System.out.println("子类实例代码块, a: "+a); a = 10; } public Child(){ System.out.println("子类构造方法, a: "+a); a = 20; } protected void step(){ System.out.println("child s: " + s +", a: "+a); } }
Child继承了Base,也定义了和基类同名的静态变量s和实例变量a,静态初始化代码块,实例初始化代码块,构造方法,重写了方法step。post
这是使用的代码:优化
public static void main(String[] args) { System.out.println("---- new Child()"); Child c = new Child(); System.out.println("\n---- c.action()"); c.action(); Base b = c; System.out.println("\n---- b.action()"); b.action(); System.out.println("\n---- b.s: " + b.s); System.out.println("\n---- c.s: " + c.s); }
建立了Child类型的对象,赋值给了Child类型的引用变量c,经过c调用action方法,又赋值给了Base类型的引用变量b,经过b也调用了action,最后经过b和c访问静态变量s并输出。这是屏幕的输出结果:spa
---- new Child() 基类静态代码块, s: 0 子类静态代码块, s: 0 基类实例代码块, a: 0 基类构造方法, a: 1 子类实例代码块, a: 0 子类构造方法, a: 10 ---- c.action() start child s: 10, a: 20 end ---- b.action() start child s: 10, a: 20 end ---- b.s: 1 ---- c.s: 10
下面咱们来解释一下背后都发生了一些什么事情,从类的加载开始。code
类的加载
在Java中,所谓类的加载是指将类的相关信息加载到内存。在Java中,类是动态加载的,当第一次使用这个类的时候才会加载,加载一个类时,会查看其父类是否已加载,若是没有,则会加载其父类。
一个类的信息主要包括如下部分:
类初始化代码包括:
实例初始化代码包括:
类加载过程包括:
须要说明的是,关于类初始化代码,是先执行父类的,再执行子类的,不过,父类执行时,子类静态变量的值也是有的,是默认值。对于默认值,咱们以前说过,数字型变量都是0,boolean是false,char是'\u0000',引用型变量是null。
以前咱们说过,内存分为栈和堆,栈存放函数的局部变量,而堆存放动态分配的对象,还有一个内存区,存放类的信息,这个区在Java中称之为方法区。
加载后,对于每个类,在Java方法区就有了一份这个类的信息,以咱们的例子来讲,有三份类信息,分别是Child,Base,Object,内存示意图以下:
咱们用class_init()来表示类初始化代码,用instance_init()表示实例初始化代码,实例初始化代码包括了实例初始化代码块和构造方法。例子中只有一个构造方法,实际中可能有多个实例初始化方法。
本例中,类的加载大概就是在内存中造成了相似上面的布局,而后分别执行了Base和Child的类初始化代码。接下来,咱们看对象建立的过程。
建立对象
在类加载以后,new Child()就是建立Child对象,建立对象过程包括:
分配的内存包括本类和全部父类的实例变量,但不包括任何静态变量。实例初始化代码的执行从父类开始,先执行父类的,再执行子类的。但在任何类执行初始化代码以前,全部实例变量都已设置完默认值。
每一个对象除了保存类的实例变量以外,还保存着实际类信息的引用。
Child c = new Child();会将新建立的Child对象引用赋给变量c,而Base b = c;会让b也引用这个Child对象。建立和赋值后,内存布局大概以下图所示:
引用型变量c和b分配在栈中,它们指向相同的堆中的Child对象,Child对象存储着方法区中Child类型的地址,还有Base中的实例变量a和Child中的实例变量a。建立了对象,接下来,来看方法调用的过程。
方法调用
咱们先来看c.action();这句代码的执行过程是:
寻找要执行的实例方法的时候,是从对象的实际类型信息开始查找的,找不到的时候,再查找父类类型信息。
咱们来看b.action();,这句代码的输出和c.action是同样的,这称之为动态绑定,而动态绑定实现的机制,就是根据对象的实际类型查找要执行的方法,子类型中找不到的时候再查找父类。这里,由于b和c指向相同的对象,因此执行结果是同样的。
若是继承的层次比较深,要调用的方法位于比较上层的父类,则调用的效率是比较低的,由于每次调用都要进行不少次查找。大多数系统使用一种称为虚方法表的方法来优化调用的效率。
虚方法表
所谓虚方法表,就是在类加载的时候,为每一个类建立一个表,这个表包括该类的对象全部动态绑定的方法及其地址,包括父类的方法,但一个方法只有一条记录,子类重写了父类方法后只会保留子类的。
对于本例来讲,Child和Base的虚方法表以下所示:
对Child类型来讲,action方法指向Base中的代码,toString方法指向Object中的代码,而step()指向本类中的代码。
这个表在类加载的时候生成,当经过对象动态绑定方法的时候,只须要查找这个表就能够了,而不须要挨个查找每一个父类。
接下来,咱们看对变量的访问。
变量访问
对变量的访问是静态绑定的,不管是类变量仍是实例变量。代码中演示的是类变量:b.s和c.s,经过对象访问类变量,系统会转换为直接访问类变量Base.s和Child.s。
例子中的实例变量都是private的,不能直接访问,若是是public的,则b.a访问的是对象中Base类定义的实例变量a,而c.a访问的是对象中Child类定义的实例变量a。
小结
本节,咱们经过一个例子,介绍了类的加载、对象建立、方法调用以及变量访问的内部过程。如今,咱们应该对继承的实现有了一个比较清楚的理解。
以前咱们提到过,继承实际上是把双刃剑,为何这么说呢?让咱们下节来探讨。
----------------
未完待续,查看最新文章,敬请关注微信公众号“老马说编程”(扫描下方二维码),从入门到高级,深刻浅出,老马和你一块儿探索Java编程及计算机技术的本质。原创文章,保留全部版权。
-----------
更多相关原创文章