java 内存管理_JAVA内存管理和JVM运行机制、垃圾回收、内存调优

一、JAVA内存管理

java是跨平台语言,java预编译.class文件放置JVM虚拟机中运行;

2052d1c79feb3d44e451968b674fe4c0.png

Java的内存结构,也就是运行时的数据区域

958429dcca6c99489b83cdd888b40793.png

运行时数据区

方法区:常量池、变量等存储地方;(持久区)

堆:实例对象存储地方;GC重点关照位置;(新生代和老年代)

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

Java方法栈:方法程序运行地方;Java栈总是与线程关联在一起的,每当创建一个线程,JVM就会为该线程创建对应的Java栈;

本地方法栈:java方法与本地相关联

Java内存模型(即Java Memory Model,简称JMM)本身是一种抽象的概念,并不真实存在;每个线程创建时JVM都会为其创建一个工作内存(有些地方称为栈空间),用于存储线程私有的数据,而Java内存模型中规定所有变量都存储在主内存,主内存是共享内存区域,所有线程都可以访问,但线程对变量的操作(读取赋值等)必须在工作内存中进行,首先要将变量从主内存拷贝的自己的工作内存空间,然后对变量进行操作,操作完成后再将变量写回主内存,不能直接操作主内存中的变量;JMM与Java内存区域的划分是不同的概念层次,Java内存模型是围绕着并发编程中原子性、可见性、有序性这三个特征来建立的;

a08168376aa99ce3d3b4b8361a8cb57c.png

JVM运行机制:JVM转入环境和配置(java.exe-->jvm.cfg)、装载JVM(通过LoadJavaVM来装载)、启动JVM获得本地调用接口、运行java程序;

1.类加载器:

包括启动类加载器(核心库)(jre/lib内所有class)、扩展类(扩展库)(jre/lib/ext/内所有class)、应用程序类(加载classpath下的所有class)

流程(五个阶段):加载、连接(验证、准备、解析)、初始化

2.执行引擎:执行classes中指令。任何JVM specification实现(JDK)的核心都是执行引擎。

3.运行时数据区又叫JVM内存;

内存区域包括堆和栈,栈后进先出,有大小限制,堆可以存放很大内容;java虚拟机中的堆主要存放对象实例,栈存放引用和基本类型;

存放大内容:对象实例

大小有限制,后进先出,存放:引用和基本类型

速度快

所有线程共享,生命周期和JVM同步

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

如:Object o=new Object();   Object存放在堆中,o存放在栈中;

程序计数器:标记程序运行到的位置,JVM相当于一台抽象计算机;

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

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

2.装载JVM.dll

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

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

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

JAVA虚拟机的GC有自动的垃圾回收机制,不需要开发人员主动调用;由于是自动,优先级又低,所以并不能保证一定执行垃圾回收;

(1)什么时候回收?

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

(2)判断什么是垃圾?

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

(3)垃圾回收方法

a.finalize方法工作原理:一旦垃圾回收器准备好释放对象存储空间时,会调用一次并且仅调一次finalize方法(通过调用System.gc()实现),并且当下一次垃圾回收开始时才真正回收对象占用内存;所以重现finalize方法即可在回收时重新建立引用关联,而达到不被回收效果;

b.标记(mark):标记出哪些不是垃圾,回收的时候把没有标记到的认为是垃圾,进行回收;

c.引用计数法:针对每个对象实例的引用进行计算,将计算为0的作为垃圾进行回收,弊端,循环引用将无法进行回收;

d.年轻代:新生成对象放在年轻代中,收集生命周期短的;老年代:被多次垃圾回收的对象;持久代:用于存放静态文件,如java类或方法等;

(1)对新生代的对象的收集称为minor GC;

(2)对旧生代的对象的收集称为Full GC;

(3)程序中主动调用System.gc()强制执行的GC为Full GC。

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

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

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

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

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

三、内存调优

主要针对年轻代、年老代、持久代;堆栈等大小进行设置;

内存溢出分2类:

1. 年老代溢出,表现为:java.lang.OutOfMemoryError:Javaheapspace

