当前位置: 首页 > news >正文

上海门户网站建设/域名信息查询

上海门户网站建设,域名信息查询,电商学习网站,企业电子商城网站建设JVM系列之:内存与垃圾回收篇(一) ##本篇内容概述: 1、JVM结构 2、类加载子系统 3、运行时数据区之:PC寄存器、Java栈、本地方法栈一、JVM与JAVA体系结构 JAVA虚拟机与JAVA语言并没有必然的联系,它只是与特…

JVM系列之:内存与垃圾回收篇(一)

##本篇内容概述:
1、JVM结构
2、类加载子系统
3、运行时数据区之:PC寄存器、Java栈、本地方法栈

一、JVM与JAVA体系结构

JAVA虚拟机与JAVA语言并没有必然的联系,它只是与特定的二进制文件格式:Class文件格式关联,Class文件中包含了JAVA虚拟机指令集(字节码、Bytecodes)和符号表,还有一些其他辅助信息。

1、JVM整体结构

【说明:线程共享->方法区和堆; 各个线程独享:Java栈、本地方法栈和程序计数器】

在这里插入图片描述

【程序计数器Program Counter Register即:PC寄存器】

2、JAVA代码执行流程

在这里插入图片描述

3、JVM的架构模型

java编译器输入的指令流基本上基于以下两种指令集架构:

  • 基于栈的指令集架构(基于零地址指令,出栈入栈,指令集小,但指令多。不依赖于硬件可移植性好)
  • 基于寄存器的指令集架构(以一地址、二地址、三地址指令为主,依赖于硬件可移植性差,但执行效率更高)

总结:

由于跨平台性的设计,Java的指令都是根据栈来设计的。不同的平台CPU架构不同,所以不能设计为基于寄存器的。有点事跨平台,指令集小,编译器容易实现,
缺点是性能下降,实现同样的功能需要更多的指令。

4、JVM发展

虚拟机特点
SUN Classic VM只有解释器,没有JIT编译器
Exact VM编译器+解释器 混合工作模式
HotSpot(Oracle)引入了方法区Method Area。 热点代码探测+编译器解释器协同工作
JRockit(Oracle)专注于服务器端,JRockit不包含解释器实现,全部靠即时编译器编译后执行
J9(IBM)

二、类加载子系统

  • 类加载器子系统负责从文件系统或者网络中加载Class文件,class文件在文件开头有特定的文件标识;
  • ClassLoader只负责class文件的加载,至于它是否可以运行,则有执行引擎Execution Engine决定;
  • 加载的类信息存放于一块称谓方法区Method Area的内存空间。除了类的信息外,方法区中还会存放运行时常量池信息,可能还包括字符串字面量和数字常量(这部分常量信息是Class文件中常量池部分的内存映射)

在这里插入图片描述

类加载器子系统分为三个阶段:加载阶段、链接阶段、初始化阶段

class file加载到 JVM 中,被称为DNA元数据模板,放在方法区。

.class文件 => JVM => 最终成为元数据模板,此过程就要一个运输工具(类装载器Class Loader)。

类的加载过程:
在这里插入图片描述

1、加载Loading

##加载
1、通过一个类的全限定名获取定义此类的二进制字节流(从本地、网络、ZIP、动态代理、加密文件中等)2、将这个字节流锁代表的静态存储结构转化为方法区的运行时数据结构3、在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口

类加载器:

JVM支持两种类型的类加载器:引导类加载器(Bootstrap ClassLoader)自定义类加载器(User-Defined ClassLoader)==>派生于抽象类ClassLoader的类加载器都是自定义类加载器extention classloader和application classloader都是自定义类加载器Bootstrap ClassLoader是C/C++编写的,其他类加载器是java编写的他们是包含关系:其他自定义类加载器< Application classloader < extention classloader < bootstap classloaderJava的核心类库都是使用bootstrap classloader加载的
1.1、引导类加载器Bootstrap ClassLoader (启动类加载器)
  • 这个类加载使用C/C++语言实现,嵌套在JVM内部
  • 它用来加载java的核心类库,用于提供JVM自身需要的类
  • 加载extention classloader和application classloader,并指定为它们的父类加载器
  • 出于安全考虑,Bootstrap加载器只加载包名为java、javax、sun等开头的类
  • 它不继承自java.lang.ClassLoader,没有父加载器
1.2、扩展类加载器Extension ClassLoader
  • 使用java编写,由sun.misc.Launcher$ExtClassLoader实现
  • 派生于ClassLoader类
  • 父类加载器为Bootstrap ClassLoader
  • 从java.ext.dir系统属性所指定的目录或JDK的安装目录的jre/lib/ext子目录下加载类库,如果用户自定义创建的JAR包放在此目录下,也会由扩展类自动加载
1.3、应用程序类加载器Application ClassLoader(系统类加载器)
  • java语言编写,由sun.misc.Launcher$AppClassLoader实现
  • 派生于ClassLoader类
  • 父类加载器为Extension ClassLoader
  • 负责加载环境变量classpath或系统属性java.class.path指定路径下的类
  • 它是程序中默认的类加载器,一般java应用的类都是由它来完成加载
  • 通过ClassLoader.getSystemClassLoader()方法来获取该加载器

java的日常应用开发中,类的加载几乎都是由上述3种类加载器配合执行的

演示:


