Java中运行机制和内存机制的原理是什么

发布时间:2021-06-18 16:00:37 作者:Leah
来源:亿速云 阅读:156

本篇文章为大家展示了Java中运行机制和内存机制的原理是什么,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。

一、Java的运行机制

JVM工作原理和特点主要是指操作系统装入JVM是通过jdk中Java.exe来完成,通过下面4步来完成JVM环境.

1.创建JVM装载环境和配置

2.装载JVM.dll

3.初始化JVM.dll并挂界到JNIEnv(JNI调用接口)实例

4.调用JNIEnv实例装载并处理class类。

Java中运行机制和内存机制的原理是什么

类从被加载到虚拟机类存中开始,到被卸载出内存为止,它的整个生命周期包括:

加载 → 验证 → 准备 → 解析 → 初始化 → 使用 → 卸载   (7个部分)

加载

1.JDK在执行程序运行命令时会去JRE目录中找到jvm.dll , 并初始化JVM,这时会产生一个Bootstrap Loader(启动类加载器)

2.Bootstrap Loader 自动加载 Extended Loader(标准扩展类加载器)

3.Bootstrap Loader 自动加载 AppClass Loader(系统类加载器)

4.最后由 AppClass Loader 加载 我们指定(想要运行)的 java 类 

验证

文件格式验证->元数据验证->字节码验证->符号引用验证

1.文件格式验证:主要是检查字节码的字节流是否符合Class文件格式的规范,验证该文件是否能被当前的 jvm 所处理,

如果没问题,字节里就可以进入方法区进行保存了;

2.元数据验证:对字节码描述的信息进行语义分析,保证其描述的内容符合java语言的语法规范,能被java虚拟机识别;

3.字节码验证:该部分最为复杂,对方法体内的内容进行验证,保证代码在运行时不会做出什么危害虚拟机安全的事件;

4.符号引用验证:来验证一些引用的真实性与可行性,比如代码里面引了其他类(符号中通过字符串描述的全限定名是否能找到对应的类),这里就要去检测一下那些来究竟是否存在;或者说代码中访问了其他类的一些属性,这里就对那些属性的可以访问行进行了检验

准备

准备阶段会为类变量(指的是静态变量,这就是我们常说的,静态变量/方法 在类加载的时候就执行了,通过类名.静态**来调用)分配内存并设置类的初始值;   值得一提的是 如果有以下语句: 

public static int i = 123 ;

在准备阶段的初始值是 0 ,而不是 123 ,是因为此时 只是分配内存空间而已, 并没有对 i 进行初始化, 真正的对 i 赋值是在 初始化 阶段

解析

1.类或接口的解析;

2.字段解析;

3.类方法解析;

4.接口方法解析;

初始化:(也可以在解析之前执行)

类初始化阶段是类加载过程中的最后一步,这才是执行类中定义的java程序代码(也可以说是字节码)在准备阶段,已经为变量赋过一次系统要求的初始值,到了初始化阶段会根据程序员的要求出初始化变量赋值。

1.遇到new,get static,put static,invoke static这4条字节码指令时,假如类还没进行初始化,则马上对其进行初始化工作

2.使用java.lang.reflect.*的方法对类进行反射调用时,如果类还没有进行过初始化,立即马上光速对其进行初始化!!!

3.初始化一个类的时候,如果其父类还没有被初始化,那么会先去初始化其父类;

二、java的内存结构

Java中运行机制和内存机制的原理是什么

在java中内存的占用主要分为四块:堆、栈、方法区和程序计数器其中,堆和栈是使用最多的。
堆(heap):堆是一个运行时数据区,主要存放new出来的一些对象和数组
栈(stack):栈中主要保存一些基本的数据类型和对象的引用变量

方法区
        1、常量池:存放一些字符串常量和基本类型常量。2、静态区:内存在程序编译时就分配好的区域,主要存放一些静态变量(static的);3、 代码区:存放程序方法的二进制代码,而且是多个对象共享一个代码空间区域;

程序计数器:记录程序下一步指令;

堆栈的运行机智和优缺点

中分配的内存,是运行时动态分配的,所以生存期也不必告诉编译区,由java虚拟机的垃圾回收机制回收,缺点:因为是动态分配内存,所以存取速度较慢。堆可以理解为一颗完全二叉树

的优势是:当在一段代码块定义一个变量时,Java就在栈中为这个变量分配内存空间,当超过变量的作用域后,Java会自动释放掉为该变量所分配的内存空间,该内存空间可以立即被另作他用。优点是:存取速度快,仅次于CPU中的寄存器,同时,数据可以共享。缺点是:数据大小和生存期必须是确定的,缺乏灵活性

存放大内容:对象实例大小有限制,后进先出,存放:引用和基本类型
速度快
所有线程共享,生命周期和JVM同步

每个线程有自己独立的栈,线程销毁的时候被销毁

三、java的内存管理机制

JVM内存管理机制说到底就是为了解决两个问题:给对象分配内存以及回收分配给对象的内存

Java变量在内存中的分配

1、类变量(static修饰的变量):在程序加载时系统就为它在堆中开辟了内存,堆中的内存地址存放于栈以便于高速访问。静态变量的生命周期–一直持续到整个”系统”关闭。 

2、实例变量:当你使用java关键字new的时候,系统在堆中开辟并不一定是连续的空间分配给变量(比如说类实例),然后根据零散的堆内存地址,通过哈希算法换算为一长串数字以表征这个变量在堆中的”物理位置”。 实例变量的生命周期–当实例变量的引用丢失后,将被GC(垃圾回收器)列入可回收“名单”中,但并不是马上就释放堆中内存。 
3、局部变量:局部变量,由声明在某方法,或某代码段里(比如for循环),执行到它的时候在栈中开辟内存,当局部变量一但脱离作用域,内存立即释放。 

