前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >JVM --- 结构体系介绍

JVM --- 结构体系介绍

作者头像
贪挽懒月
发布2021-03-23 12:05:46
3200
发布2021-03-23 12:05:46
举报
文章被收录于专栏:JavaEEJavaEE

1. JVM的位置:

JVM位置

JVM是运行在操作系统之上的虚拟机,跟硬件没有直接交互。这也就体现了它跨平台的优越性。只要你这个操作系统能运行JVM,那么就可以运行java程序。

2. JVM体系结构:

下图为JVM的体系结构,灰色的表示线程私有,且该区域不存在垃圾回收,橙色的表示线程共享,且存在垃圾回收。

jvm体系结构

3. 类加载器ClassLoader:

负责加载class文件,class文件中文件开头有特定的标识,随便写个txt改成class,jvm是不能加载的。classLoader的作用就是将class文件加载到内存中,并将这些内容转换成方法区中的运行时数据结构。通过classLoader加载后,就会生成类的模板,之后new实例的时候,都是通过这个模板生成。类的模板就放在方法区。

(1). 类加载器分为以下几类:

  • 启动类加载器(Bootstrap Class Loader):C++语言写的,加载的是jre中的rt.jar。为什么我们可以直接用Object类,为什么能直接用ArrayList类?因为这些类都在rt.jar中,并且通过启动类加载器加载到了内存中。
  • 扩展类加载器(Extension Class Loader):java写的,加载的是jre/lib/ext/*.jar。因为java编写之初没写那么多功能,后来陆续加了一些功能,就被称为扩展类,扩展类加载器就是用来加载这些类的。
  • 应用程序类加载器(App Class Loader):加载用户自己写的类。比如我写了一个Test类,就是靠应用类加载器去加载。
  • 用户自定义类加载器:用户可以继承java.lang.ClassLoader类,实现自己的类加载器,定制加载方式。

(2). 类加载器之间的关系:

代码语言:javascript
复制
public class JvmDemo {
    
    public static void main(String[] args) {
        Object obj = new Object();
        System.out.println(obj.getClass().getClassLoader()); // 启动类加载器,打印出来是null
        
        JvmDemo jvmDemo = new JvmDemo();
        System.out.println(jvmDemo.getClass().getClassLoader()); // 应用类加载器
        System.out.println(jvmDemo.getClass().getClassLoader().getParent());
        System.out.println(jvmDemo.getClass().getClassLoader().getParent().getParent());
    }
}

打印的结果是:

运行结果

得出结论

  • 如果是启动类加载器,打印出来的是null;
  • 三个类加载器的关系为:应用类加载器是孙子,扩展类加载器是父亲,启动类加载器是爷爷,它们就是祖孙三辈的关系。

(3). 双亲委派机制:

别把这个想得有多难,可以借用一句话辅助理解:“我爸是李刚,有事找他”。应用类加载器要加载一个类的时候,它会先让它的父亲扩展类加载器去加载,同样的,扩展类加载器又会让它的父亲启动类加载器去加载。启动类加载器如果没加载到,就告诉扩展类加载器,扩展类加载器如果没加载到,再告诉应用类加载器,这个时候才轮到应用类加载器加载。也就是说,孙子要办事先找父亲,父亲又找爷爷,能不自己动手坚决不自动动手。

(4). 沙箱安全:

为什么要用双亲委派机制?看下面的例子:

代码语言:javascript
复制
package java.lang;

public class String {
        public static void main(String[] args) {
        System.out.println("are you ok?");
    }
}

我自己新建了一个java.lang包,写了一个String类。运行却发现报错了:

运行结果

我写的String中明明就有main方法,它却说没有,这也验证了双亲委派机制,说明加载的不是我自己写的String类,而是rt.jar中的String类。用双亲委派机制,在类路径相同的情况下,优先加载java自带的,这样就可以保证java自带的那些类的安全,保证它们不被污染,这就是沙箱安全。

4. 本地方法栈:

java有些方法底层是通过C语言实现的,这些方法栈java源码中都没有实现,并且有native关键字修饰。比如线程的start方法,new Thread().start(),其实调用的是start0()方法,这个方法就是一个native方法,调用的时候通过本地方法接口,从本地方法库中将其加载到本地方法栈中。

5. 程序计数器:

也叫PC寄存器,其实就是一个指针。比如在循环、递归的时候,程序要如何知道跳出这层循环后应该执行哪行代码,递归的执行是怎么个顺序?这些都靠程序计数器来完成,它可以让程序知道接下来该执行什么。

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档