认识java虚拟机基本结构

jvm主要组成部分及其作用

Java虚拟机的基本结构:

  • 类加载子系统:负责从文件系统或者网络中加载class信息,加载的类信息存放在方法区的内存空间中。方法区中除了类的信息外,还会存放运行时常量池信息
  • Java堆在虚拟机启动的时候建立的,他是Java程序最主要的内存工作区,几乎所有的对象实例都存放在java堆中。堆空间是线程共享的。
  • 直接内存是Java堆外的,直接向系统申请的内存区域。直接内存的访问速度会由于Java堆。直接内存不受最大堆大小限制,但是回收系统内存空间限制
  • 垃圾回收系统是java虚拟机的重要组成部分,垃圾回收器会对方法区,java堆和直接内存进行回收。
  • Java栈,每一个Java虚拟机线程都有一个私有的Java栈,一个线程的线程栈在线程创建的时候创建,Java栈中保存着帧信息,局部变量,方法参数,同时和Java方法的调用和返回密切相关。
  • 本地方法栈用户存储本地方法的调用。作为对虚拟机的重要拓展,Java虚拟机允许Java直接调用本地方法。
  • PC寄存器也是每个线程私有的空间,Java虚拟姐会为每个Java线程创建PC寄存器,在任意时刻,一个Java线程总是在执行一个方法,这个正在被执行的方法被称为当前方法,如果当前方法不是本地方法,PC寄存器就会指向当前正在被执行的指令。如果当前方法是本地方法,那么PC寄存器的值就是undefined.
  • 执行引擎是Java虚拟机最核心的组件之一,它扶着执行虚拟机的字节码。虚拟机为了提高执行效率,会使用即时编译技术将方法编译成字节码后在执行。

辨清java堆

Java堆根据垃圾回收机制的不同,Java堆可能拥有不用的结构,最常见的一种结构就是将java堆分为新生代和老年代,新生代存放新生对象或者年龄不大的对象,老年代则存放老年代对象。新生代有可能分为eden,s0,s1,其中是s0和s1也被称为from和to区域,他们是两块大小相等,可以互换角色的内存空间。

对大多数情况下,对象首先在eden区分配,在一次新生代回收后,如果对象还存活,着会进入s1或者s1,每经过一次新生代回收,他的年龄就会加1,当对象年龄到达设定的阈值后,就会被认为是老年代对象,从而进入老年代

对于一个普通的类来说:

类及方法实现存放在方法区,new出来的实例对象存放在堆区,局部变量引用存放在栈区

-Xmx设置系统最大可用堆空间

函数如果调用:出入java栈

Java栈式一块线程私有的空间,Java栈和线程执行密切相关。线程执行的基本行为是函数调用,每次函数调用的数据都是通过Java栈传递的。

Java栈与数据结构中的栈有着相同的含义,它是一块先进先出的数据结构,只支持出栈和入栈两种操作。在Java栈中保存的主要内容为栈帧,每次函数调用,都会有一个对应的栈帧被压入java栈,在每一次函数调用结束后,都会有一个对应的栈帧被弹出Java栈。

在一个栈帧中至少要包括局部变量表,操作数栈和帧数据区及部分

提示:由于每次函数调用都会生成对应的栈帧,从而占用一定的栈空间,因此如果栈空间不足,自然不能进行函数调用,当请求的栈深度大于最大可用栈深度时,系统就会抛出StackOverflowError栈溢出错误

-Xss用来设置最大栈可用大小

函数嵌套调用层次在很大程度上有栈的大小决定,栈越大,函数可支持的嵌套次数越多

栈帧详解

之前说了栈帧存储局部变量表,操作数栈和帧数据区。

局部变量表

局部变量表是栈帧的重要组成部分,它用于保存函数的参数和局部变量,局部变量表中的变量只在当前函数调用中有效(因为局部变量存在于方法对应的栈帧上,所以工作空间只有方法级别),当函数调用结束后,函数栈帧销毁,局部变量表就会销毁。

