虚拟机栈
组成
1. 局部变量表:一个数字数组,用于存储方法参数和定义在方法体内的局部变量。数据类型包括基本数据类型和引用数据类型。所需容量大小在编译器就确定下来了。
变量的分类:按照数据类型分:1. 基本数据类型
2. 引用数据类型
按照在类中的位置分:1. 成员变量:在使用前,都经历过默认初始化赋值。
1)类变量:在类加载的连接阶段赋默认值,在初始化阶段赋初始值。
2)实例变量:随着对象的创建,在堆空间中分配实例变量空间,并赋默认值
2. 局部变量:在使用前必须要显示赋值
2. 操作数栈:用栈来操作数,将数放入栈中再放入局部变量表中,需要使用时再入栈出栈。如果方法有返回值,会将返回值放入栈中。
3. 动态链接:指向运行时常量池的方法引用
4. 方法返回地址:调用的方法栈帧出栈之后将返回值压入调用者的操作数栈中以供使用。
作用
每个线程运行的时候,需要的内存空间。一个线程对应一个栈
每个栈内由多个栈桢组成。
栈桢
每个方法运行时需要的内存(参数,局部变量,返回地址),每个方法执行时都要提前分配内存地址
当调用一个方法的时候,就会给方法划分一个栈桢并压入栈内,当方法执行完毕,就会让这个栈桢出栈释放内存。
一个方法中可能会划分多个栈桢:如调用该方法时内部调用了其他方法。但每个线程只能有一个活动栈桢,对应着当前正在执行的那个方法。
主方法也对应着一个栈桢
问题
1. 垃圾回收是否设计栈内存?
否,栈内存即每次方法执行的栈桢内存,每个栈桢内存在方法调用结束后都会被释放
2. 栈内存分配越大越好吗?
在运行java代码的时候,可以设置栈内存的大小,默认大小为1024kb
栈内存越大,只是能进行多次的方法递归调用,反而会让线程数变少
3. 方法内的局部变量是否线程安全?
看一个变量是否线程安全就要看它是否是对一个线程共享的还是独占的。
如果是一个基本变量是线程安全的,一个线程对应一个栈,不同的线程在调用同一个方法的时候,会有不同的栈。线程内每一次方法调用都会产生一个新的栈桢。
但如果是staic的变量,是针对多个的线程共享的,则会涉及到线程安全问题
如果方法内局部变量没有逃离方法的作用范围,它是线程安全的。反之,如果它作为行参传入或return了这个变量,则它可能会被其他线程操作,为线程所共享。
栈内存溢出
stackoverflowError
什么情况会导致栈内存溢出?
在栈内存不能动态扩展的情况下:
1. 栈桢过多导致栈内存溢出(如果方法的递归没有设置一个递归退出的条件)
2. 栈桢过大也会导致栈内存溢出(这种情况很少,栈桢直接超出栈内存)
如何设置栈的内存大小
线程运行诊断
nohup:后台运行Java程序
案例1: cpu占用过多
使用命令
1. 用top定位到那个进程对cpu占用过高
2. ps H -eo pid,tid,%cpu | grep 进程id,定位到哪个线程对cpu占用过高
3. 使用jdk提供的 jstack 进程id:定位该进程到所有的java输出的线程,将linux下的线程id转换为16进制即可找到对应的java线程