In addition to Weibo, there is also WeChat
Please pay attention
WeChat public account
Shulou
2025-01-17 Update From: SLTechnology News&Howtos shulou NAV: SLTechnology News&Howtos > Development >
Share
Shulou(Shulou.com)06/02 Report--
This article introduces you how to understand and solve the Java memory leak, the content is very detailed, interested friends can refer to, I hope it can be helpful to you.
Java memory management mechanism
In C++ language, if a piece of memory needs to be allocated dynamically, the programmer is responsible for the whole life cycle of the memory. From applying for distribution, to use, and then to the release of *. This process is very flexible, but it is so tedious that programmers can easily forget to release memory due to carelessness, resulting in memory leaks. The Java language optimizes memory management, which is the garbage collection mechanism. Almost all memory objects in Java are allocated on heap memory (except for basic data types), and then GC (garbage collection) is responsible for automatically reclaiming memory that is no longer in use.
The above is the basic situation of Java memory management mechanism. But if we only understand this, we will still encounter the problem of memory leakage in the actual project development. Some people may wonder, since Java's garbage collection mechanism can automatically reclaim memory, how can there be a memory leak? To solve this problem, we need to know when GC reclaims memory objects and what memory objects will be considered by GC to be "no longer in use".
Access to memory objects in Java is referenced. In the Java code, we maintain a reference variable of the memory object, and through the value of this reference variable, we can access the memory object space in the corresponding memory address. In Java programs, this reference variable itself can be stored either in heap memory or in code stack memory (the same as the basic data type). The GC thread starts tracking from reference variables in the code stack to determine which memory is being used. If the GC thread cannot trace a piece of heap memory in this way, GC assumes that the memory will no longer be used (because it is no longer accessible in the code).
Through this directed graph memory management approach, when a memory object loses all references, GC can recycle it. Conversely, if the object still has a reference, it will not be reclaimed by the GC, even if the Java virtual machine throws the OutOfMemoryError.
Java memory leak
Generally speaking, there are two kinds of memory leaks. In one case, if the memory allocated in the heap is not released, all the ways to access this memory are deleted (such as pointer re-assignment); the other is that the memory and its access method (reference) are still retained when the memory object is clearly no longer needed. The situation of * has been well solved in Java due to the introduction of garbage collection mechanism. Therefore, the memory leak in Java mainly refers to the second case.
Maybe the concept is too abstract. You can take a look at this example:
Vector v = new Vector (10); for (int I = 1; I)
< 100 ; i ++ ){ Object o = new Object(); v.add(o); o = null ; } 在这个例子中,代码栈中存在Vector 对象的引用 v 和 Object 对象的引用 o 。在 For 循环中,我们不断的生成新的对象,然后将其添加到 Vector 对象中,之后将 o 引用置空。问题是当 o 引用被置空后,如果发生 GC ,我们创建的 Object 对象是否能够被 GC 回收呢?答案是否定的。因为, GC 在跟踪代码栈中的引用时,会发现 v 引用,而继续往下跟踪,就会发现 v 引用指向的内存空间中又存在指向 Object 对象的引用。也就是说尽管 o 引用已经被置空,但是 Object 对象仍然存在其他的引用,是可以被访问到的,所以 GC 无法将其释放掉。如果在此循环之后, Object 对象对程序已经没有任何作用,那么我们就认为此 Java 程序发生了内存泄漏。 尽管对于C/C++ 中的内存泄露情况来说, Java 内存泄露导致的破坏性小,除了少数情况会出现程序崩溃的情况外,大多数情况下程序仍然能正常运行。但是,在移动设备对于内存和 CPU 都有较严格的限制的情况下, Java 的内存溢出会导致程序效率低下、占用大量不需要的内存等问题。这将导致整个机器性能变差,严重的也会引起抛出 OutOfMemoryError ,导致程序崩溃。 一般情况下内存泄漏的避免 在不涉及复杂数据结构的一般情况下,Java 的内存泄露表现为一个内存对象的生命周期超出了程序需要它的时间长度。我们有时也将其称为"对象游离"。 例如: public class FileSearch{ private byte [] content; private File mFile; public FileSearch(File file){ mFile = file; } public boolean hasString(String str){ int size = getFileSize(mFile); content = new byte [size]; loadFile(mFile, content); String s = new String(content); return s.contains(str); } } 在这段代码中,FileSearch 类中有一个函数 hasString ,用来判断文档中是否含有指定的字符串。流程是先将mFile 加载到内存中,然后进行判断。但是,这里的问题是,将 content 声明为了实例变量,而不是本地变量。于是,在此函数返回之后,内存中仍然存在整个文件的数据。而很明显,这些数据我们后续是不再需要的,这就造成了内存的无故浪费。 要避免这种情况下的内存泄露,要求我们以C/C++ 的内存管理思维来管理自己分配的内存。***,是在声明对象引用之前,明确内存对象的有效作用域。在一个函数内有效的内存对象,应该声明为 local 变量,与类实例生命周期相同的要声明为实例变量……以此类推。第二,在内存对象不再需要时,记得手动将其引用置空。 复杂数据结构中的内存泄露问题 在实际的项目中,我们经常用到一些较为复杂的数据结构用于缓存程序运行过程中需要的数据信息。有时,由于数据结构过于复杂,或者我们存在一些特殊的需求(例如,在内存允许的情况下,尽可能多的缓存信息来提高程序的运行速度等情况),我们很难对数据结构中数据的生命周期作出明确的界定。这个时候,我们可以使用Java 中一种特殊的机制来达到防止内存泄露的目的。 之前我们介绍过,Java 的 GC 机制是建立在跟踪内存的引用机制上的。而在此之前,我们所使用的引用都只是定义一个" Object o; "这样形式的。事实上,这只是 Java 引用机制中的一种默认情况,除此之外,还有其他的一些引用方式。通过使用这些特殊的引用机制,配合 GC 机制,就可以达到一些我们需要的效果。 Java中的几种引用方式 Java中有几种不同的引用方式,它们分别是:强引用、软引用、弱引用和虚引用。下面,我们首先详细地了解下这几种引用方式的意义。 强引用 在此之前我们介绍的内容中所使用的引用 都是强引用,这是使用最普遍的引用。如果一个对象具有强引用,那就类似于必不可少的生活用品,垃圾回收器绝不会回收它。当内存空 间不足,Java 虚拟机宁愿抛出 OutOfMemoryError 错误,使程序异常终止,也不会靠随意回收具有强引用的对象来解决内存不足问题。 软引用(SoftReference ) SoftReference 类的一个典型用途就是用于内存敏感的高速缓存。 SoftReference 的原理是:在保持对对象的引用时保证在 JVM 报告内存不足情况之前将清除所有的软引用。关键之处在于,垃圾收集器在运行时可能会(也可能不会)释放软可及对象。对象是否被释放取决于垃圾收集器的算法 以及垃圾收集器运行时可用的内存数量。 弱引用(WeakReference ) WeakReference 类的一个典型用途就是规范化映射( canonicalized mapping )。另外,对于那些生存期相对较长而且重新创建的开销也不高的对象来说,弱引用也比较有用。关键之处在于,垃圾收集器运行时如果碰到了弱可及对象,将释放 WeakReference 引用的对象。然而,请注意,垃圾收集器可能要运行多次才能找到并释放弱可及对象。 虚引用(PhantomReference ) PhantomReference 类只能用于跟踪对被引用对象即将进行的收集。同样,它还能用于执行 pre-mortem 清除操作。 PhantomReference 必须与 ReferenceQueue 类一起使用。需要 ReferenceQueue 是因为它能够充当通知机制。当垃圾收集器确定了某个对象是虚可及对象时, PhantomReference 对象就被放在它的 ReferenceQueue 上。将 PhantomReference 对象放在 ReferenceQueue 上也就是一个通知,表明 PhantomReference 对象引用的对象已经结束,可供收集了。这使您能够刚好在对象占用的内存被回收之前采取行动。 Reference与 ReferenceQueue 的配合使用。 GC、 Reference 与 ReferenceQueue 的交互 A、 GC无法删除存在强引用的对象的内存。 B、 GC发现一个只有软引用的对象内存,那么: ① SoftReference对象的 referent 域被设置为 null ,从而使该对象不再引用 heap 对象。 ② SoftReference引用过的 heap 对象被声明为 finalizable 。 ③ 当 heap 对象的 finalize() 方法被运行而且该对象占用的内存被释放, SoftReference 对象就被添加到它的 ReferenceQueue (如果后者存在的话)。 C、 GC发现一个只有弱引用的对象内存,那么: ① WeakReference对象的 referent 域被设置为 null , 从而使该对象不再引用heap 对象。 ② WeakReference引用过的 heap 对象被声明为 finalizable 。 ③ 当heap 对象的 finalize() 方法被运行而且该对象占用的内存被释放时, WeakReference 对象就被添加到它的 ReferenceQueue (如果后者存在的话)。 D、 GC发现一个只有虚引用的对象内存,那么: ① PhantomReference引用过的 heap 对象被声明为 finalizable 。 ② PhantomReference在堆对象被释放之前就被添加到它的 ReferenceQueue 。 值得注意的地方有以下几点: 1、 GC 在一般情况下不会发现软引用的内存对象,只有在内存明显不足的时候才会发现并释放软引用对象的内存。 2、 GC 对弱引用的发现和释放也不是立即的,有时需要重复几次 GC ,才会发现并释放弱引用的内存对象。 3、软引用和弱引用在添加到 ReferenceQueue 的时候,其指向真实内存的引用已经被置为空了,相关的内存也已经被释放掉了。而虚引用在添加到 ReferenceQueue 的时候,内存还没有释放,仍然可以对其进行访问。 代码示例 通过以上的介绍,相信您对Java 的引用机制以及几种引用方式的异同已经有了一定了解。光是概念,可能过于抽象,下面我们通过一个例子来演示如何在代码中使用 Reference 机制。 String str = new String( " hello " ); // ① ReferenceQueue < String >Rq = new ReferenceQueue
< String >(); / / ② WeakReference
< String >Wf = new WeakReference
< String >(str, rq); / / ③ str = null; / / ④ cancels the strong reference to the "hello" object String str1 = wf.get (); / / ⑤ if the "hello" object is not recycled, str1 references the "hello" object / / if the "hello" object is not recycled, rq.poll () returns null Reference
Welcome to subscribe "Shulou Technology Information " to get latest news, interesting things and hot topics in the IT industry, and controls the hottest and latest Internet news, technology news and IT industry trends.
Views: 0
*The comments in the above article only represent the author's personal views and do not represent the views and positions of this website. If you have more insights, please feel free to contribute and share.
Continue with the installation of the previous hadoop.First, install zookooper1. Decompress zookoope
"Every 5-10 years, there's a rare product, a really special, very unusual product that's the most un
© 2024 shulou.com SLNews company. All rights reserved.