由于局部变量表在栈帧之中,因此,如果函数的参数和局部变量较多,会使局部变量表膨胀,从而每一次函数调用就会占用更多的栈空间,最终导致嵌套调用的次数变少。所以减少函数的参数和局部变量就可以减少栈空间的使用。

操作数栈

操作数栈也是栈帧中重要的内容之一,它主要用于保存计算过程的中间结果,同时作为计算过程中变量临时的储存空间。

操作数栈也是一个先进后出的数据结构,支支持入栈和出栈操作,许多Java字节码指令都需要通过操作数栈进行参数传递。

帧数据区

除了局部变量表和操作数栈,Java栈帧还需要一些数据来支持常量池解析,正常方法返回和异常处理,大部分Java字节码指令需要进行常量池访问,帧数据区中保存着访问常量池的指针,方便程序访问常量池。

此外,当函数返回或者出现异常的时,虚拟机必须恢复调用者函数的栈帧,并让调用者继续执行,对于异常处理,虚拟机必须有一个异常处理表,方便在发生异常的时候找到处理异常的代码,因此异常处理表也是帧数据区中重要的一部分。

所以帧数据区中存储访问常量池的指针和异常处理表(记录方法出现异常时,异常处理代码的位置)

栈上分配

栈上分配时Java虚拟机提供的一项优化技术,他的基本思想,对于那些线程私有的对象(这里只不可能被其他线程访问的对象),可以将他们打散分配在栈上,而不是分配在堆上,分配栈栈上的好处时可以在函数调用结束后自行销毁,而不需要垃圾回收的介入,从而提高系统的性能

逃逸分析

栈上分配的一个技术基础就是进行逃逸分析。逃逸分析的目的是判断对象的作用域是否有可能逃逸除函数体。

private static user u;

public static void alloc(){

	u  = new User();

 	u.id = 5;

	u.name = "Vstone"

}

对象User u 是类的成员变量,该字段有可能被任和线程访问,因此属于逃逸对象,而以下的代码片段显示了一个非逃逸的对象:

public static void alloc(){

	User u = new User();

	u.id = 5;

	u.name="Vstone";

}

在上述代码中,对象User以局部变量的形式存在,并且该对象没有被alloca()函数返回或者出现了任何形式的公开,因此它并未发生逃逸。所以者中情况下,虚拟机就有可能将User分配在栈上,而不是堆上。

如何启用栈上分配

server 在server模式下,才可以启用逃逸分析
-Xmx10m
-Xms10m
-XX:+DoEscapeAlalysis 启用逃逸分析
-XX:+PrintGC 打印GC日志
-XX:-UseTLAB 关闭TLAB
-XX:+EliminateAllocations 开启标量替换(默认打开),允许将对象打散分配在栈上,比如对象拥有id和name两个字段,那么这两个字段会被视为两个独立的局部变量进行分配

类去哪儿了:识别方法区

和Java堆一样,方法区是一块所有线程共享的内存区域,它用于保存系统的类信息,比如类的字段,方法,常量池等。方法区的大小决定了系统可以保存多少个类,如果定义了太多的类,导致方法区溢出,虚拟机同样会抛除内存溢出错误。

在JDK1.6,JDK1.7中,方法区可以理解为永久区(Perm)。永久区可以使用-XX:PermSize和-XX:MaxPermSize指定。默认情况下,-XX:MaxPermSize为64MB,一个大的永久区可以存储更多的类信息。但是如果系统使用了一些动态代理,那么有可能在运行是生成大量的类,这时就需要设置一个合理的永久区大小,确保不发生永久区内存溢出。

在JDK1.8,JDK1.9,JDK1.10中,永久区已经被彻底移除了。取而代之的是元数据区。元数据区大小可以使用参数-XX:MaxMetaSpaceSize指定(一个大的元数据区域可以使系统支持更多的类),这是一块堆外内存,与永久区不同,如果不指定大小,默认情况下,虚拟机会耗尽所有可用系统。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值