public class ClassLoaderTest {public static void main(String[] args) {//获取系统类加载器ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();System.out.println(systemClassLoader);//sun.misc.Launcher$AppClassLoader@18b4aac2//获取扩展类加载器ClassLoader extentionClassLoader = systemClassLoader.getParent();System.out.println(extentionClassLoader);//sun.misc.Launcher$ExtClassLoader@77459877//获取引导类加载器ClassLoader bootstrapClassLoader = extentionClassLoader.getParent();System.out.println(bootstrapClassLoader);//null//获取自定义的ClassLoaderTest类的类加载器//说明:对于用户自定义的类默认使用  系统类加载器ClassLoader classLoader = ClassLoaderTest.class.getClassLoader();System.out.println(classLoader);//sun.misc.Launcher$AppClassLoader@18b4aac2//获取String的类加载器//Java的核心类库都是使用bootstrap classloader加载的ClassLoader stringClassLoader = String.class.getClassLoader();System.out.println(stringClassLoader);//null}
}

自定义自己的类加载器,可以继承ClassLoader或者URLClassLoader类 重写findClass()方法即可

1.4、双亲委派机制
	JAVA虚拟机对class文件采用的是"按需加载"的方式,也就是说当需要使用该类的时候才会将他的class文件加载到内存中生成class对象。而且加载某个类的class文件时,java虚拟机采用的是"双亲委派模式",即把请求交由父类处理,也是一种任务委派模式

在这里插入图片描述

##工作原理:
1、如果一个类加载器收到了类加载的请求,它并不会自己先去加载,而是把这个请求委托给父类的加载器去执行;2、如果父类加载器还存在其父类加载器,则进一步向上委托,依次递归,请求最终将到达顶层的启动类加载器;3、如果父类加载器可以完成累的加载任务,就成功返回,若父类加载器无法完成此家在任务,子加载器才会
尝试自己去加载,这就是双亲委派模式##作用:
【只有这样如果我们自定义的java.lang.String类才不会被执行,也就不会覆盖系统自带的String了】
1、避免了类的重复加载
2、保护了程序安全,防止核心API被恶意篡改
1.5、沙箱安全机制
如上,我们自定义java.lang.String类,但是在加载自定义String类的时候会率先使用引导类加载器加载。
从而引导类加载器在加载的过程中会先加载JDK自带的文件(rt.jar包下的java.lang.String.class)。这样可以保证对java核心源代码的保护,这就是"沙箱安全机制"。
1.6、其他
1、在JVM中表示两个class对象是否为同一个类,有两个条件:a、类的完整类名必须一致,包括包名b、加载这个类的ClassLoader类加载器必须相同2、JVM必须知道一个类型是由启动类加载器加载的还是用户自定义类加载器加载的。
如果一个类型是由用户自定义类加载器加载的,那么JVM会将这个类加载器的一个引用作为类型信息的一部分保存在方法区中。
当解析一个类型到另一个类型的引用的时候,JVM需要保证这两个类型的类加载器是相同的。3、JAVA程序对类的使用分为:主动使用 和  被动使用
主动使用,分为7种情况
a、创建类的实例
b、访问某个类或接口的静态变量,或者对该静态变量赋值
c、调用类的静态方法
d、反射(如:Class.forName("com.lee.MyTest"))
e、初始化一个类的自雷
f、Java虚拟机启动的时候被标明为启动类的类
g、JDK7开始提供的动态语言支持除了以上情况,其他使用Java类的方式都被看做是队类的被动使用。主动使用会导致类的Initailization初始化(即下面的初始化),被动使用不会

2、链接Liking

##验证verify文件格式验证、元数据验证、字节码验证、符号引用验证(保证类加载的正确性)[.class文件一般以 CA FE BA BE开头]##准备Prepare为类"变量"分配内存并设置该类变量的默认初始值,即零值。(private static int a = 1;在prepare环节a = 0。在initial环节才被赋值为1)这里不包含用final修饰的static,因为final在编译的时候就会分配了,准备阶段会显示初始化。(private final static int a = 1;在prepare环节a = 1)这里不会为实例变量分配初始化,类变量会分配在方法区中,而实例变量是会随着对象一起分配到JAVA堆中。##解析Resolve将常量池内的符号引用转换为直接引用的过程 

3、初始化Initailization

##初始化
初始化阶段就是执行类构造器方法<clinit>()的过程【<clinit>()不同于类的构造器<init>()】此方法不需要定义,是javac编译器自动收集类中的所有 "类变量的赋值动作(static?)" 和 "静态代码块中的语句" 合并而来构造方法中指令按语句在源文件中出现的顺序执行若该类具有父类,JVM会保证子类<clinit>()执行前父类<clinit>()已经执行完毕虚拟机必须保证一个类的<clinit>()方法在多线程下被同步加锁(一个类只能被执行一次<clinit>())##解释
类变量与实例变量的区别:1)类变量属于类,可以共享,属于公共属性;实例变量属于某个对象个体;2)加上static 为类变量或者静态变量,否则为实例变量;

按顺序执行

