Java中的内存泄露、内存溢出与栈溢出

news/2024/12/2 20:29:18/

内存泄露、内存溢出与栈溢出

  • 1、概述
  • 2、内存泄漏、内存溢出和栈溢出
    • 2.1、内存泄漏
    • 2.2、内存溢出
    • 2.3、栈溢出
  • 2、总结

1、概述

大家好,我是欧阳方超。本次就Java中几个相似而又不同的概念做一下介绍。内存泄漏、内存溢出和栈溢出都是与内存相关的问题,但它们之间有所不同。

2、内存泄漏、内存溢出和栈溢出

我们经常会遇到内存泄漏、内存溢出和栈溢出等问题,这些问题都与内存的使用有关。

2.1、内存泄漏

内存泄漏(memory leak)指的是程序在使用内存时,未将不再使用的内存释放,导致内存不断占用而无法再次使用。内存泄漏的原因可能是程序中存在未释放的资源、对象引用未被清理、内存分配过多等。当程序中存在大量的内存泄漏时,可能会导致系统性能下降、程序崩溃等问题。
解决方法:及时释放不再使用的资源、对象引用,避免内存分配过多,使用内存检测工具进行检测和修复。

2.2、内存溢出

内存溢出(out of memory)指的是程序在运行时,申请的内存空间超过了系统可用的内存空间。内存溢出的原因可能是程序中存在大量的内存泄漏、对象过多、内存分配过多等。当程序中出现内存溢出时,可能会导致程序崩溃、系统异常等问题。
解决方法:及时释放不再使用的资源、对象引用,避免内存分配过多,使用内存检测工具进行检测和修复,增加系统内存等。

注意,一次内存泄露危害可以忽略,但内存泄露堆积后果很严重,如果内存泄漏持续发生而又得不到控制的话,无论多少内存,迟早会被耗尽。memory leak会最终会导致out of memory!

既然内存泄漏与内存溢出有关系,我们就用一个例子来验证一下“内存泄漏会导致内存溢出”这一现象,

import java.util.ArrayList;public class Test {public static void main(String[] args) {ArrayList<Integer> integerArrayList = new ArrayList<>();long i = 1;while (true) {integerArrayList.add(1);System.out.println(i + "times");i++;}}
}

在上面的示例代码中,我们创建了一个包含整数的列表,并在一个无限循环中不断向其中添加整数。由于没有终止循环的条件,程序将不断向列表中添加整数,直到内存溢出为止。当内存不再足够容纳更多的整数时,程序将崩溃,并且会抛出一个OutOfMemoryError异常。将上面的程序运行起来,很快就会发生内存溢出的错误(循环进行了70091070次后发生了内存溢出):

70091068times
70091069times
70091070times
Exception in thread "main" java.lang.OutOfMemoryError: Java heap spaceat java.util.Arrays.copyOf(Arrays.java:3210)at java.util.Arrays.copyOf(Arrays.java:3181)at java.util.ArrayList.grow(ArrayList.java:261)at java.util.ArrayList.ensureExplicitCapacity(ArrayList.java:235)at java.util.ArrayList.ensureCapacityInternal(ArrayList.java:227)at java.util.ArrayList.add(ArrayList.java:458)

2.3、栈溢出

栈溢出指的是程序在执行过程中,栈空间超过了系统所能支持的范围。栈溢出的原因可能是程序中存在过多的递归调用、方法嵌套过深等。当程序中出现栈溢出时,可能会导致程序崩溃、系统异常等问题。
解决方法:优化递归算法,减少方法的嵌套层数,增加系统栈空间等。
下面是一个使用递归计算阶乘的例子:

