1.jvm
1.JAva Virtual Machiine - java程序的运行环境(java二进制节码的运行环境)
好处:
一次编写,到处运行
自动内存管理,垃圾回收功能
数组下标越界时越界检查
通过虚方法表实现多态
2.jvm、jre、jdk的区分
jvm虚拟机是为了屏蔽java代码和底层操作系统之间的差异,就是一个运行环境
jre是虚拟机加了些基础类库,比如java.lang包,集合、io、线程、日期
jdk在jre基础上加了些编译工具,比如javac、javap
javase在jdk基础上加些ide工具,比如idea、ecplise
javaee在javase基础上加应用服务器,比如tomcat
2.程序计数器
1.java代码流程
java代码形成 .Java文件,编译成.class文件,读取二进制字节码,传递给解释器,形成机器码,交予cpu运算。
2.程序计数器
作用:记住下一条jvm指令的执行地址。
解释器直接找程序计数器。
解释器是一种计算机程序,它将每个字节码转换成机器代码。
3.特点
-
线程私有的。(切换线程时,可以记住当前线程走到那一步了)
-
不会存在内存溢出
3.虚拟机栈
1.定义
Java Virtual Machine Stacks(Java 虚拟机栈)
-
每个线程运行时所需要的内存,称为虚拟机栈
-
每个栈由多个栈帧(Frame)组成,对应者每次方法调用时所占用的内存
-
每个线程只能由一个活动栈帧,对应着当前正在执行的哪个方法。活动栈帧也就是顶部的栈帧
-
和程序计数器一样,也是线程私有
-
每个方法在执行的同时都会创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。
问题辨析:
-
垃圾回收机制是否涉及栈内存
-
不涉及,栈内存中的活动栈帧完成之后就会弹出,不需要垃圾回收
-
-
栈内存分配越大越好嘛?
-
不是,栈内存分配越多,线程越少,减慢运行速度
-
-
方法内的局部变量是否线程安全
-
如果是普通的局部变量,是线程私有,安全
-
如果是static静态变量,是线程共享的,不安全
-
关键在于,方法内的局部变量是否会逃离方法的作用范围,没有就是安全的。若当作参数传入或者参数返回,就是不安全的。
-
2.栈内存溢出
栈帧太多导致栈内存溢出(递归方法没有break条件)
栈帧太大导致栈内存溢出(idea设置vm option)
3.线程运行诊断
案例一、cpu占用过多
定位:
-
用top定位哪个进程对cpu的占用过高
-
ps H -eo pid,tid,%cpu | grep 进程id (用ps命令进一步定位是哪个线程引起的cpu占用过高)
-
jstack 进程id
-
jstack可以根据线程id找到有问题的线程,进一步定位到问题代码的源码行号
-
4.本地方法栈
定义:不是java的代码,Java代码具有一定的限制性,不能直接与操作系统打交道,需要用c或者c++编写的方法来和底层操作系统的API打交道。本地方法栈就是用来放本地方法的,虚拟机使用到的Native方法服务。