2. 持久代溢出,表现为:java.lang.OutOfMemoryError:PermGenspace

参数说明

-Xmx3550m:设置JVM最大堆内存为3550M。

-Xms3550m:设置JVM初始堆内存为3550M。此值可以设置与-Xmx相同,以避免每次垃圾回收完成后JVM重新分配内存。

-Xss128k:设置每个线程的栈大小。JDK5.0以后每个线程栈大小为1M,之前每个线程栈大小为256K。应当根据应用的线程所需内存大小进行调整。在相同物理内存下,减小这个值能生成更多的线程。但是操作系统对一个进程内的线程数还是有限制的,不能无限生成,经验值在3000~5000左右。需要注意的是:当这个值被设置的较大(例如>2MB)时将会在很大程度上降低系统的性能。

-Xmn2g:设置年轻代大小为2G。在整个堆内存大小确定的情况下,增大年轻代将会减小年老代,反之亦然。此值关系到JVM垃圾回收,对系统性能影响较大,官方推荐配置为整个堆大小的3/8。

-XX:NewSize=1024m:设置年轻代初始值为1024M。

-XX:MaxNewSize=1024m:设置年轻代最大值为1024M。

-XX:PermSize=256m:设置持久代初始值为256M。

-XX:MaxPermSize=256m:设置持久代最大值为256M。

-XX:NewRatio=4:设置年轻代(包括1个Eden和2个Survivor区)与年老代的比值。表示年轻代比年老代为1:4。

-XX:SurvivorRatio=4:设置年轻代中Eden区与Survivor区的比值。表示2个Survivor区(JVM堆内存年轻代中默认有2个大小相等的Survivor区)与1个Eden区的比值为2:4,即1个Survivor区占整个年轻代大小的1/6。

-XX:MaxTenuringThreshold=7:表示一个对象如果在Survivor区(救助空间)移动了7次还没有被垃圾回收就进入年老代。如果设置为0的话,则年轻代对象不经过Survivor区,直接进入年老代,对于需要大量常驻内存的应用,这样做可以提高效率。如果将此值设置为一个较大值,则年轻代对象会在Survivor区进行多次复制,这样可以增加对象在年轻代存活时间,增加对象在年轻代被垃圾回收的概率,减少Full GC的频率,这样做可以在某种程度上提高服务稳定性。

四、内存溢出常见的哪几种?

三种:

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大小即可。

五、java自带分析工具:

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

问题汇总:

1.什么是java虚拟机?为什么java平台被称为“平台无关的编程语言”?

答:java虚拟机是一个可以执行java字节码的虚拟机进程;java虚拟机支持任意平台上运行,java程序编写好后,只要编译成java字节码(class文件)就可以在java虚拟机上运行,java程序无须根据不同环境进行编译;

2.java代码是怎样运行的?

答:java虚拟机运行时数据区分五个区域:方法区、堆、java方法栈、本地方法栈和pc寄存器;java程序被编译成java字节码后,首先要存放在方法区;方能在java虚拟机中运行;为了提高运行效率,标准JDK中的HotSpot虚拟机采用的是一种混合执行的策略。

3.java虚拟机是如何加载java类的?

答:java虚拟机将字节流转换java类三个步骤:装载、链接(验证、准备、解析)、初始化;

4.jvm的永久代会进行垃圾回收吗?

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

5.对象什么时候可以被垃圾回收?

答:当对象不再被任何对象引用时,可以被回收

6.System.gc()和Rumtime.gc()会做什么事情?

答:都是提示JVM要进行垃圾回收,但是具体什么时候进行垃圾回收要看JVM虚拟机;

7.堆的形状是一颗()。

完全二叉树

满二叉树

二叉排序树

平衡二叉树

