继续接着上次【http://www.cnblogs.com/webor2006/p/8447586.html】的类加载的话题继续往下学习,这次并非纯理论了,会涉及到代码的论证,赶紧开始。
类的加载、连接与初始化:
先用一张图来对之前讲过的完整过程复习一下:
接下来进入新知识的学习:
- Java程序对类的使用方式可以分为两种:
①、主动使用
②、被动使用
听着是不是有点懵逼,莫要急,接下来会举例说明的,先明确概念~
- 所有的JVM实现必须在每个类或接口被Java程序“首次主动使用”时才初始化他们。
其中标红的说明需满足两个条件:1、主动使用时;2、首次使用时。 - 主动使用(七种)
①、创建类的实例。
②、访问某个类或接口的静态变量,或者对该静态变量赋值。
③、调用类的静态方法。
说明:关于第二和第三点从Java字节码助记符【用javap命令就可以查看相关的信息,JVM提供的助记符特别的多,所以记住是不现实的,也不用记,对重要的了解既可】层面来说:访问类的静态变量是用getstatic来表示的,而对静态变量进行赋值是用putstatic来表示的,而调用类的静态方法是用invokestatic来表示的。
④、反射(如Class.forName("com.test.Test"))。
⑤、初始化一个类的子类。
如果有两个类,一个是Parent,一个是Child,如下:
对于Child的初始化也会对Parent进行初始化的。
⑥、JVM启动时被标明了启动类的类,其意思就是包含了main方法的类。
⑦、JDK1.7开始提供的动态语言支持:java.lang.invoke.MethodHandle实例的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic句柄对应的类没有初始化,则初始化。【了解既可】 - 除了以上七种情况,其他使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化。
下面用具体的例子来感受一下主动使用与被动使用,新建工程,先定义两个类,如下:
由于想研究class,所以定义的都是静态相关的东东,接着在子类中也定义一个静态代码块:
接着在main函数中使用一下:
那此时的结果是?
好奇怪,原因是什么呢?原因是由于“对于静态字段来说,只有直接定义了该字段的类才会被初始化”, 回到这个程序上来,因为str静态字段是在MyParent1中定义的,所以只有主动使用MyParent1,并不会主动使用MyChild1,也就是论证了上面的这句话:
接着将程序进行改造一下:
这时结果又会如何呢?
因为str2是定义在MyChild1当中的,所以主动使用它会被初始化,但是还要记得之前谈到的七种主动使用的方式当中,这种是属于:
也就是当一个类进行初始化时,要求其父类全部都已经初始化完毕了,所以父类初始化时就会执行静态代码块了,所以其输出结果就如上所示。
类的加载【进一步详述】:
- 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在内存中创建一个java.lang.Class对象(规范并未说明Class对象位于哪里,Oracle的HotSpot虚拟机将其放在了方法区中)用来封装类在方法区内的数据结构。
- 加载.class文件的方式:
①、从本地系统中直接加载。【最常见的方式】
②、通过网络下载.class文件。
③、从zip,jar等归档文件中加载.class文件。【这就是为什么三方的库通常都是以jar包的形式提供的原因】
④、从专有的数据中提取.class文件。【用的比较少,了解既可】
⑤、将Java源文件动态编译为.class文件。【动态代理就是最典型的例子】