首先复习了一下jvm运行时数据区
记忆那幅图吧
- 线程共享部分 堆内存、方法区
- 线程独占部分 虚拟机栈、本地方法栈、程序计数器
1.类加载器
- bootstrap加载器 基础加载器 负责加载jre_home/jre/lib下的类
- extendClassLoader 负责加载jre_home/jre/lib/ext下的类
- appClassLoader 负责加载程序员自己编写的各个java程序的类,具体就是加载各个java程序的"java.class.path"属性 指定的路径下的类,或者运行java命令式附带的-cp参数指定的地址
2.查看类加载器
任何一个对象,都可以使用getClassLoader这个方法来查看到它对应的类加载器,extendClassLoader或者appClassLoader或者自定义的ClassLoader,如果是bootstrapLoader的话,那么输出的是null,因为bootstrapLoader是由c/C++实现的,它并不是一个java类
3.JVM如何知道类在何方
- String这种基础类的话,就在环境变量jre_home下的jre/lib
- jdk的其他类的话,就在环境变量jre_home下的jre/lib/ext
- 用户应用程序的话,去读取"java.class.path"这个属性,获取路径getProperty("java.class.path")
- ips命令
ips命令可以看到本机有什么java进程 - jcmd命令
jcmd pid VM.system_properties 可以看到某一个java线程的属性,其中就包括"java.class.path"这个属性
4.类不会重复加载
加载器实例+package+类名 可以唯一确定一个类,jvm不会对同一个类进行重复加载
验证方法:写一个while循环,不停地用类类加载机制加载类,期间对源码进行修改,发现修改并不生效
5.卸载类
什么时候类会被卸载
要满足2个条件
- 类的所有实例对象被GC
- 类的ClassLoader实例被GC
-verbose:class 参数,可以输出类加载和卸载日志
课后作业:如何在idea设置VM_Option 和-verbose
6.双亲委派模型
记住那幅图:appClassLoader、extClassLoader、bootStrapClassLoader由下至上排列,有一顺时针的箭头链接三个类加载器,向上是委派、向下是查询
核心:父加载器无法执行类加载任务时,才由自己完成