- 加载到内存中的程序,被操作系统调度给CPU执行
- 程序即为进程,进程除了可执行代码,还包括内存栈空间和堆空间以及数据结构
- 程序创建数组的过程,就是在堆中申请一块内存空间,并将这块内存的首地址信息记录到栈中
- 堆空间是一块无序的空间,栈空间记录函数的局部变量、内存地址等
- 进程内会创建很多线程来处理并发用户请求,同样会有线程栈空间,堆空间依旧共享
- 当多个线程对堆中的数据进行修改时,就会出现线程安全问题,但只要不是高并发,正常情况下不需要考虑线程安全
- 加锁可以解决线程安全问题,但是也会让系统响应变慢,因为多个线程需要等待释放锁
- 当被阻塞的线程越来越多,超过了系统资源的极限,就会导致系统宕机,应用崩溃
- 为什么内存溢出一定是堆空间?
- 一般来说,绝大部分Java的内存溢出都属于堆溢出。原因是因为大量对象占据了堆空间,这些对象都持有强引用导致无法回收,当对象大小之和大于Xmx参数指定的堆空间时就会发生堆溢出。而且方法调用完成之后就会弹栈,所以通常不会出现问题
- 堆是用来存储对象实例和数组值,可以认为是java中所有通过new创建的对象的内存都在此分配 heap中,对象所占用的内存由gc回收。在32位操作系统上最大为2G,64位上则没有限制。其大小可以通过-Xms(最小内存small)和-Xmx(最大内存max)来控制。 当默认堆内存空间小于40%时,jvm会增大堆内存到-xmx的大小,通过指定 -xx:minHeapFreeRatio可以来自己设定比例,当空余堆大于70%
- 注意OutOfMemoryError和StackOverflowError的区别,前者内存溢出,后者内存泄漏。内存溢出是说程序需要申请的内存超过了JVM当前可以分配的最大内存,溢出。 内存泄漏是说期望被回收的内存对象没有被回收,泄漏。 内存泄漏持续发生,很可能引起内存溢出
- 以下代码解释了应用程序程序在堆空间、栈空间是如何分配的
public class Demo {
public static void main(String[] args) {
String str = "";
Student student = new Student();
str = student.getName();
}
}
public class Student {
private String name;
private Integer age;
public String getName() {
return name;
}
}