答案:A

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java可视化内存管理是指通过可视化界面展示出Java程序运行中的内存情况,并提供相关图表和数据分析,从而帮助开发人员更好地理解和管理内存资源。 Java程序在运行过程中,需要通过内存来存储各种对象和数据。如果内存管理不当,会引发内存泄漏、内存溢出等问题,严重影响程序的性能和稳定性。为了解决这些问题,Java提供了一套自动内存管理机制,即垃圾回收器。垃圾回收器负责自动释放不再使用的内存,使得开发人员不需要关心手动内存管理,从而提高开发效率和程序的可维护性。 可视化内存管理工具可以帮助开发人员实时监控Java程序的内存使用情况。通过界面展示当前内存分配情况、垃圾回收的执行情况及效果,开发人员可以及时了解到程序的内存状况,及时进行优化和调整。 这些工具一般提供了一些核心功能,如内存使用曲线图、内存分配情况、垃圾回收器的执行情况等。通过这些图表和数据,开发人员可以直观地了解到内存使用的趋势和规律,以及垃圾回收的效果。开发人员可以根据这些信息,采取相应的优化措施,例如调整内存分配策略、提高垃圾回收效率等,从而改善程序的性能和稳定性。 总之,Java可视化内存管理工具能够帮助开发人员更好地了解和管理程序的内存资源,提高程序的性能和稳定性。通过实时监控和分析内存使用情况,开发人员可以针对性地进行优化和调整,从而提高程序的运行效果。 ### 回答2: Java的可视化内存管理主要是指Java虚拟机(JVM)对内存资源的分配、使用和回收进行可视化展示,以便开发者更好地了解和优化程序的内存使用情况。 首先,Java内存管理是由JVM负责的,JVM内存区域划分为多个部分,包括堆内存、栈内存、方法区等。Java可视化内存管理能够展示这些内存区域的使用情况,如堆内存的大小、已使用空间和剩余空间等,开发者可以根据这些信息进行内存调优。 其次,Java的可视化内存管理能够显示对象的创建和销毁过程。开发者可以观察对象的生命周期,了解对象的创建时间、存活时间和销毁时间,从而判断对象是否存在内存泄漏或过早销毁等问题。 此外,Java的可视化内存管理还能够展示对象之间的引用关系。开发者可以查看对象的引用链,了解对象之间的依赖关系和循环引用等情况,以便及时解除无用的引用,避免内存泄漏问题。 除了以上功能,Java的可视化内存管理还可以显示内存的使用情况和内存泄漏等警告信息。开发者可以通过监控内存的使用情况,及时发现内存占用过高的问题,并进行相应的调优。同时,当存在内存泄漏时,可视化内存管理工具会给出相应的警告信息,提醒开发者进行修复。 综上所述,Java的可视化内存管理可以帮助开发者更好地了解和优化程序的内存使用情况,提高程序的性能和稳定性。通过可视化的展示,开发者可以及时发现和解决内存相关的问题,提升程序的质量。 ### 回答3: Java提供了一个可视化的内存管理机制,通过Java虚拟机(JVM)来实现。在Java程序中,内存分为堆(Heap)和栈(Stack)两部分。 堆是用来存储对象的地方,所有的对象都在堆中分配空间。堆是在Java虚拟机启动时自动创建的,其大小由启动参数决定。堆的管理是自动的,即当对象不再被引用时,垃圾回收器会自动释放该对象占用的内存空间。为了保证堆的高效利用,Java提供了分代垃圾回收机制,将堆分为新生代和老年代,不同的对象会被分配到不同的代中,并采用不同的垃圾回收算法。 栈是用来存储变量和方法调用的地方。每个线程在运行时都会有一个独立的栈,用来存储局部变量和方法的调用信息。栈中的内存分配和释放是自动的,一旦方法调用结束,栈帧中的数据就会被立即释放。 Java通过可视化工具,例如Java VisualVM,来监控和管理内存使用情况。这些工具提供了图形化的界面,可以实时查看堆和栈的使用情况,包括对象的数量、大小、引用关系等。同时,它们还提供了垃圾回收的相关信息,例如回收时间、频率等。通过这些工具,开发人员可以及时发现内存泄漏和性能问题,并采取相应的措施进行调优。 总的来说,Java的可视化内存管理机制提供了方便、高效的方式来管理内存,帮助开发人员更好地优化程序性能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值