繁体   English   中英

关于用java导致过多的打开文件?

[英]About causing too many open files with java?

当审查同事的代码时,发现下面的代码

    BufferedReader br = new BufferedReader(new FileReader(PATH + fileName));
    //...

只是读取一个文件并将这些行作为一行连接,但我没有找到任何密切的代码,所以我认为它应该导致资源泄漏,最后导致too many open files error ,所以为了证明这一点,我写了一个测试

for (int i = 0; i < 7168; i++) { // ulimit -n ==> 7168
    BufferedReader br = new BufferedReader(new FileReader("src/main/resources/privateKey/foo.pem"));
    System.out.println(br.readLine());
}
System.in.read();

很奇怪,一切都好,不会抛出预期的异常。

并在命令行中检查实际打开的文件

➜  ~ lsof -p 16276 | grep 'foo.pem' | wc -l
    2538

为什么只有2538,而不是7168?

那有什么不对? 如何导致too many open files error


正如@GhostCat建议的那样,更改7168 - > Integer.MAX_VALUE,这次造成的

java.io.FileNotFoundException: src/main/resources/privateKey/foo.pem (Too many open files in system)
at java.io.FileInputStream.open0(Native Method)
at java.io.FileInputStream.open(FileInputStream.java:195)

当我是27436 ,并在这种情况下检查命令行中真正打开的文件是

➜  ~ lsof | grep foo.pem | wc -l
    7275

但在哪里留下文件(27346 - 7275)? 为什么ulimit号不起作用?

我假设垃圾收集器正在运行,发现许多无法访问的BufferedReader对象并收集它们。 这导致底层流对象被最终化......这将关闭它们。

要使此代码中断,请将BufferedReader对象添加到列表中,以便它们保持可访问状态。


这就是我认为将7168更改为MAXINT的原因。

当JVM启动时,它将使用相对较小的堆。 GC期间发生的一件事是JVM决定是否需要调整堆大小。 所以这是可能发生的事情:

  • JVM以一个太小而无法容纳7168个打开文件+ BufferedReader对象的堆开始。 (请记住,后者中的每一个都可能有一个预先分配的缓冲区!)

  • 你开始打开文件。

  • 在大约N = 7168 - 2538时,堆填满了所有BufferedReader对象+ FileInputStream对象+来自JVM启动/预热的各种碎片。

  • GC运行,并导致(可能)所有BufferedReader对象被收集/完成/关闭。

  • 然后GC决定它需要扩展堆。 您现在有足够的堆空间用于比ulimit允许的更多打开的BufferedReader对象。

  • 您恢复打开文件...然后点击打开文件限制。

这是一种可能的模式。


如果您真的想对此进行调查,我建议您打开GC日志记录,看看是否可以将lsof报告的FD数量与GC运行相关联。

(您可以尝试在每次打开之间添加sleep调用,以便更容易获得lsof测量。但这可能会以其他方式改变JVM行为......)

我没有一个确切的解释,但有一些额外的想法:“我们”必须明白,事情并不像表面上看起来那么简单。

关键是:有几层抽象可以发挥作用。 有JVM和JIT; 然后是那些操作系统。

含义:给定这些抽象,期望每个新的BufferReader直接导致另一个文件句柄是太天真了。 如果Linux内核在这里开始,我不会感到惊讶; 并且只是“告诉”JVM“是的,我打开了那个文件;然后为你阅读;这是它的内容”。 但在“现实”中,Linux内核理解这个文件没有被触及,并且自上次读取请求后没有改变...

  1. jvm隐式更新ulimit值

     String [] cmdArray = {"sh","-c","ulimit -n"}; Process p = Runtime.getRuntime().exec(cmdArray); BufferedReader in = new BufferedReader(new InputStreamReader(p.getInputStream())); System.out.println(in.readLine()); //it is 10240 not 7168 
  2. @Stephen C是对的,GC参与其中。

我创建一个MyBufferedReader扩展BufferedRead并覆盖finalize方法

@Override
protected void finalize() throws Throwable {
    System.out.printf("Thread: %s finalize it and total: %d %n",Thread.currentThread().getName(),count.getAndAdd(1));
}

得到以下信息

Thread: Finalizer finalize it and total: 9410 

并在命令行中

➜  ~ lsof -p 5309 | grep 'taicredit_private_key_pkcs8' | wc -l
     830

9410 + 830 = 10240

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM