- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试向我的程序中添加一个 javaeditor 以在运行时扩展该程序。一切正常,除非广泛使用该程序(我模拟了 1000-10000 次编译器执行)。内存使用率越来越高,看起来有内存泄漏。
在我的程序中,类被加载,构造函数被执行,类被卸载(没有剩余的实例并且类加载器变得无效,因为我将指针设置为空)。我用 JConsole 分析了这个过程,当垃圾收集器被执行时,这些类被卸载了。
我做了一个 heapdum 在内存分析器中打开它,问题似乎在 java.net.FactoryURLClassLoader 内部(在 com.sun.tools.javac.util.List 对象中)。由于 (com.sun.tools.javac) 是 JDK 的一部分而不是 JRE 中的一部分,并且 SystemToolClassLoader 是一个 FactoryURLClassLoader 对象,因此我会在某处找到泄漏。当我第一次执行编译器时,SystemToolClassLoader 中加载的类数从 1 增加到 521,但之后保持不变。
所以我不知道泄漏在哪里,有没有办法重置 SystemToolClassLoader?我怎样才能更准确地定位泄漏点。
编辑:好的,我发现它也出现在一个非常非常简单的示例中。所以它似乎是编译的一部分,我不需要加载类或实例化它:
import java.io.File;
import java.io.IOException;
import java.util.Arrays;
import javax.tools.JavaCompiler;
import javax.tools.JavaFileObject;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;
public class Example {
public static void main(String[] args)
{
for (int i =0; i<10000;i++){
try {
System.out.println(i);
compile();
} catch (InstantiationException | IllegalAccessException
| ClassNotFoundException | IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
public static void compile() throws IOException, InstantiationException, IllegalAccessException, ClassNotFoundException
{
File source = new File( "src\\Example.java" ); // This File
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
StandardJavaFileManager fileManager = compiler.getStandardFileManager( null, null, null );
Iterable<? extends JavaFileObject> units;
units = fileManager.getJavaFileObjectsFromFiles( Arrays.asList( source ) );
compiler.getTask( null, fileManager, null, null, null, units ).call();
fileManager.close();
}
}
最佳答案
起初,我认为这肯定是内存泄漏;然而,它与 SoftReference
的工作方式直接相关。
Oracle 的 JVM 仅在堆完全用完时才尝试收集软引用。似乎无法以编程方式强制收集软引用。
为了找出问题所在,我在“无限”堆上使用了三个转储:
明显的(双倍)实例数量增加:Names
(500 -> 1k), SharedNameTable
(500->1k), SharedNameTable$NameImpl
(数十万)和 [LSharedNameTable$NameImpl
(500->1k)。
在使用 EMA 分析后,很明显 SharedNameTable
有一个对 com.sun.tools.javac.util.List
的静态引用,这显然是 SoftReference
s 曾经创建的每一个 SharedNameTable
(所以一个对应于您在运行时编译的每个源文件)。所有 $NameImpl
都是您的源文件被分割成的标记。显然,所有 token 都从未从堆中释放并无休止地累积……或者确实如此?
我决定测试一下是否真的如此。了解软引用与弱引用的区别后,我决定使用小堆 (-Xms32m -Xmx32m)。通过这种方式,JVM 将被迫释放 SharedNameTable
或因 OutOfMemoryError
而失败。结果不言自明:
-Xmx512m -Xms512m
Total memory: 477233152
Free memory: 331507232
Used memory: 138.97506713867188 MB
Loaded scripts: 500
Total memory: 489816064
Free memory: 203307408
Used memory: 273.23594665527344 MB
Loaded scripts: 1000
The classloader/component "java.net.FactoryURLClassLoader @ 0x8a8a748" occupies 279.709.192 (98,37%) bytes.
-Xmx32m -Xms32m
Total memory: 29687808
Free memory: 25017112
Used memory: 4.454322814941406 MB
Loaded scripts: 500
Total memory: 29884416
Free memory: 24702728
Used memory: 4.941642761230469 MB
Loaded scripts: 1000
One instance of "com.sun.tools.javac.file.ZipFileIndex" loaded by "java.net.FactoryURLClassLoader @ 0x8aa4cc8" occupies 2.230.736 (47,16%) bytes. The instance is referenced by *.*.script.ScriptFileManager @ 0x8ac8230.
(这只是一个指向 JDK 库的链接。)
脚本:
public class Avenger
{
public Avenger()
{
JavaClassScriptCache.doNotCollect(this);
}
public static void main(String[] args)
{
// this method is called after compiling
new Avenger();
}
}
不收集:
private static final int TO_LOAD = 1000;
private static final List<Object> _active = new ArrayList<Object>(TO_LOAD);
public static void doNotCollect(Object o)
{
_active.add(o);
}
System.out.println("Loaded scripts: " + _active.size());
关于reflection - 在运行时使用 JDK 编译器时的内存泄漏,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14617340/
IntentReceiver 正在泄漏 由于 onDetachedFromWindow 在某些情况下未被调用。 @Override protected void onDetachedFromWind
好吧,我很难追踪这个内存泄漏。运行此脚本时,我没有看到任何内存泄漏,但我的 objectalloc 正在攀升。 Instruments 指向 CGBitmapContextCreateImage >
我编写了一个测试代码来检查如何使用 Instrument(Leaks)。我创建了一个单一 View 应用程序,单击按钮后我加载了一个像这样的新 View ... - (IBAction)btn_clk
我正在使用这个简单的代码并观察单调增加的内存使用量。我正在使用这个小模块将内容转储到磁盘。我观察到它发生在 unicode 字符串上而不是整数上,我做错了什么吗? 当我这样做时: >>> from u
我有以下泄漏的代码。 Instruments 表示,泄漏的是 rssParser 对象。我“刷新”了 XML 提要,它运行了该 block 并且发生了泄漏...... 文件.h @interface
我在我编写的以下代码片段中发现了内存泄漏 NSFileManager *fileManager=[[NSFileManager alloc] init]; fileList=[[fileManager
因此,我正在开发HTML5 / javascript rts游戏。观察一直有几种声音在播放。因此,对我来说,是一段时间后声音听起来像是“崩溃”,并且此浏览器选项卡上的所有声音都停止了工作。我只能通过重
下面是我正在使用的一段代码及其输出。 my $handle; my $enterCount = Devel::Leak::NoteSV($handle); print "$date entry $en
在这篇关于 go-routines 泄漏的帖子之后,https://www.ardanlabs.com/blog/2018/11/goroutine-leaks-the-forgotten-sende
我想知道为什么在执行 ./a.out 后随机得到以下结果。有什么想法我做错了吗?谢谢 http://img710.imageshack.us/img710/8708/trasht.png 最佳答案 正
我正在 Swift 中开发一个应用程序,在呈现捕获我放在一起的二维码的自定义 ViewController 后,我注意到出现了巨大的内存跳跃。 该代码本质上基于以下示例:http://www.appc
下面是我的 javascript 代码片段。它没有按预期运行,请帮我解决这个问题。 function getCurrentLocation() { console.log("insi
我们在生产环境中部署了 3 个代理 Kafka 0.10.1.0。有些应用程序嵌入了 Kafka Producer,它们将应用程序日志发送到某个主题。该主题有 10 个分区,复制因子为 3。 我们观察
我正在使用仪器来检测一些泄漏,但有一些泄漏我无法解决; NSMutableString *textedetails = [[NSMutableString alloc] init];
如果我使用性能工具测试我的代码 - 泄漏,它没有检测到任何泄漏。这是否意味着代码没有泄漏任何内存? 我有一个越狱的 iPhone,我可以监控可用内存。如果有人知道,那就是 SBSettings。我测试
我在从 AddressBook 中获取图像时遇到了很大的问题,下面我粘贴了我的代码。此 imageData 从未被释放,在我的 Allocations Instruments 上它看起来总是在内存中它
- (NSMutableArray *)getArrayValue:(NSArray *)array{ NSMutableArray *valueArray = [NSMutableArra
Instruments 工具说这是一个泄漏,有什么想法吗? 我在 for 循环结束时释放变量对象 在上述方法的开头,这就是我设置变量对象的方式,即自动释放; NSMutableArray *varia
我正在跟踪我的 iOS 应用程序的内存泄漏,我有一个奇怪的泄漏导致我的应用程序崩溃......负责的框架是:CGImageMergeXMPPropsWhithLegacyProps。在某些时候,我的应
我正在尝试使用 NSOperationQueue 在后台线程中执行一个方法,如下所示: NSOperationQueue *queue = [NSOperationQueue new]; NS
我是一名优秀的程序员,十分优秀!