public class ClassInitTest{static{number = 20;//System.out.println(number);//这里可以赋值,但不可以调用-》非法前向引用}private static int number=10;//在linking的prepare:number=0 --->  initail:按顺序执行 20 ->10public static void main(String[] args){System.out.println(number);//number的值为10}}

加载子类()执行前必须保证父类()的执行

public class ClinitTest1{static class Father{public static int A = 1;static{A=2;}}static class Son extends Father{public static int B = A;}public static void main(String[] args){//先加载Father类,prepare:A=0 initial A=1 然后A=2//再加载Son类,prepare:B=0 initail B=A=2System.out.println(Son.B);//2}
}

三、运行时数据区

1、运行时数据区概述及线程

在这里插入图片描述

Java虚拟机定义了若干种程序运行期间会使用到的运行时数据区,其中有些会随着虚拟机的启动二创建,随着虚拟机的退出而销毁。另外一些则是与线程一一对应,这些与线程对应的数据区域会随着线程的开始和结束而创建和销毁。##上图:
Method Area方法区 +  Heap Area堆空间 是共享的
Stack Area + PC Registers +  Native Method stack是各个线程独享的1)、每个线程:独立使用程序计数器、栈、本地栈。
2)、线程间共享:堆、堆外内存(永久代或元空间、代码缓存)Method Area方法区 在 JDK8 之后  成了 元空间,使用的是本地内存

关于线程间共享的说明:

Class Runtime  (java.lang.Runtime)每个JVM只有一个Runtime实例。即为运行时环境(Run data Area 运行时数据区)
[所以Runtime是单例的]

线程:

线程(Thread)是一个程序里的运行单元。JVM允许一个应用程序有多个线程并行执行。在Hotspot JVM里,每个线程都与操作系统的本地线程直接映射。
(当一个Java线程准备好执行以后,此时操作系统的本地线程也同时创建。Java线程执行终止后,本地线程也会被回收)操作系统负责所有线程的安排调度到任何一个可用的CPU上,一旦本地线程初始化成功,他就会调用Java线程的run()方法##Hotspot JVM里主要有如下几种线程:·虚 拟 机 线 程·周 期 任 务 线 程·GC 线 程·编 译 线 程·信 号 调 度 线 程

2、PC寄存器

没有GC垃圾回收,不会报OOM(OutOfMemoryError)

##PC寄存器 :Program Counter Register 【类似一个游标】
(JVM中的PC寄存器是对屋里PC寄存器的一种抽象模拟,也可以成为程序的钩子,程序的行号指示器)##作用:
PC寄存器用来存储指向下一条指令的地址,也就是即将要执行的指令代码。
由执行引擎读取下一条指令##特点:
·它是一块很小的内存空间,也是运行速度最快的存储区域·JVM规范中:每个线程都有它自己的PC寄存器,是线程私有的,生命周期与线程保持一致·任何时间一个线程都只有一个方法在执行,也就是所谓的当前方法。PC寄存器会存储当前线程
正在执行的Java方法的JVM指令地址,或者,如果是在执行native方法,则是未指定值[undefined]下面的 栈帧 也就是 方法,当前栈帧就是程序中的当前Java方法

在这里插入图片描述

·PC寄存器是程序控制流的指示器,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。·字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令。·它是唯一一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的区域
javap -v PCRegister.class下方右侧 0 2 3···即 指令地址 或 偏移地址

在这里插入图片描述

常见问题:

一、##使用PC寄存器存储字节码指令地址有什么用?##为什么使用PC寄存器记录当前线程的执行地址?因为CPU需要不停的快速切换各个线程,切换回来后,就得以知道接着从哪开始继续执行了。JVM的字节码解释器就需要通过改变PC寄存器的值来明确下一条应该执行什么样的字节码指令。二、##PC寄存器为什么被设定为线程私有的?所谓的多线程在特定的时间段内只会执行其中的某一个线程,CPU会不停地做快速切换而已。
这样必然导致线程的经常中断或恢复。为了能够准确地记录各个线程正在执行的当前字节码指令地址,最好的办法自然是为每个线程都
分配一个PC寄存器,这样一来各个线程之间变可以进行独立计算,从而不会出现相互干扰的情况。

拓展:CPU时间片:

CPU时间片即CPU分配给各个程序的时间,每个线程被分配一个时间段,称作它为时间片。宏观上:我们可以同时打开多个应用程序,每个程序并行不悖,同时运行。
微观上:由于只有一个CPU,一次只能处理程序要求的一部分,如何处理公平,一种方法就是引入时间片,每个程序轮流执行
并行: 多个任务 被多个CPU同时执行,相对的就是  "串行"并发:多个任务 被一个CPU快速切换的交替执行

3、虚拟机栈(Java栈)

3.1、概述

没有GC垃圾回收问题,但会存在OOM(OutOfMemoryError)的问题

一、##出现背景:前面在JVM架构模型中讲到,指令流基本上基于两种指令集架构:基于栈的指令集架构、基于寄存器的指令集架构。由于跨平台性的设计Java的指令都是根据栈来设计的。(不同平台的CPU架构不同,所以不能使用基于寄存器的)。优点是:跨平台,指令集小,编译器容易实现。
缺点是:性能下降,实现同样的功能需要更多的指令(指令集小但指令多)二、##内存中栈与堆的区别:栈是运行时的单位,堆是存储的单位即:栈解决程序的运行问题,即程序如何执行或者说如何处理数据。
堆解决的是数据存储的问题,即数据怎么放,放哪儿。三、##Java虚拟机栈是什么?Java虚拟机栈,早期也叫Java栈。
每个线程在创建时都会创建一个虚拟机栈,其内部保存一个个的栈帧(Stack Frame),每一个栈帧对应着一个的Java方法调用。
一个个方法的调用对应着一个个栈帧的入栈和出栈操作。Java栈是线程私有的。其生命周期和线程一致。主管Java程序的运行,它保存 方法的 "局部变量(8种基本数据类型、对象的引用地址)"、 "部分结果",并参与方法的调用和返回。 四、##栈的有点:栈是一种快速有效的分配存储方式,访问速度仅次于程序计数器。JVM直接对Java栈的操作只有两个:
·方法的执行,伴随着进栈(入栈、压栈)
·执行结束后的 出栈 工作对于栈来说不存在GC垃圾回收问题

在这里插入图片描述

五、##栈中可能出现的异常:Java虚拟机规范允许Java栈的大小是 "动态的" 或者 "固定不变"的。 ·如果采用固定大小的Java虚拟机栈,那每一个线程的Java虚拟机栈容量可以在线程创建的时候独立选定。
如果线程请求分配的栈内容超过Java虚拟机允许的最大容量,Java虚拟机将会抛出一个StackOverflowError 异常【递归的时候可能会出现】·如果Java虚拟机栈可以动态扩展,并且在尝试扩展的时候无法申请到足够的内存,或者在创建新的线程时没有足够的内存去创建对应的虚拟机栈,那么Java虚拟机将会抛出一个OutOfMemoryError异常六、##设置栈的大小
【java -Xss256k 设置stack大小】
【java -Xmx512m -Xmx512m 其中-Xmx设置堆最大值 -Xms设置堆初始值。】
3.2、栈的存储结构和运行原理
一、##栈的存储单位每个线程都有自己的栈,栈中的数据都是以栈帧Stack Frame的格式存在。在这个线程上正在执行的每一个方法都各自对应一个栈帧Stack Frame。栈帧是一个内存区块,是一个数据集,维系着方法执行过程中的各种数据信息。二、##栈运行原理JVM直接对Java栈的操作只有两个,就是对栈帧的“压栈”和“出栈”,遵循“先进后出”、“后进先出”的原则。在一条活动线程中,一个时间点上,只会有一个活动的栈帧,即只有当前正在执行的方法的栈帧
(栈顶栈帧)是有效的,这个栈帧被称谓当前栈帧Current Frame,与当前栈帧相对应的方法
就是当前方法Current Method,定义这个方法的类就是当前类Current Class.执行引擎运行的所有字节码指令只针对当前栈帧进行操作。如果在该方法中调用了其他方法,对应的新的栈帧就会被创建出来,放在栈的顶端,称谓新的当前栈帧。不同线程中锁包含的栈帧是不允许存在相互引用的,即不可能在一个栈帧中引用另外一个线程的栈帧。如果当前方法调用了其他方法,方法返回之际,当前栈帧会传回此方法的执行结果给前一个栈
帧,接着,虚拟机会丢弃当前栈帧,使得前一个栈帧重新成为当前栈帧。Java方法有两种返回函数的方式:一种是正常的函数返回,使用return指令;另一种是抛异常。
不管使用哪种方式,都会导致栈帧被弹出。
3.3、栈帧的内部结构
一、##栈帧的内部结构每个栈帧中存储着:
·局部变量  Local Variables
·操作数栈  Operand Stack  (或表达式)
·动态链接  Dynamic Linking (或指向运行时常量池的方法引用)
·方法返回地址 Return Address (或方法正常退出 或者异常退出的定义)
·一些附加信息

在这里插入图片描述

1>、局部变量表Local Variables

·局部变量表也被称谓 局部变量数组  或  本地变量表·定义为一个数字数组,主要用于存储方法参数和定义在方法体内的局部变量,
这些数据类型包括8种基本数据类型、对象引用(refrence),以及returnAddress类型·由于局部变量表是建立在线程的栈上,是线程的私有数据,因此不存在数据的安全问题·局部变量表所需的容量大小是在编译器确定下来的,并保存在方法的Code属性的maximum local variable是数据项中。
在方法运行期间是不会改变局部变量表的大小的。

在这里插入图片描述

【上图 LineNumberTable表示 字节码的行号 和 java代码行号对应的关系】·方法嵌套调用的次数由栈的大小决定(如递归)。·局部变量表中的变量只在当前方法调用中有效。方法执行时,虚拟机通过使用局部变量表完成参数值
到参数变量列表的传递过程。当方法调用结束后,随着方法栈帧的销毁,局部变量表也会随之销毁。【局部变量表是主要影响栈帧大小的】##上图中LocalVariableTable中Slot的理解·参数值的存放总是在局部变量数组的index0开始,到数组长度的-1的索引结束。·局部变量表,最基本的存储单元是Slot 变量槽·局部变量表中存放编译期可知的8中基本数据类型,引用类型,returnAddress类型的变量·局部变量表里,32位以内的类型只占用一个slot(包括returnAddress类型)
64位的类型(long和double)占用两个slot
> byte\short\char在存储前被转换为int
> boolean也被转换为int 0表示false,非0表示true
> long和double则占据两个slotbyte\short\char\float\int 占1个slot
long\double占2个slot·JVM回味局部变量表中每个slot都分配一个访问索引,通过这个索引即可成功访问到局部变量表中指定的局部变量值。·当一个实例方法被调用的时候,他的方法参数和方法体内部定义的局部变量将会按照顺序被赋值到局部变量表中的每一个slot上。·如果需要访问局部变量表中的一个64bit的局部变量值时,只需要使用前一个索引即可。(long和double)·如果当前栈帧是由构造方法或者实例方法创建的,那么该对象引用this将会存放在index为0的slot处,其余的参数按照参数表顺序继续排列。
(因此被static修饰的方法中是不能够使用this的)·栈帧中的局部变量表中的槽位slot是可以重复利用的,如果一个局部变量过了其作用域,那么在其作用域之后
声明的新的局部变量就很有可能会服用过期局部变量的槽位,从而达到节省资源的目的。
例如:
public void test{int a = 0;{int b = 0;b = a + 1;}int c = a +1;
}
其中b的作用域在其{}内,变量c是使用之前已经销毁的变量b占据的slot的位置。

在这里插入图片描述

变量 f 的位置被重复利用了

##拓展变量的分类:按照数据类型分:①基本数据类型 ②引用数据类型按照在类中的声明位置分:①成员变量:类变量(或叫静态变量,被static修饰)实例变量②局部变量	
##类变量linking的prepare阶段:给类变量默认赋值-->initial阶段:给类变量显式赋值即静态代码块赋值##实例变量随着对象的创建,会在堆空间中分配实例变量空间,并进行默认赋值
##局部变量在使用前,必须要进行显式赋值!否则,编译不通过
##补充说明:·在栈帧中,与性能调优关系最为模切的部分就是 局部变量表。
在方法执行时,虚拟机使用局部变量表完成方法的传递。·局部变量表中的变量也是重要的垃圾回收根节点,只要被局部变量表中引用数据类型直接或间接引用的对象都不会被回收。(局部变量表中 存储:基本数据类型 和 引用数据类型,引用数据类型引用的是堆空间中的对象,所以影响了堆空间的GC垃圾回收。如果局部变量表中的引用数据类型不存在了,那么堆空间中的数据可能就会被回收)

2>、操作数栈Operand Stack

##操作数栈:·后进先出,也被称之为表达式栈·操作数栈,主要是用于保存计算过程中的中间结果,同时作为计算过程中变量临时的存储空间·操作数栈,在方法执行的过程中,根据字节码指令,往栈中写入数据或提取数据,即入栈PUSH/出栈POP
(有些字节码指令是将值PUSH进操作栈,有些字节码指令是将操作数POP出栈,使用后再把结果PUSH进栈)
(比如执行赋值、交换、求和等操作)

在这里插入图片描述

·操作数栈,主要是用于保存计算过程中的中间结果,同时作为计算过程中变量临时的存储空间。·操作数栈就是JVM执行引擎的一个工作区,当一个方法开始执行的时候,一个新的栈帧也会随
之被创建出来,这个方法的操作数栈是空的。·每一个操作数栈都会拥有一个明确的栈深度用于存储数值,其所需的最大深度在编译期就定义
好了。保存在方法的code属性中,为max_stack的值。·栈中的任何一个元素都是可以任意的Java数据类型:>32bit的类型占用一个栈单位深度>64bit的类型占用两个栈单位深度·操作数栈并非采用访问索引的方式来进行数据访问的,而是只能通过标准的入栈和川站操作来完成一次数据访问·如果被调用的方法带返回值的话,其返回值将会被压入当前栈帧的操作数栈中。并更新PC寄存器中下一条需要执行的字节码指令。
(当当前方法用到上一条方法的返回值时会从操作数栈中load)·操作数栈中元素的数据类型必须与字节码指令的序列严格匹配,这由编译器在编译器期间进行
验证,同时在类家在过程中的类检验阶段的数据流分析阶段要再次验证。·另外,我们说Java虚拟机的解释引擎就是基于栈的执行引擎,其中的栈值得就是操作数栈。
(也就是说 JVM的执行引擎和操作数栈 配合操作的)

栈顶缓存技术Top-Of-Stack Cashing

·由于操作数是存储在内存中的,因此频繁第执行内存读/写操作必然会影响执行速度。为了解决这个问题,HotSpot JVM的设计者提出了栈顶缓存技术。即将栈顶元素全部缓存在物理CPU的寄存器中,以此降低对内存的读/写次数,提升执行殷勤的执行效率。

3>、动态链接Dynamic Linking

动态链接:指向运行时常量池的方法引用。
(运行时常量池  在  "方法区" 中)·每一个栈帧内部都包含一个执行运行时常量池中该栈帧所属方法的引用。
包含这个引用的目的就是为了支持当前方法的代码能够实现动态链接:invokedynamic指令·在Java源文件被编译到字节码文件中时,所有的变量和方法引用都作为符号引用保存在class文件的常量池里。
(比如描述一个方法调用了另外的其他方法时,就是通过常量池中指向方法的符号引用来表示的)·动态链接的作用就是为了将这些符号引用转换为调用方法的直接引用

在这里插入图片描述

·符号引用:以一组符号来描述所引用的目标 符号可以是任何形式的字面量,只要使用时能够无歧义的定位到目标即可
如:
public static void main(java.lang.String[]);....Code:stack=2, locals=3, args_size=1....11: invokevirtual #4      // Method methods1:()I·直接引用:直接引用可以是直接指向目标的指针、相对偏移量、一个能间接定位到目标的句柄
如:
Constant pool:....#4 = Methodref  #2.#39  // com/lee/jvm/rundataarea/LocalVariablesTest.methods1:()I##JVM指令拓展:
普通调用指令:
invokestatic:调用静态方法
invokespecial:调用<init>方法、私有及父类方法
invokevirtual:调用所有虚方法
invokeinterface:调用接口方法
动态调用指令:
invokedynamic:动态解析出需要调用的方法,然后执行

4>、方法返回地址 Return Address

·方法返回地址:存放调用该方法的PC寄存器的值·一个方法的结束,有两种方式:
>正常执行完成
>出现未处理异常,非正常退出·无论哪种方式退出,在方法退出后都返回到该方法被调用的位置。
方法正常退出是,调用者的PC计数器的值作为返回地址,即调用该方法的指令的下一条指令的地址。
而通过异常退出的,返回地址是要通过异常表来确定,栈帧中一般不会保存这部分信息。
1、执行引擎遇到任意一个方法返回的字节码指令return,会有返回值传递给上层的方法调用这,简称正常完成出口:
>一个方法在正常调用完成之后究竟需要使用哪一个返回指令还需要根据方法返回值的实际数据类型而定。
>在字节码指令中,返回指令包含为ireturn(当返回值时boolean\byte\chat\short\int类型时使用)、lreturn(long)、freturn(float)、dreturn(double)以及areturn
另外还有一个return指令提供声明为void的方法、实力初始化方法、类和接口的初始化方法使用。2、在方法执行的过程中遇到异常Exception,并且这个异常没有在方法内处理,也就是只要在本方法的异常表中没有搜索到匹配的异常处理器,就会导致方法退出。简称异常完成出口方法执行过程中抛出异常时的异常处理,存储在一个异常处理表,方便在发生异常的时候找到处理异常的代码。##本质上,方法的退出就是当前栈帧出栈的过程。
此时需要恢复上层方法的局部变量表、操作数栈、将返回值亚茹调用者栈帧的操作数栈、设置PC寄存器值等,让调用者方法继续执行下去。正常完成出口和异常完成出口的区别在于:通过异常完成的出口退出的不会给他的上层调用者产生任何的返回值。

5>、一些附加信息

栈帧中还允许携带与Java虚拟机实现相关的一些附加信息。如:对程序调试提供支持的信息

6>、关于Java栈的思考

一、##举例栈溢出的情况?如果设置了栈的大小,栈空间固定,虚拟机栈增加栈帧溢出会出现StackOverflowError
如果没有设置了栈的大小,虚拟机栈增加栈帧内存溢出或者增加新的线程时会出现OutOfMemeryError二、##调整栈的大小,就能保证不出现溢出吗?不能保证,只能保证栈溢出的阈值增大而已。三、##分配的栈内存越大越好吗?不是,内存空间固定,某个栈空间变大了,挤占了线程数或者其他结构的内存空间四、##垃圾回收是否会涉及到虚拟机栈?不会,栈的生命周期是随 线程 的创建和销毁的,所以不存在GC垃圾回收,如果溢出直接报错五、##方法中定义的局部变量是否线程安全?不一定,举例说明
public static void method() {int x = 0;for(int i = 1;i<=10;i++) {x *= i;}System.out.println(x);
}
上面线程安全的,当多个线程同时执行此方法时,每个线程都会产生一个自己的变量x,每个线程之间互不干扰,不会对其他线程的变量x有影响。public static void method(StringBuilder sb) {sb.append(1);sb.append(2);System.out.println(sb.toString());
}  
上面线程不安全,在这里sb对象是作为参数传入的,这意味着它并不是线程私有的,多个线程都可以同时访问它。所以不是线程安全的。                           public static StringBuilder method() {StringBuilder sb = new StringBuilder();sb.append(1);sb.append(2);return sb;
}
上面线程不安全,虽然sb对象是在方法内生成的对象。但是sb作为一个返回变量返回,其他线程可以去拿取它,对它进行并发的操作。                            public static String method() {StringBuilder sb = new StringBuilder();sb.append(1);sb.append(2);return sb.toString();
}     
上面线程安全,sb消亡了,sb.toString()相当于我们又new 了一个string,这个new出来的string可能不安全,但是方法内的sb是安全的。

4、本地方法栈

在写本地方法栈前 先 了解一下本地方法库本地方法接口

在这里插入图片描述

##本地方法本地方法就是,一个Native Method就是一个Java调用一个本地非Java代码的接口,比如是C或者C++。(即一个Java方法其方法内部是由非Java语言实现的)
如:public final native Class<?> getClass();
如:Thread 中 private native void start0();标识符native可以与所有其他的java标识符连用,但是abstract除外

本地方法栈:

·Java虚拟机栈用于管理Java方法的调用。而本地房发展用于管理本地方法的调用。·本地方法栈,也是线程私有的·允许被是线程固定或者是可动态扩展的内存大小(在内存溢出方面和虚拟机栈是相同的)·本地方法时C语言实现的·具体做法是 虚拟机栈 中登记native方法,在 执行引擎执行时加载本地方法库。

相关文章:

16-01、JVM系列之:内存与垃圾回收篇(一)

JVM系列之&#xff1a;内存与垃圾回收篇&#xff08;一&#xff09; ##本篇内容概述&#xff1a; 1、JVM结构 2、类加载子系统 3、运行时数据区之&#xff1a;PC寄存器、Java栈、本地方法栈一、JVM与JAVA体系结构 JAVA虚拟机与JAVA语言并没有必然的联系&#xff0c;它只是与特…...

聊聊系统的弹力设计-服务器性能指标篇(一)

一、什么是弹性机制 弹性&#xff0c;大家可以轻易的联想到橡胶&#xff0c;可伸缩性是弹性机制的一个很重要的特点&#xff0c;但是实际上弹性不等同于可伸缩性 弹性&#xff08;Elasticity&#xff09; 通常指的是系统能够自动适应负载的变化&#xff0c;即自动扩展和收缩资…...

MQ:kafka-消费者的三种语义

文章目录 前言(一) 创建topic(二) 生产者&#xff08;三&#xff09;消费者1. At-most-once Kafka Consumer2. At-least-once kafka consumer3. 使用subscribe实现Exactly-once4. 使用assign实现Exactly-once 前言 本文主要是以kafka 09的client为例子&#xff0c;详解kafka c…...

中国1km分辨率SSP119情景(SSP119、SSP245 SSP585),模式逐月降水量数据集(2021-2100)

目录 简介 摘要 代码 引用 网址推荐 知识星球 机器学习 干旱监测平台 中国1km分辨率SSP119情景EC-Earth3模式逐月降水量数据集(2021-2100) 简介 该数据集为中国多情景多模式逐月降水量数据&#xff0c;空间分辨率为0.0083333&#xff08;约1km),时间为2021年1月-2100年…...

21天掌握javaweb-->第8天:前后端分离架构与Axios请求

前后端分离架构概念 前后端分离架构是一种现代Web应用开发模式,其中前端和后端分别独立开发和部署,通过API进行数据交互。这种架构使得前端专注于用户界面和用户体验,而后端则专注于业务逻辑和数据处理。 优势 开发效率高:前后端可以并行开发,减少了开发时间。技术栈灵活…...

基于阻塞队列的生产者消费者模型动画演示

一个基于阻塞队列的生产者消费者模型的动画演示&#xff1a; 这是打包好的程序。程序是用 QT 写的。 通过网盘分享的文件&#xff1a;CP模型.7z 链接: https://pan.baidu.com/s/1YjC7YiSqHGqdr6bbffaDWg?pwde6g5 提取码: e6g5 CP模型...

DHCP和BOOTP选项及DHCP协议操作详解

DHCP和BOOTP选项及DHCP协议操作详解 DHCP与BOOTP简介 1. BOOTP&#xff08;Bootstrap Protocol&#xff09; 功能&#xff1a;提供静态配置的IP分配。用途&#xff1a;在早期用于无盘工作站启动时获取IP地址和基本配置。缺点&#xff1a;只能提供静态IP配置&#xff0c;无法动…...

数据结构--链表和单链表详解及实现

一.前言 数据结构思维导图如下&#xff0c;灰色标记的是之前讲过的&#xff0c;本文将带你走近单链表(红色标记部分)&#xff0c;希望大家有所收获&#x1f339;&#x1f339; 二.链表的定义和概念 在讲单链表之前&#xff0c;我们先学习一下链表 2.1 链表的定义 链表是一种…...

vue3基础知识

书接上文&#xff0c;这篇继续来学习vue3的核心语法&#xff0c;可以先看上一篇再来看这篇效果更好。 1. computed computed 用于创建 计算属性&#xff0c;即基于其他响应式数据的值动态计算并缓存的属性。它的主要作用是优化性能和提高代码的可维护性&#xff0c;避免不必要…...

【Linux系统】Ubuntu 缓冲区机制

在Ubuntu中&#xff0c;和其他操作系统有个不一样的机制&#xff1a;缓冲区。这篇文章是对与缓冲区的详细介绍。 在 Ubuntu 中&#xff08;以及其他基于 Linux 的操作系统&#xff09;&#xff0c;缓冲区&#xff08;Buffer&#xff09;是内核用于优化 I/O 操作的重要机制。它…...

ChatGPT 最新推出的 Pro 订阅计划,具备哪些能力 ?

OpenAI 最近推出了 ChatGPT Pro&#xff0c;这是一个每月收费 200 美元的高级订阅计划&#xff0c;旨在为用户提供对 OpenAI 最先进模型和功能的高级访问。 以下是 ChatGPT Pro 的主要功能和能力&#xff1a; 高级模型访问&#xff1a; o1 模型&#xff1a;包括 o1 和 o1 Pro…...

数据结构理论

内容来源青岛大学数据结构与算法课程&#xff0c;链接&#xff1a;数据结构与算法基础&#xff08;青岛大学-王卓&#xff09;_哔哩哔哩_bilibili 绪论 数据结构概述 数据结构和算法的定义&#xff1a;我们如何把现实中大量而复杂的问题以特定的数据类型和特定的存储结构保存…...

es 3期 第14节-全文文本分词查询

#### 1.Elasticsearch是数据库&#xff0c;不是普通的Java应用程序&#xff0c;传统数据库需要的硬件资源同样需要&#xff0c;提升性能最有效的就是升级硬件。 #### 2.Elasticsearch是文档型数据库&#xff0c;不是关系型数据库&#xff0c;不具备严格的ACID事务特性&#xff…...

六安市第二届网络安全大赛复现

misc 听说你也喜欢俄罗斯方块&#xff1f; ppt拼接之后 缺三个角补上 flag{qfnh_wergh_wqef} 流量分析 流量包分离出来一个压缩包 出来一张图片 黑色代表0白色代表1 101010 1000 rab 反的压缩包 转一下 密码&#xff1a;拾叁拾陆叁拾贰陆拾肆 密文&#xff1a;4p4n5758…...

Sarcomere仿人灵巧手ARTUS,20个自由度拓宽机器人作业边界

Sarcomere Dynamics 是一家深度技术先驱&#xff0c;通过开发和商业化仿人机械来改变机器人行业。专注于为科研人员&#xff0c;系统集成商和制造商提供更实惠、更轻便且更灵活的末端执行器替代品。凭借创新的致动器技术&#xff0c;创造了一款紧凑、轻便且非常坚固的机械手Art…...

Django drf 基于serializers 快速使用

1. 安装: pip install djangorestframework 2. 添加rest_framework到您的INSTALLED_APPS设置。 settings.pyINSTALLED_APPS [...rest_framework, ] 3. 定义模型 models.pyfrom django.db import modelsclass BookModel(models.Model):name models.CharField(max_length64)…...

pycharm集成环境中关于安装sklearn库报错问题分析及解决

在输入pip install sklearn后&#xff0c;出现如下提示&#xff1a; pip install sklearn Collecting sklearn Using cached sklearn-0.0.post12.tar.gz (2.6 kB) Installing build dependencies ... done Getting requirements to build wheel ... error error: subprocess-…...

AI - 浅聊一下基于LangChain的AI Agent

AI - 浅聊一下基于LangChain的AI Agent 大家好&#xff0c;今天我们来聊聊一个很有意思的主题&#xff1a; AI Agent &#xff0c;就是目前非常流行的所谓的AI智能体。AI的发展日新月异&#xff0c;都2024年末了&#xff0c;如果此时小伙伴们对这个非常火的概念还不清楚的话&a…...

《【Linux】深入理解进程管理与 fork 系统调用的实现原理》

一、引言 在 Linux 操作系统中&#xff0c;进程管理是核心功能之一。进程是操作系统进行资源分配和调度的基本单位。理解进程管理的原理以及 fork 系统调用的实现对于深入掌握 Linux 系统的运行机制至关重要。本文将深入探讨 Linux 中的进程管理以及 fork 系统调用的实现原理&a…...

docker-compose部署skywalking 8.1.0

一、下载镜像 #注意 skywalking-oap-server和skywalking java agent版本强关联&#xff0c;版本需要保持一致性 docker pull elasticsearch:7.9.0 docker pull apache/skywalking-oap-server:8.1.0-es7 docker pull apache/skywalking-ui:8.1.0二、部署文件docker-compose.yam…...

AI 总结的的 AI 学习路线

一、入门阶段&#xff1a;数学基础与编程语言 数学基础 线性代数 当年白纸黑字推演&#xff0c; 都是泪啊&#xff0c;草稿本都用了一卷。 学习向量、矩阵的基本概念&#xff0c;包括向量的加法、减法、点积和叉积&#xff0c;矩阵的乘法、转置等运算。例如&#xff0c;在计算…...

离散傅里叶级数(DFS)详解

1. 引言 离散傅里叶级数&#xff08;Discrete Fourier Series, DFS&#xff09;是信号处理领域中一项基础且重要的数学工具&#xff0c;用于分析和处理周期性的离散信号。它通过将离散时间信号表示为一组正弦和余弦的和&#xff0c;从而使得信号在频域上得到更清晰的描述。与连…...

Java 类加载机制详解

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家&#xff0c;历代文学网&#xff08;PC端可以访问&#xff1a;https://literature.sinhy.com/#/literature?__c1000&#xff0c;移动端可微信小程序搜索“历代文学”&#xff09;总架构师&#xff0c;15年工作经验&#xff0c;…...

1.1 Beginner Level学习之“编写简单的发布服务器和订阅服务器”(第十一节)

学习大纲&#xff1a; 1. 编写发布服务器节点 在 ROS 中&#xff0c;节点是连接到 ROS 网络的可执行文件。我创建了一个名为 talker 的发布者节点&#xff0c;它会向一个主题 chatter 不断发送消息。 首先&#xff0c;进入你的工作包 beginner_tutorials&#xff08;假设你已…...

AIQuora:开启论文写作新篇章

在这个信息爆炸的时代&#xff0c;学术写作已成为研究者不可或缺的技能。然而&#xff0c;面对繁重的写作任务&#xff0c;许多学者和学生常常感到力不从心。AIQuora&#xff0c;一个专业的文理工科论文智能写作助手&#xff0c;以其免费开题报告生成功能&#xff0c;为学术写作…...

【C语言】库函数常见的陷阱与缺陷(1):字符串处理函数

目录 一、 strcpy 函数 1.1. 功能与常见用法 1.2. 陷阱与缺陷 1.3. 安全替代 1.4. 代码示例 二、strcat 函数 2.1. 功能与常见用法 2.2. 陷阱与缺陷 2.3. 安全替代 2.4. 代码示例 三、strcmp 函数 3.1. 功能与常见用法 3.2. 陷阱与缺陷 3.3. 安全替代 3.4. 代…...

Mysql索引原理及优化——岁月云实战笔记

根据Mysql索引原理及优化这个博主的视频学习&#xff0c;对现在的岁月云项目中mysql进行优化&#xff0c;我要向这个博主致敬&#xff0c;之前应用居多&#xff0c;理论所知甚少&#xff0c;于是将学习到东西&#xff0c;应用下来&#xff0c;看看是否有好的改观。 1 索引原理…...

AGCRN论文解读

一、创新点 传统GCN只能基于静态预定义图建模全局共享模式&#xff0c;而AGCRN通过两种GCN的增强模块&#xff08;NAPL、DAGG&#xff09;实现了更精细的节点特性学习和图结构生成。 1 节点自适应参数学习模块&#xff08;NAPL&#xff09; 传统GCN通过共享参数&#xff08;权重…...

Python机器学习笔记(五、决策树集成)

集成&#xff08;ensemble&#xff09;是合并多个机器学习模型来构建更强大模型的方法。这里主要学习两种集成模型&#xff1a;一是随机森林&#xff08;random forest&#xff09;&#xff1b;二是梯度提升决策树&#xff08;gradient boosted decision tree&#xff09;。 1…...

Kafka单机及集群部署及基础命令

目录 一、 Kafka介绍1、kafka定义2、传统消息队列应用场景3、kafka特点和优势4、kafka角色介绍5、分区和副本的优势6、kafka 写入消息的流程 二、Kafka单机部署1、基础环境2、iptables -L -n配置3、下载并解压kafka部署包至/usr/local/目录4、修改server.properties5、修改/etc…...