读《深入理解Java虚拟机》有感——第二部分:虚拟机类加载机制
1.关于“类加载器”和“生成的这个Class对象”: 1)类加载器 Bootstrap Loader(启动类加载器)、Extended Loader(标准扩展类加载器ExtClassLoader)、AppClass Loader(系统类加载器/应用程序类加载器AppClassLoader) 启动类加载器: 目的:加载java_home\lib目录下的字节码文件(如:rt.jar,含有java.lang.Object等基本类) 具体有哪些文件及加载顺序? 方式:加载System.getProperty("sun.boot.class.path")所指定的路径或jar,在使用Java运行程序时,也可以指定其搜索路径,例如:java -Djava.ext.dirs=d:\projects\testproj\classes HelloWorld 参考:http://www.cnblogs.com/ITtangtang/p/3978102.htm
标准扩展类加载器: 目的:加载java_home\lib\ext目录下的字节码文件
方式:加载System.getProperty("java.ext.dirs")所指定的路径或jar。在使用Java运行程序时,也可以指定其搜索路径,例如:java -Djava.ext.dirs=d:\projects\testproj\classes HelloWorld 结果:<实现类>sun.misc.Luncher@ExtClassLoader————————<继承关系>ClassLoader>URLClassLoader>AppClassLoader
应用程序类加载器: 方式:加载System.getProperty("java.class.path")所指定的路径或jar。在使用Java运行程序时,也可以加上-cp来覆盖原有的Classpath设置,例如: java -cp ./lavasoft/classesHelloWorld 结果:<实现类>sun.misc.Luncher@AppClassLoader————————<继承关系>ClassLoader>URLClassLoader>AppClassLoader
自定义类加载器: 方式:1)继承java.lang.ClassLoader并重写loadClass方法;2)继承java.lang.ClassLoader并重写findClass方法/**JDK1.2后推荐,原因见下方红色部分*/; 相关:
<一>java.lang.Object
1.getClass()
public final native Class<?> getClass(); //拿到运行时的Class对象【通过本地方法】
/**
*例子:class-Test>main>
* Class c =Person.getClass();
*/
/**
*实现:
*1)虚拟机启动
*2)虚拟机类加载——Test.class————加载阶段:方法区>外部接口>new java.lang.Class //Class<?>
*3)虚拟机类加载——java.lang.Object————【Test加载阶段】触发
*3)虚拟机类加载——Person.class ————【Test加载阶段】触发
*4)应用程序启动
*5)调用java.lang.ClassLoader>xxx1()、xxx2().....——返回运行时<Person>Class对象
*/
<二>java。lang.ClassLoader
1.loadClass(String name, boolean resolve) /**加载指定名称(包括包名)的二进制类型,同时指定是否解析*/
loadClass(String name)
protected Class<?> findClass(String name) throws ClassNotFoundException { //空方法
throw new ClassNotFoundException(name);
}
protected Class<?> loadClass(String name, boolean resolve) //拿到类加载器【通过本地方法】
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// First, check if the class has already been loaded
Class c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
if (parent != null) {
c = parent.loadClass(name, false);
} else {
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found
// from the non-null parent class loader
}
if (c == null) {
// If still not found, then invoke findClass in order
// to find the class.
long t1 = System.nanoTime();
c = findClass(name);
// this is the defining class loader; record the stats
sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
sun.misc.PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
}
<三>java.lang.Class
1.getClassLoader()
/**例子:class-Test>main>
* Object o =new Object();
* System.out.println(o.getClass().getClassLoader());
结果>java.lang.NullPointerException
*/
/**
*Returns the class loader for the class.
*This method will return null in such implementations if this class was loaded by the bootstrap class loader.
*如果没有指定classLoader就默认返回bootstrap classLoader(启动类加载器),因为这个bootstrap classLoader
*用户拿不到实例,所以返回null表示返回的是bootstrap classLoader
*/
native ClassLoader getClassLoader0(); //拿到类加载器【通过本地方法】
public ClassLoader getClassLoader() {
ClassLoader cl = getClassLoader0();
if (cl == null)
return null;
SecurityManager sm = System.getSecurityManager();
if (sm != null) {
ClassLoader.checkClassLoaderPermission(cl, Reflection.getCallerClass());
}
return cl;
} 2.指定类加载器加载类 Class.forName(name, initialize, loader)
2)与“生成的Class对象”关系 同一Class文件只有被同一类加载器加载,才能判断为相等————>相等的意义在于:1)不相等,会生成多个Class对象;相等,只会生成一个Class对象 2) 只有同一个Class对象,equals()、isAssignabaleFrom()、instanceof、isInstance()返回结果才相同;
3)双亲委派模型关系? 上一级持有下一级的一个引用,属于 has A关系 分工?(为什么叫双亲委派) 一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给上一层的类加载器去完成。只有当上级加载器在搜索范围找不到所需类,无法完成这个加载请求时,下级加载器才会尝试自己去加载 好处? Java类随着它的类加载器一起具备了一种优先级的层次关系
2.加载“类/接口”的策略: 非数组类/接口、数组类(组件类型为基本类型,如int[][]): 加载生成代表这个类的java.lang.Class对象后,将在类加载器的“类名称空间”上标识为“已加载”(因为前面已经讨论了,同一Class文件对应同一个类加载器才确定生成的是同一个Class对象)
连接阶段:
验证——文件格式、元数据、字节码
准备——在方法区为类变量分配内存并初始化
例子 编译时期 (类加载时)验证阶段—准备时期
static int i =20; 产生constantValue属性,但不会存入常量20 常规方式进行准备:初始化、分配内存
结果—————> 无 0
final static int i =20; 产生constantValue属性,并存20到constantValue 从对应constantValue取出来初始化
结果—————> 20 20
解析——将运行时常量池的符号引用替换为直接引用(指向内存目标的句柄),这一过程又叫“静态绑定/静态连接”
//例子(笔试常考题目)
public class Father {
private static int i =20;
static{
System.out.println("Father;(类构造器-before)"+i);
i =50;
System.out.println("Father;(类构造器-after)"+i);
}
public Father() {
System.out.println("Father:(实例构造器)"+ }
}
public class Son extends Father{
private static int i =20;
static{
System.out.println("Son;(类构造器-before)"+i);
i =50;
System.out.println("Son;(类构造器-after)"+i);
}
public Son() {
System.out.println("Son:(实例构造器)"+ }
}
public class Main {
public static void main(String[] args) {
new Son();
}
}
//输出结果:
/**
*Father;(类构造器-before)20
*Father;(类构造器-after)50
*Son;(类构造器-before)20
*Son;(类构造器-after)50
*Father:(实例构造器)50
*Son:(实例构造器)50
*/
读《深入理解Java虚拟机》有感——第二部分:虚拟机类加载机制的更多相关文章
- 深入理解Java内存模型中的虚拟机栈
深入理解Java内存模型中的虚拟机栈 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域,这些区域都会有各自的用途,以及创建和销毁的时间,有的区域会随着虚拟机进程的启 ...
- Java基础 -- 深入理解Java类型信息(Class对象)与反射机制
一 RTTI概念 认识Claa对象之前,先来了解一个概念,RTTI(Run-Time Type Identification)运行时类型识别,对于这个词一直是 C++ 中的概念,至于Java中出现RT ...
- (转)《深入理解java虚拟机》学习笔记6——类加载机制
Java虚拟机类加载过程是把Class类文件加载到内存,并对Class文件中的数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的java类型的过程. 在加载阶段,java虚拟机需要完成以下 ...
- Java程序员的必备知识-类加载机制详解
类加载器的概念 类加载器是一个用来加载类文件的类. Java源代码通过javac编译器编译成类文件.然后JVM来执行类文件中的字节码来执行程序.类加载器负责加载文件系统.网络或其他来源的类文件. JV ...
- JVM虚拟机内存溢出垃圾收集及类加载机制总结
1.Java内存区域与内存溢出异常 虚拟机栈:为虚拟机执行Java方法服务 本地方法栈:为虚拟机使用到的native方法服务. Java堆:是Java虚拟机所管理的内存中最大的一块,被所有线程共享的一 ...
- 深入理解Java虚拟机(第2版) 笔记目录
本篇为读深入理解Java虚拟机(第2版)一书的笔记目录. Java 运行期数据区 Java 垃圾回收算法 Java 内存分配策略 Java 类文件结构 Java 加载.链接.初始化 Java 类加载器
- 深入理解Java虚拟机(八)——类加载机制
是什么是类加载机制 Java虚拟机将class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这个过程就是类加载机制. 类的生命周期 一个类从加载到内存 ...
- 全面理解Java内存模型(JMM)及volatile关键字(转载)
关联文章: 深入理解Java类型信息(Class对象)与反射机制 深入理解Java枚举类型(enum) 深入理解Java注解类型(@Annotation) 深入理解Java类加载器(ClassLoad ...
- 全面理解Java内存模型(JMM)及volatile关键字(转)
原文地址:全面理解Java内存模型(JMM)及volatile关键字 关联文章: 深入理解Java类型信息(Class对象)与反射机制 深入理解Java枚举类型(enum) 深入理解Java注解类型( ...
随机推荐
- DoTween小结
using UnityEngine; using System.Collections; using DG.Tweening; public class GetStart : MonoBehaviou ...
- 不变(Invariant), 协变(Covarinat), 逆变(Contravariant) : 一个程序猿进化的故事
阿袁工作的第1天: 不变(Invariant), 协变(Covarinat), 逆变(Contravariant)的初次约 阿袁,早!开始工作吧. 阿袁在笔记上写下今天工作清单: 实现一个scala类 ...
- 六个创建模式之简单工厂模式(Simple Factory Pattern)
定义: 定义一个工厂类,它可以根据参数的不同生成对应的类的实例:被创建的类的实例通常有相同的父类.因为该工厂方法尝尝是静态的,所以又被称为静态工厂方法(Static Factory Method) 结 ...
- play framework学习笔记之 模板引擎
模板语法 ${client.name} ${client?.name} 不能确定client是否存在的时候? #{extends /} #{doLayout /}#{get} #{set} 比如 #{ ...
- IOS6学习笔记(一)
一.ARC 1.ARC环境下可以使用-(void)dealloc{};处理一些事情(比如移除KVO观察),但不要调用[super dealloc]; 2.ARC与非ARC混编要注意符合Cocoa命名约 ...
- webservice 的wsdl文件生成客户端java类
提供两个方法: 第一个: 发布webservice项目后, 地址栏地址 http://localhost:8888/lxitedu.webservice.cxf-ch2/services/userS ...
- 拖拽改变div的大小
拖拽改变div的大小 <!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type&qu ...
- C#加密算法总结
C#加密算法总结 MD5加密 /// <summary> /// MD5加密 /// </summary> /// <param name="strPwd&qu ...
- ABAP指针
1. 什么是ABAP指针:在ABAP里面,field symbol就相当于c语言的指针.如果你定义并且分配了相应的结构或者变量给它,其实它就指向这个结构或者变量的地址,如果修改了field symbo ...
- Atitit. Xss 漏洞的原理and应用xss木马
Atitit. Xss 漏洞的原理and应用xss木马 1. XSS漏洞1 2. XSS的用途2 2.1. 盗取cookie2 2.2. 刷新流量 刷分3 2.3. DOS 窃取隐私”.“假冒身份”. ...