一、JAVA内存管理
java是跨平台语言,java预编译.class文件放置JVM虚拟机中运行;
Java的内存结构,也就是运行时的数据区域
运行时数据区
方法区:常量池、变量等存储地方;(持久区)
堆:实例对象存储地方;GC重点关照位置;(新生代和老年代)
程序计数器:记录程序下一步指令;
Java方法栈:方法程序运行地方;Java栈总是与线程关联在一起的,每当创建一个线程,JVM就会为该线程创建对应的Java栈;
本地方法栈:java方法与本地相关联
Java内存模型(即Java Memory Model,简称JMM)本身是一种抽象的概念,并不真实存在;每个线程创建时JVM都会为其创建一个工作内存(有些地方称为栈空间),用于存储线程私有的数据,而Java内存模型中规定所有变量都存储在主内存,主内存是共享内存区域,所有线程都可以访问,但线程对变量的操作(读取赋值等)必须在工作内存中进行,首先要将变量从主内存拷贝的自己的工作内存空间,然后对变量进行操作,操作完成后再将变量写回主内存,不能直接操作主内存中的变量;JMM与Java内存区域的划分是不同的概念层次,Java内存模型是围绕着并发编程中原子性、可见性、有序性这三个特征来建立的;
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