Java对象一定分配在堆上吗?

简介: 本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。

1. 引入

首先回答标题中的问题:Java对象一定会被分配到堆上吗?答案是:不一定

Java中创建的对象一般会分配到堆上,当堆空间不足时,就会触发GC进行垃圾回收,但是GC次数太多会影响程序的性能。

在编译期间,编译器会对代码做很多优化,为了减少内存分配压力,JVM提供了一项重要优化技术:逃逸分析。逃逸分析得出的结论为后续优化措施提供依据。

2. 什么是逃逸分析

逃逸分析Escape Analysis):JVM提供的一种优化技术,用于分析对象会不会发生逃逸。

Q:如何理解逃逸?

逃逸可以理解为会不会在作用域范围外被调用。如:一个方法内定义的变量,会不会在这个方法外被使用,如果否,则认为未逃逸;如果是:则认为会发生逃逸,这就是方法逃逸

根据上述的理解,可以分为不同的逃逸方式。对象的逃逸程度从高到低

  • 线程逃逸:一个对象在方法内被定义后,可能被外部线程访问,如:赋值给可以在其他线程中访问的实例变量;
  • 方法逃逸:一个对象在方法内被定义后,可能会被外部方法引用;
  • 不逃逸:仅在作用域范围内使用。

根据逃逸分析的结果来决定优化策略

3. 优化策略

3.1 栈上分配(Stack Allocations

  • 将对象分配到上,对象占用的内存空间可以随着栈帧出栈(即方法的结束)而销毁,这样垃圾收集的压力会下降很多。
  • 整个过程通过判断对象,来决定其是否必须要存在堆上,如果不需要的话,则可以被分配到栈上,栈随着线程的消逝而消逝,这样能够减少了GC的频率,从而提高性能。
  • 栈上分配支持方法逃逸,不支持线程逃逸

【例如】

Java

代码解读

复制代码

public void test(){
    Student s = new Student();
    s.setName("张三");
    s.setAge(22);
    System.out.println(s.getAge());
}

逃逸分析后得出的结论为:不逃逸,对象s只作用于该方法内,不会被其他方法/线程引用,所以该对象可以分配到上。

Java

代码解读

复制代码

public Student test(){
    Student s = new Student();
    s.setName("张三");
    s.setAge(22);
    return s;
}

该方法的返回值为Student对象,逃逸分析后,得出的结论是:对象s可能会被其他方法/线程引用,所以该对象只能分配到上。

3.2 标量替换(Scalar Replacement

Q:什么是标量?

标量可以理解为:不可拆解的数据,如:intlong等数值类型。

与之相对的概念为聚合量(Aggregate:即可以拆解的数据,如:Java中的对象

标量替换就是将Java对象拆散,根据程序访问的情况,将其用到的成员变量恢复到原始类型来访问。

这样做的好处:对象的成员变量在栈上分配和读写;为后续进一步优化创造条件。可以将标量替换看作栈上分配的一种特例,实现更加简单,但对逃逸的要求更高,不允许对象逃逸出方法范围内

【例如】

Java

代码解读

复制代码

//标量替换前的代码
public static void main(String[] args) {
   User user =  new User("张三",33);
   System.out.println("姓名:" + s.getName() + " 年龄:" + s.getAge());
}

// 标量替换后的代码
public static void main(String[] args) {
   String name = "张三";
   int age = 33;
   System.out.println("姓名:" + name + " 年龄:" + age);
}

3.3 同步消除(Synchronization Elimination

线程同步本身是一个相对耗时的过程,如果逃逸分析能够确定一个变量不会逃逸出线程,无法被其他线程访问,那么这个变量的读写肯定就不会有竞争,对这个变量实施的同步措施也就可以安全消除掉。

如果JVM通过逃逸分析,发现一个对象只能从一个线程访问到,访问这个对象时可以不加同步锁,如:如果程序中使用了synchronized锁,JVM会将synchronized锁消除。

4. Java对象内存分配流程


转载来源:https://juejin.cn/post/7376861773839237157

目录
打赏
0
2
4
0
122
分享
相关文章
Java中判断一个对象是否是空内容
在 Java 中,不同类型的对象其“空内容”的定义和判断方式各异。对于基本数据类型的包装类,空指对象引用为 null;字符串的空包括 null、长度为 0 或仅含空白字符,可通过 length() 和 trim() 判断;集合类通过 isEmpty() 方法检查是否无元素;数组的空则指引用为 null 或长度为 0。
Java快速入门之类、对象、方法
本文简要介绍了Java快速入门中的类、对象和方法。首先,解释了类和对象的概念,类是对象的抽象,对象是类的具体实例。接着,阐述了类的定义和组成,包括属性和行为,并展示了如何创建和使用对象。然后,讨论了成员变量与局部变量的区别,强调了封装的重要性,通过`private`关键字隐藏数据并提供`get/set`方法访问。最后,介绍了构造方法的定义和重载,以及标准类的制作规范,帮助初学者理解如何构建完整的Java类。
|
2月前
|
Object取值转java对象
通过本文的介绍,我们了解了几种将 `Object`类型转换为Java对象的方法,包括强制类型转换、使用 `instanceof`检查类型和泛型方法等。此外,还探讨了在集合、反射和序列化等常见场景中的应用。掌握这些方法和技巧,有助于编写更健壮和类型安全的Java代码。
50 17
8G的容器Java堆才4G怎么就OOM了?
本文记录最近一例Java应用OOM问题的排查过程,希望可以给遇到类似问题的同学提供参考。
|
2月前
|
java代码优化:判断内聚到实体对象中和构造上下文对象传递参数
通过两个常见的java后端实例场景探讨代码优化,代码不是优化出来的,而是设计出来的,我们永远不可能有专门的时间去做代码优化,优化和设计在平时
37 15
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
103 5
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
111 0
关于java堆内存溢出的几种情况(转)
【情况一】:   java.lang.OutOfMemoryError: Java heap space:这种是java堆内存不够,一个原因是真不够,另一个原因是程序中有死循环;   如果是java堆内存不够的话,可以通过调整JVM下面的配置来解决:   -Xms3062m   -Xmx3062m   【情况二】   java.lang.OutOfMemoryError: GC overhead limit exceeded   【解释】:JDK6新增错误类型,当GC为释放很小空间占用大量时间时抛出;一般是因为堆太小,导致异常的原因,没有足够的内存。
1383 0
|
3天前
|
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
100 60
【Java并发】【线程池】带你从0-1入门线程池
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
75 14

热门文章

最新文章

AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等