JVM在操作系统的上一层,是运行在操作系统上的。JRE是运行环境,而JVM是包含在JRE中
垃圾回收主要在方法区和堆,所以”JVM调优“大部分也是发生在方法区和堆中
可以说调优就是发生在堆中,方法区可以理解为在堆中元空间,但是又以“非堆”区分。调优发生在堆中
1.作用:加载Class文件
2.代码:
public class Car {
public static void main(String[] args) {
//类是模板,对象是具体的
//同一套类模板,new出三个对象
//从类到对象
Car car1 = new Car();
Car car2 = new Car();
Car car3 = new Car();
//打印出三个对象的hashcode值---不一样
System.out.println(car1.hashCode());
System.out.println(car2.hashCode());
System.out.println(car3.hashCode());
//从对象到类(getClass)
Class<? extends Car> aClass1 = car1.getClass();
Class<? extends Car> aClass2 = car2.getClass();
Class<? extends Car> aClass3 = car3.getClass();
//打印出由三个实例返回来得到的class的hashcode值---一样
System.out.println(aClass1.hashCode());
System.out.println(aClass2.hashCode());
System.out.println(aClass3.hashCode());
/* 输出:460141958
* 1163157884
* 1956725890
* 685325104
* 685325104
* 685325104
* */
}
}
3.分为:
类加载器加载类时,会先从应用程序加载器找,一层一层往上找
ClassLoader classLoader = aClass1.getClassLoader();
System.out.println(classLoader);//AppClassLoader
System.out.println(classLoader.getParent());//ExtClassLoader
System.out.println(classLoader.getParent().getParent());//null
null—两种情况
ExtClassLoader
4.双亲委派机制 :安全
当加载一个类时,app–>ext–>boot 向上查找,但是执行时向下执行 ,向上加载,向下执行
例如,当自己写一个Stirng类的时候,这个时候是首先在application加载器就能找到,但是还要继续找,boot中也有String类,所以执行时,往下执行,先执行根加载器的类。所以不管你自己重写什么类,系统加载的还是根加载器中的,除非你把JVM改了
步骤:
简单来说,就是虽然说类加载器加载请求,是从最下层开始查找,但是即便子类可以加载,也还是要往上委托,一直委托到启动类加载器,再来看是否能加载,不能再让子加载器去加载
向上委托,向下加载
类装载器采用的机制是双亲委派机制:
5.沙箱安全机制
Java安全模型的核心就是Java沙箱(sandbox);沙箱是一个限制程序运行的环境,沙箱机制就是将Java代码限定在虚拟机特定的运行范围中,并且严格限制代码对本地系统的访问,通过这样的措施来保证对代码的有效隔离,防止对本地系统造成破坏。沙箱主要限制系统资源访问 ,系统资源包括:CPU,内存,文件系统,网络。不同级别的沙箱对这些资源访问的限制也可以不一样。
所有的Java程序运行都可以指定沙箱,可以定制安全策略
在Java中将执行代码分为本地代码和远程代码两种,本地代码默认视为可信任的,而远程代码则被看作是不受信任的。对于授信的本地代码,可以访问一切本地资源,而对于非授信的远程代码在早期 的Java实现中,完全依赖于沙箱机制,如下图:
但如此严格的安全机制也给程序的功能扩展带来障碍,比如当用户希望远程代码访问本地系统的文件时,就无法实现(远程代码都被沙箱限制),因此在后续Java1.1版本中,针对安全机制做了改进,增加了安全策略,允许用户指定代码对本地资源的访问权限。----在沙箱下,加了受信任权限,用户可以指定代码是受信任的,因此也就可以访问本地资源!!
在Java1.2版本中,再次改进了安全机制,增加了代码签名 ,不论是本地代码还是远程代码,都会按照用户的安全策略设定,由类加载器加载到虚拟机中权限不同的运行空间,来实现差异化的代码执行权限控制。
如下图
当前最新的安全机制实现,则引入了域的概念,虚拟机会把所有代码加载到不同的系统域和应用域,系统域部分专门负责与关键资源进行交互,而各个应用域部分则通过系统域的部分代理来对各种需要的资源进行访问。虚拟机中不同的的受保护域,对应不一样的权限,存在于不同域中的类文件就具有了当前域的全部权限
如下图:
组成沙箱的基本组件:
-防止恶意代码去干涉善意的代码;(双亲委派机制)
-它守护了被信任的类库边界
-它将代码归入保护域,确定了代码可以进行哪些操作
虚拟机为不同的类加载器载入的类提供不同的命名空间,命名空间由一系列唯一的名称组成,每一个被装载的类将有一个名字,这个命名空间是由java虚拟机为每一个类装载器维护的 ,它们互相之间甚至不可见
重点理解双亲委派机制
每个线程都有一个程序计数器,是线程私有的,就是一个指针,指向方法区中的方法字节码(用来存储指向一条命指令的地址,也可以是即将要执行的指令代码),在执行引擎读取下一条指令,是一个非常小的内存空间,几乎可以忽略不计
方法区是被所有线程共享,所有字段和方法字节码,以及一些特殊方法,如构造函数,接口代码也在此定义,简单说,所有定义的方法的信息都保存在该区域,此区域属于共享空间
静态变量、常量、类信息(构造方法、接口定义)、运行时的常量池存在方法区中,但是实例变量存在堆内存中,和方法区无关
方法区:static、final、Class、常量池
定义一个Test类,new一个对象test1,实例化,赋值,都是在方法区中
1、栈:是一种数据结构
先进后出、后进先出 :类似一个桶
队列:先进先出(FIFO:First Input First Output)
栈~main方法先执行,(先进后出),要等其他方法执行完结束,main方法才会结束
栈~函数调用是通过栈这种数据结构实现的,每当程序执行进入一个函数调用,栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,就会导致栈溢出。
2、栈:栈内存,主管 程序的运行,生命周期和线程同步;
线程结束,栈内存也就释放,对于栈来说,不存在垃圾回收问题 。一旦线程结束,栈就over。栈是运行 时才发生的!线程结束栈就over
3、栈里面:八大基本类型+对象引用+实例的方法
栈运行原理:栈帧
程序正在执行的方法,一定在栈的顶部!
栈满了-----栈溢出StackOverFlowError
创建一个类,在栈中引用方法,在堆中对象具体的实例,常量又调用方法区中的常量池
执行代码:
public class Person {
String name;
int age;
void sing(){
System.out.println("人的姓名:"+name);
System.out.println("人的年龄:"+age);
}
public static void main(String[] args) {
Person person = new Person();
person.age=12;
person.name="xqh";
person.sing();
}
}
1–首先,类中的成员变量和方法体会进入到方法区中
2–程序执行到main()方法时,main()函数方法体会进入栈区,这一过程叫做压栈,定义了一个用于指向Person实例的变量person
3–程序执行到 Person person = new Person(); 就会在堆内存开辟一块内存区间 ,用于存放 Person 实例对象,然后将成员变量和成员方法放在 new 实例中都是取成员变量&成员方法的地址值 如图:
4–接下来对 person 对象进行赋值, person.name = “xqh” ; perison.age = 12;
先在栈区找到 person引用变量,然后根据地址值找到 new Person() 进行赋值操作。
5–当程序走到 sing() 方法时,还是一样,先到栈区找到 person这个引用变量,然后根据其地址值在堆内存中找到 new Person() 实例,然后进行方法调用。
在方法体void speak()被调用完成后,就会立刻马上从栈内弹出(出栈 )
最后,在main()函数完成后,main()函数也会出栈
Java HotSpot™ 64-Bit Server VM (build 25.321-b07, mixed mode)
JRockit:BEA公司 JRockit JVM是世界上最快的JVM
J9VM : IBM公司
GC垃圾回收主要在新生区和养老区,对应轻GC和重GC(Full GC)
4、新生区 、养老区
分为伊甸园和幸存者(0,1)
真理:百分之99的对象都是临时对象。new
伊甸园区满—>触发轻GC---->还需要引用的放到幸存者区,不再引用的清除 ----->新生区满了后,进行重GC,存活下来的进入养老区---->都满了之后,就报错误 ,堆内存已满
5、永久区
这个区域常驻内存的,用来存放jdk自身携带的Class对象,interface元数据。存储的是java运行时的一些环境或类信息~这个区域不存在垃圾回收,关闭JVM虚拟机就会释放这个区域的内存
报错:一个启动类,加载了大量的第三方jar包,Tomcat部署了太多的应用,大量动态生成的反射类。这些东西不断的被加载,直到内存满,就会出现OOM
jdk1.8以后,就叫元空间,常量池在方法区中,而方法区则在元空间中。----但是有时候会把方法区叫做非堆,和堆区分开来,理解意思就行。
元空间 — 逻辑上存在,物理上不存在(通过计算内存,发现只有新生代和老年代才算内存)
public class Hello {
public static void main(String[] args) {
String str = "xqhxuejava";
while(true){
str += str+new Random().nextInt(888888)+new Random().nextInt(999999);
}
}
}
设置vm options:-Xms8m -Xmx8m -XX:+PrintGCDetails
[GC (Allocation Failure) [PSYoungGen: 1536K->488K(2048K)] 1536K->664K(7680K), 0.0019815 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[GC (Allocation Failure) [PSYoungGen: 1673K->511K(2048K)] 1849K->1008K(7680K), 0.0009834 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[GC (Allocation Failure) [PSYoungGen: 1607K->400K(2048K)] 2103K->1775K(7680K), 0.0010556 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[GC (Allocation Failure) [PSYoungGen: 1844K->432K(2048K)] 3220K->2863K(7680K), 0.0008990 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[GC (Allocation Failure) [PSYoungGen: 1896K->416K(2048K)] 5735K->4959K(7680K), 0.0009835 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[Full GC (Ergonomics) [PSYoungGen: 416K->0K(2048K)] [ParOldGen: 4543K->1495K(5632K)] 4959K->1495K(7680K), [Metaspace: 3238K->3238K(1056768K)], 0.0047233 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[GC (Allocation Failure) [PSYoungGen: 737K->64K(2048K)] 3640K->2966K(7680K), 0.0006234 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[GC (Allocation Failure) [PSYoungGen: 64K->96K(2048K)] 2966K->2998K(7680K), 0.0003145 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[Full GC (Allocation Failure) [PSYoungGen: 96K->0K(2048K)] [ParOldGen: 2902K->2714K(5632K)] 2998K->2714K(7680K), [Metaspace: 3238K->3238K(1056768K)], 0.0049125 secs] [Times: user=0.01 sys=0.00, real=0.00 secs]
[Full GC (Ergonomics) [PSYoungGen: 1518K->0K(2048K)] [ParOldGen: 5530K->2014K(5632K)] 7048K->2014K(7680K), [Metaspace: 3309K->3309K(1056768K)], 0.0044064 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[Full GC (Ergonomics) [PSYoungGen: 1473K->0K(2048K)] [ParOldGen: 4829K->4854K(5632K)] 6303K->4854K(7680K), [Metaspace: 3324K->3324K(1056768K)], 0.0052257 secs] [Times: user=0.16 sys=0.00, real=0.01 secs]
[GC (Allocation Failure) [PSYoungGen: 0K->0K(2048K)] 4854K->4854K(7680K), 0.0003076 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
[Full GC (Allocation Failure) [PSYoungGen: 0K->0K(2048K)] [ParOldGen: 4854K->4835K(5632K)] 4854K->4835K(7680K), [Metaspace: 3324K->3324K(1056768K)], 0.0047963 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
Heap
PSYoungGen total 2048K, used 46K [0x00000000ffd80000, 0x0000000100000000, 0x0000000100000000)
eden space 1536K, 3% used [0x00000000ffd80000,0x00000000ffd8b868,0x00000000fff00000)
from space 512K, 0% used [0x00000000fff80000,0x00000000fff80000,0x0000000100000000)
to space 512K, 0% used [0x00000000fff00000,0x00000000fff00000,0x00000000fff80000)
ParOldGen total 5632K, used 4835K [0x00000000ff800000, 0x00000000ffd80000, 0x00000000ffd80000)
object space 5632K, 85% used [0x00000000ff800000,0x00000000ffcb8c10,0x00000000ffd80000)
Metaspace used 3358K, capacity 4496K, committed 4864K, reserved 1056768K
class space used 366K, capacity 388K, committed 512K, reserved 1048576K
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
可以清楚的看出,先是轻GC,新生区满了后,进行full GC,然后又是一系列GC,一直到最后,新生区、老生区都满了,于是报堆满了的错误 java.lang.OutOfMemoryError: Java heap space
在一个项目中,突然出现了OOM故障,在扩大了堆内存后,仍然报错,如何排除?
MAT、Jprofiler作用:
IDEA下载插件Jprofiler ----> 下载客户端Jprofiler ----->在IDEA里测试如何分析OOM
—分析OOM
代码
//DUMP
//-Xms 设置初始化内存分配大小 1/64
//-Xmx 设置最大分配内存 ,默认 1/4
// -XX:PrintGCDetails 打印GC垃圾回收信息
//-XX:+HeapDumpOnOutOfMemoryError dump出一个文件,来分析出错的代码行和大的对象
//-Xms1m -Xmx8m -XX:+HeapDumpOnOutOfMemoryError
public class Demo03 {
byte[]array = new byte[1*1024*1024]; //1M
public static void main(String[] args) {
ArrayList<Demo03>list = new ArrayList<>();
int count = 0;
try{
while(true){
list.add(new Demo03()); //问题所在
count+=1;
}
}catch(Error e){
System.out.println("count:"+count);
e.printStackTrace();
}
}
}
报错OOM
修改VM options:-Xms1m -Xmx8m -XX:+HeapDumpOnOutOfMemoryError
运行,会有一个dump文件,位置在项目文件夹内
输出:
dump文件在
打开这个dump出来的文件 ,主要就是看Biggest Objects 和 Thread Dump这两个位置
从大对象这里,可以明显看出哪个对象是占内存大的,如图,ArrayList明显有问题,占87%
可以看出,有问题的代码是第12行,也就是 list.add(new Demo03()); //问题所在
这样就是一个分析OOM的过程,通过JProfiler来加载dump出来的文件,从而分析出有问题的对象和有问题的代码出现在哪一行
//DUMP
//-Xms 设置初始化内存分配大小 1/64
//-Xmx 设置最大分配内存 ,默认 1/4
// -XX:PrintGCDetails 打印GC垃圾回收信息
//-XX:+HeapDumpOnOutOfMemoryError dump出一个文件,来分析出错的代码行和大的对象
//-Xms1m -Xmx8m -XX:+HeapDumpOnOutOfMemoryError
-XX:+HeapDumpOnOutOfMemoryError dump出一个文件,来分析出错的代码行和大的对象
Dump后面加上报的错误,就可以给这个错误dump出一个文件,从而分析!!
1、GC:垃圾回收 ,作用区域在堆
轻GC、FullGC(重GC,也叫全局GC)
2、GC算法—引用计数法
对每一个引用的对象标记引用次数,引用次数为0的就垃圾清除。但是这种方法弊端很多,本身标记次数也消耗内存,计数器本身也会有消耗。JVM一般不采用这个算法,不高校,用的比较少
再清除后,防止内存碎片的产生,再次扫描,向一段移动存活的对象。这样就没有了内存碎片,是比较高效的。
再优化:
先标记清除几次(多几次产生的内存碎片多),再进行一次标记压缩。这样节约移动成本—一次移动解决更多的内存碎片
参考官方,百度,博客,视频等途径进行学习
学习新东西是常态!!