@ZEAL Blog·厉
We stand alone,
TOGETHER.

Too many open files 错误

Posted by zeal on 2005-06-17 13:53 , 2010 characters |  + 1 - 0   English
转载请保留本行原始出处声明信息 : http://www.zeali.net/entry/93 MaDe1nZEAL
打开的文件过多,一般来说是由于应用程序对资源使用不当造成,比如没有及时关闭Socket或数据库连接等。但也可能应用确实需要打开比较多的文件句柄,而系统本身的设置限制了这一数量。

异常 1 
java.net.SocketException: Too many open files

    at java.net.PlainSocketImpl.accept(Compiled Code)
    at java.net.ServerSocket.implAccept(Compiled Code)
    at java.net.ServerSocket.accept(Compiled Code)
    at weblogic.t3.srvr.ListenThread.run(Compiled Code)
 

异常 2
java.io.IOException:打开的文件过多

    at java.lang.UNIXProcess.forkAndExec(Native Method)
    at java.lang.UNIXProcess.(UNIXProcess.java:54)
    at java.lang.UNIXProcess.forkAndExec(Native Method)
    at java.lang.UNIXProcess.(UNIXProcess.java:54)
    at java.lang.Runtime.execInternal(Native Method)
    at java.lang.Runtime.exec(Runtime.java:551)
    at java.lang.Runtime.exec(Runtime.java:477)
    at java.lang.Runtime.exec(Runtime.java:443)

...

第一个异常在错误影响到基础 TCP 协议时抛出,而第二个异常则在错误影响到 I/O 操作时抛出。

文件打开数过多最坏的情况可以使系统崩溃,到时候只能是重起服务器了。

原因:   

    操作系统的中打开文件的最大句柄数受限所致,常常发生在很多个并发用户访问服务器的时候.因为为了执行每个用户的应用服务器都要加载很多文件(new一个socket就需要一个文件句柄),这就会导致打开文件的句柄的缺乏.

解决: 

  • 尽量把类打成jar包,因为一个jar包只消耗一个文件句柄,如果不打包,一个类就消耗一个文件句柄.
  • java的垃圾回收不能关闭网络连接打开的文件句柄,如果没有执行close()(例如:java.net.Socket.close())则文件句柄将一直存在,而不能被关闭.你也可以考虑设置socket的最大打开数来控制这个问题.
  •  对操作系统做相关的设置,增加最大文件句柄数量。
    1. Linux
      在Linux内核2.4.x中需要修改源代码,然后重新编译内核才生效。编辑Linux内核源代码中的 include/linux/fs.h文件,将 NR_FILE 由8192改为65536,将NR_RESERVED_FILES 由10 改为 128。编辑fs/inode.c 文件将MAX_INODE 由16384改为262144。或者编辑 /etc/sysctl.conf  文件增加两行 fs.file-max = 65536 和 fs.inode-max = 262144 。一般情况下,系统最大打开文件数比较合理的设置为每4M物理内存256,比如256M.可以用lsof -p <pid of process>看打开的文件句柄数.
    2. Windows
      最大文件句柄是16,384,你在任务管理器的性能这一项中可以看到当前打开的句柄数.

http://www.bea.com.cn/support_pattern/Too_Many_Open_Files_Pattern.html

Last Modified on 2007-05-24 18:00
2 条评论:
- jmsyoung () 于 2005-06-29 16:58
我在长时间使用Java的Runtime.exec("ps -ef | grep XXX")就会发生第2类的异常,请问一下你认为是什么原因。
- zeal (link) 于 2005-06-29 17:27
用java的Runtime.exec会出现很多莫名其妙的问题,尤其在多线程的情况之下。我之前作一个项目,里面要用到网络通讯的功能,一开始因为已经有了perl的版本,所以就直接用Runtime来调用外部脚本作这部分任务,后来上线使用发现程序跑了一段时间之后就会把服务器都搞崩掉。最后还是重新写了java的版本来解决问题。所以现在打死我也不会去使用Runtime的外部程序调用方法,特别是当这样的调用会被频繁循环的触发的时候。
Since 2005.04.27  梦想 就像鸡蛋 要么孵化 要么臭掉RSS Feed (Entries) | Back To Home | @ZEAL | 沪ICP备05024379号