4、String str =  new String("abc")和String str = "abc";

当通过new产生一个字符串时,先去常量池中查看是否有"abc"对象,如果没有,则在常量池中创建一个"abc"对象,然后在堆中创建一个常量池中"abc"的copy对象

第二种是先在栈中创建一个String对象的引用变量str的引用,然后找到栈中有没有存放值为"abc"的地址,如果没有,就开创一个字面值为"abc"的地址,然后在常量池中创建一个"abc"对象,并指向这个地址

不同的对象引用类型, GC会采用不同的方法进行回收,JVM对象的引用分为了四种类型

(1)强引用:默认情况下,对象采用的均为强引用(这个对象的实例没有其他对象引用,GC时才会被回收)

(2)软引用:软引用是Java中提供的一种比较适合于缓存场景的应用(只有在内存不够用的情况下才会被GC)

(3)弱引用:在GC时一定会被GC回收

(4)虚引用:由于虚引用只是用来得知对象是否被GC

垃圾回收(GC,全称:garbage colletion)

(1)什么时候回收?

当内存不足或者当前空闲的时候进行垃圾回收,GC线程优先级都不太高;

(2)判断什么是垃圾?

一般来说,对于没有引用指向的对象,被标识为垃圾,没有对象指向它,也就无法对它进行操作,这个对象对于我们来说就是没用的;

(3)垃圾回收方法

标记清除算法、复制算法、标记整理算法以及分代回收算法

标记清除算法

该算法分为“标记”和“清除”两个阶段:标记阶段的任务是标记出所有需要被回收的对象,清除阶段就是回收被标记的对象所占用的空间。它是最基础的收集算法,效率也很高,但是会带来两个明显的问题:

Java中运行机制和内存机制的原理是什么

复制算法

将内存分为大小相同的两块,每次使用其中的一块。当第一块的内存使用完后,就将还存活的对象复制到另一块去,然后再把使用的空间一次清理掉。这样就使每次的内存回收都是对内存区间的一半进行回收。

这种算法的优缺点也比较明显

标记整理算法

为了解决复制算法的缺陷,充分利用内存空间,提出了标记整理算法。该算法标记阶段和标记清除一样,但是在完成标记之后,它不是直接清理可回收对象,而是将存活对象都向一端移动,然后清理掉端边界以外的内存。 如下图所示:

Java中运行机制和内存机制的原理是什么

分代收集算法

在堆中,采用分代收集算法,这种算法就是根据具体的情况选择具体的垃圾回收算法。一般将 java 堆分为新生代和老年代(Young Generation 和 Old Gereration),这样我们就可以根据各个年代的特点选择合适的垃圾收集算法。

Java中运行机制和内存机制的原理是什么

新生代

老年代

老年代的对象存活几率是比较高的,而且没有额外的空间对它进行分配担保,所以我们必须选择“标记-清除”或“标记-整理”算法进行垃圾收集。且回收频率不高

对象来源:

永久代

堆=新生代+老年代,不包括永久代(方法区)。

jvm的永久代(方法区)、常量池和变量的存储区,垃圾回收不会发生在永久代,当永久代存储满了或超过了临界值后才会触发完全垃圾回收(full gc);jdk8后永久代已经去除,改存元空间区(本地内存区)中;最大可利用空间就变成了整个系统内存的可用空间.

在 JDK 1.8 中, HotSpot 已经没有 “PermGen space”这个区间了,取而代之是一个叫做 Metaspace(元空间) 的东西。下面我们就来看看 Metaspace 与 PermGen space 的区别。

Java中运行机制和内存机制的原理是什么

移除永久代的原因
  1. 移除永久代是为融合HotSpot JVM与JRockit VM而做出的努力,因为JRockit没有永久代,不需要配置永久代。

  2. 永久代大小不确定,PermSize指定的太小很容易造成永久代OOM,因为PermSize的大小很依赖于很多因素,比如JVM加载的class总数,常量池的大小,方法的大小等。

四、内存调优

1、新生代和老年代溢出:java.lang.OutOfMemoryError:java heep space;当98%时间用于垃圾回收时,且可用的Heap size 不足2%的时候将抛出此异常信息;

解决方法:手动设置JVM Heap(堆)的大小

2、持久代溢出:java.lang.OutOfMemoryError: PermGen space

解决方法: 通过-XX:PermSize和-XX:MaxPermSize设置永久代大小即可。

3、栈溢出:java.lang.StackOverFlowError:Thread stack space

栈区远远小于堆区,栈区需要的内存大小1-2m左右;出现栈溢出,即说明单线程运行程序需要的内存太大;

解决方法:1:修改程序。2:通过 -Xss: 来设置每个线程的Stack大小即可。

4、java自带分析工具:

jstack(查看线程)、jmap(查看内存)和jstat(性能分析)命令

参数说明

上述内容就是Java中运行机制和内存机制的原理是什么,你们学到知识或技能了吗?如果还想学到更多技能或者丰富自己的知识储备,欢迎关注亿速云行业资讯频道。

推荐阅读:
  1. Java内存机制和GC回收机制是什么
  2. Java的运行机制是什么

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

java

上一篇:linux系统中怎么安装 oracle 和jdk8

下一篇:python清洗文件中数据的方法

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》