内存结构
- Virtual Machine -java 程序运行环境
好处:
- 一次编写,到处运行。
- 自动内存管理机制,垃圾回收功能。
- 数组下标越界检查
- 多态
比较:
jdk = jvm + 基础类库(io…) + 编译工具(javac,javap…)
作用
- 理解底层实现原理(自动拆装箱,动态代理…)
- 中高级程序员的必备技能 (内存溢出,响应时间缓慢)
常见的jvm(jvm是一种规范)hotspot,oracle jdk edition
jvm 的组成部分
1,classloader
2,jvm内存结构: 方法区,堆 ,虚拟机,程序计数器,本地方法栈
3,执行引擎 :解释器,即时编译器,垃圾回收
1.程序计数器
-
program Counter register(程序寄存器)
-
作用: 记住下一条jvm指令的执行地址
- 特点: 线程私有
- 不会存在内存溢出
2.虚拟机栈
2.1 定义
- xss
先进后出 - 线程运行需要的内存空间
- 组成:栈帧:(参数局部变量,放回地址) 每个方法运行时需要的内存 栈帧 一个栈帧就对应一个方法的调用
- 每个线程只能由一个活动栈帧,对应这当前正在执行的那个方法
2.2 栈内存溢出栈
- 内存越大,线程越小
方法内的局部变量是否线程安全
1,如果方法内局部变量没有逃离方法的作用访问,他是线程安全的
2,如果是局部变量引用了对象,并逃离方法的作用范围,需要考虑线程安全问题
1,栈内存溢出 (-Xss 256k)可以设置栈内存大小
- 什么情况下会导致栈内存溢出
1,方法递归调用,没有合理的设置返回条件
2, 栈帧过大,也会导致栈内存溢出
2.3 线程运行诊断
1, cpu居高不下
- Linux top 定位进程
- ps H -eo pid,tid,%cpu greap 进程id 打印所有线程pid,tid,cpu占用率 —>定位线程
- jstack pid 根据线程id找到有问题的线程
- 将10进制线程编号转换为16进制的线程编号
2,程序运行很长时间没有得到结果
1,有可能是线程死锁
2,jstack pid 查看 deadlock
3.本地方法栈
- (native method stacks)
- 运行那些不是Java编写的代码,Java不能直接与操作系统交互方法
4.Heap 堆
4.1定义
- jvm参数 -Xmx
- 通过new关键字,创建的对象使用堆内存
- 线程共享,堆中对象都需要考虑线程安全问题
- 有垃圾回收机制
4.2堆内存溢出问题
- 堆内存空间不足,比如list一直在添加数据,list里面的数据不会被垃圾回收,如果list的添加还没有结束,堆空间就满了。
4.3堆内存诊断
- jps工具
- 查看当前系统中有哪些java进程
- jmap工具(只能查询某一个时刻)
- 查看堆内存使用情况 jmap -heap pid
- jconsole工具
- 图形界面,多功能检测工具,可以连续监测
案例:垃圾回收后,heap居高不下
- jvisualvm 可视化的虚拟机