public class Test {public static void main(String[] args) {int num = 10; // 需要计算的阶乘long factorial = calcFactorial(num); // 调用递归函数计算阶乘System.out.println(num + "的阶乘是:" + factorial);}public static long calcFactorial(int n) {if (n == 1) { // 递归结束条件return 1;} else {return n * calcFactorial(n - 1); // 递归调用计算阶乘}}
}

在上述代码中,我们定义了一个静态方法calcFactorial,用于递归计算阶乘。如果n等于1,说明阶乘已经计算完成,直接返回1;否则,递归调用calcFactorial(n - 1)计算n - 1的阶乘,然后将结果乘以n,得到n的阶乘。最终,我们在main方法中调用calcFactorial方法计算阶乘,并输出计算结果。

需要注意的是,递归计算阶乘的方法在计算大数阶乘时可能会超出栈的深度限制,导致栈溢出异常。比如我们将上面程序中num的值改为一万,再次运行时立马会发生栈溢出的问题:

Exception in thread "main" java.lang.StackOverflowErrorat Test.calcFactorial(Test.java:12)at Test.calcFactorial(Test.java:15)at Test.calcFactorial(Test.java:15)at Test.calcFactorial(Test.java:15)

在递归时,栈需要保存函数的调用信息,保存的过多的话会导致栈内存不够用,进而发生栈溢出,我们可以将递归实现的阶乘计算优化成循环实现:

public class Factorial {public static void main(String[] args) {int num = 5; // 需要计算的阶乘long factorial = 1; // 阶乘初始值为1for (int i = 1; i <= num; i++) {factorial *= i; // 计算阶乘}System.out.println(num + "的阶乘是:" + factorial);}
}

在上述代码中,我们定义了一个变量num,表示需要计算的阶乘。然后,我们定义了一个long类型的变量factorial,用于存储阶乘的值,初始值为1。接着,我们使用for循环从1到num,每次将当前的i乘到factorial中,最终得到num的阶乘。最后,我们输出计算结果。

需要注意的是,阶乘可能会非常大,超出了long类型的范围,因此在实际应用中需要使用大数类进行计算,以避免计算结果溢出。另外,阶乘的计算也可以使用递归实现,但需要注意递归深度的控制,以避免栈溢出。

2、总结

内存泄漏、内存溢出和栈溢出都是程序中常见的内存问题,它们都会导致程序运行的异常和不稳定。为了避免这些问题,我们需要在编程中注意及时释放不再使用的资源和对象引用,避免内存分配过多,优化算法和代码结构等。同时,我们还可以使用内存检测工具进行检测和修复,在程序开发和测试过程中,及时发现和解决问题,保证程序运行的稳定性和可靠性。
我是欧阳方超,把事情做好了自然就有兴趣了,如果你喜欢我的文章,欢迎点赞、转发、评论加关注。我们下次见。


http://www.ppmy.cn/news/59177.html

相关文章

RabbitMQ 发布订阅模式,routing路由模式,topic模式

发布订阅模式 一个消息可以由多个消费者消费同一个消息 消费者1和2同时消费了该消息 举例 public static void main(String[] args) throws IOException, TimeoutException {//1 创建连接工厂ConnectionFactory connectionFactorynew ConnectionFactory();//2 设置rabbitmq …

pg 提升子链接源码分析

pg 上拉子链接源码分析 此分析基于 version 13.8&#xff0c; 在pg 中把子查询分为两类&#xff0c;一类是from 中的子查询&#xff1b; 一类是where 中的子查询&#xff0c;叫做子链接&#xff08;sublink&#xff09;。 下面用sublink&#xff08;子链接&#xff09;指代wh…

2023年第二十届五一数学建模竞赛题目 B题超详细思路

详细思路以及发布视频版&#xff0c;大家可以去观看&#xff0c;这里是对应的文字版&#xff0c;内容相差不多。 B题&#xff1a;快递需求分析问题 B题的问题难度不大&#xff0c;难点就在于后几问的模型求解。问题多、模型多、冗杂&#xff0c;就是B题的特点。 难度 A>B…

Flutter 组件抽取:日期(DatePicker)、时间(TimePicker)弹窗选择器【仿照】

简介 仿照《Flutter 仿ios自定义一个DatePicker》实行的日期弹窗选择器&#xff08;DatePicker&#xff09;、时间弹窗选择器&#xff08;TimePicker&#xff09; 效果 范例 class _TestPageState extends State<TestPage> {overridevoid initState() {super.initStat…

每日一题141——字符串中的最大奇数

给你一个字符串 num &#xff0c;表示一个大整数。请你在字符串 num 的所有 非空子字符串 中找出 值最大的奇数 &#xff0c;并以字符串形式返回。如果不存在奇数&#xff0c;则返回一个空字符串 "" 。 子字符串 是字符串中的一个连续的字符序列。 示例 1&#xff1…

MySQL索引概述

MySQL索引概述 当表中的数据量到达几十万甚至上百万的时候&#xff0c;SQL查询所花费的时间会很长&#xff0c;导致业务超时出错&#xff0c;此时就需要用索引来加速SQL查询。 由于索引也是需要存储成索引文件的&#xff0c;因此对索引的使用也会涉及磁盘I/O操作。如果索引创建…

探索三维世界【2】:Three.js 的 Texture 纹理

缤纷三维世界大揭秘&#xff1a;探索 Three.js 的 Texture 纹理 1、Texture纹理2、TextureLoader 纹理加载器2.1、创建纹理加载器2.2、纹理属性设置2.3、设置纹理渲染2.4、打光 3、完整代码与展示 1、Texture纹理 Texture 是 three.js 中的“纹理”概念。纹理是指将一张图像映…

Addictive Multiplicative in NN

特征交叉是特征工程中的重要环节&#xff0c;在以表格型&#xff08;或结构化&#xff09;数据为输入的建模中起到了很关键的作用。 特征交互的作用&#xff0c;一是尽可能挖掘对目标有效的模式、特征&#xff0c;二是具有较好的可解释性&#xff0c;三是能够将对数据的洞见引…