我有一个在tomcat上运行的应用程序,有时我会遇到以下错误:
SEVERE: Socket accept failed
java.net.SocketException: Too many open files
at java.net.PlainSocketImpl.socketAccept(Native Method)
at java.net.AbstractPlainSocketImpl.accept(AbstractPlainSocketImpl.java:398)
at java.net.ServerSocket.implAccept(ServerSocket.java:522)
at java.net.ServerSocket.accept(ServerSocket.java:490)
at org.apache.tomcat.util.net.DefaultServerSocketFactory.acceptSocket(DefaultServerSocketFactory.java:60)
at org.apache.tomcat.util.net.JIoEndpoint$Acceptor.run(JIoEndpoint.java:216)
at java.lang.Thread.run(Thread.java:722)
...
SEVERE: Error processed default web.xml named conf/web.xml at /local/myApp/apache-tomcat/conf/web.xml
java.io.FileNotFoundException: /local/myApp/apache-tomcat/conf/web.xml (Too many open files)
at java.io.FileInputStream.open(Native Method)
at java.io.FileInputStream.<init>(FileInputStream.java:138)
at org.apache.catalina.startup.ContextConfig.getWebXmlSource(ContextConfig.java:1838)
at org.apache.catalina.startup.ContextConfig.getGlobalWebXmlSource(ContextConfig.java:1745)
at org.apache.catalina.startup.ContextConfig.getDefaultWebXmlFragment(ContextConfig.java:1418)
at org.apache.catalina.startup.ContextConfig.webConfig(ContextConfig.java:1253)
at org.apache.catalina.startup.ContextConfig.configureStart(ContextConfig.java:878)
at org.apache.catalina.startup.ContextConfig.lifecycleEvent(ContextConfig.java:369)
at org.apache.catalina.util.LifecycleSupport.fireLifecycleEvent(LifecycleSupport.java:119)
at org.apache.catalina.util.LifecycleBase.fireLifecycleEvent(LifecycleBase.java:90)
at org.apache.catalina.core.StandardContext.startInternal(StandardContext.java:5269)
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:150)
at org.apache.catalina.core.StandardContext.reload(StandardContext.java:3926)
at org.apache.catalina.loader.WebappLoader.backgroundProcess(WebappLoader.java:426)
at org.apache.catalina.core.ContainerBase.backgroundProcess(ContainerBase.java:1345)
at org.apache.catalina.core.ContainerBase$ContainerBackgroundProcessor.processChildren(ContainerBase.java:1530)
at org.apache.catalina.core.ContainerBase$ContainerBackgroundProcessor.processChildren(ContainerBase.java:1540)
at org.apache.catalina.core.ContainerBase$ContainerBackgroundProcessor.processChildren(ContainerBase.java:1540)
at org.apache.catalina.core.ContainerBase$ContainerBackgroundProcessor.run(ContainerBase.java:1519)
at java.lang.Thread.run(Thread.java:722)
我检查了打开文件的限制及其1024,但是当我用lsof检查应用程序的打开文件数量接近200时,为什么如果它没有达到限制就会发生这种情况? 我应该增加限额吗?有没有其他理由得到这个错误? 让服务再次正常运行的唯一方法是重新启动tomcat,还有另一种方法可以恢复正常吗?
提前致谢。
修改: 这是处理doPost方法的servlet,一开始我没有关闭每一个流,可能是那个吗?我添加了finally语句:
InputStream is = null;
DataInputStream dis = null;
OutputStream os = null;
DataOutputStream dos = null;
String paramName = "";
try {
os = response.getOutputStream();
is = request.getInputStream();
dis = new DataInputStream(is);
dos = new DataOutputStream(os);
.....
}catch (Throwable e) {
LOGGER.error(e.getMessage());
} finally {
if (dis != null) {
dis.close();
}
else if(is != null) {
is.close();
}
if (dos != null) {
dos.close();
}
else if( os != null) {
os.close();
}
}
EDIT2: 在做了一些测试之后,我意识到如果我首先关闭DataInputStream然后关闭InputStream,我会在通信的另一部分中输入一个数字(我不知道为什么)。我改变了关闭流的顺序,似乎一切都没问题。但我仍然有问题。有什么想法吗?
finally {
if(is != null) {
try {
is.close();
} catch (IOException e) {
LOGGER.error(e.getMessage());
}
}
if (dis != null) {
try {
dis.close();
} catch (IOException e) {
LOGGER.error(e.getMessage());
}
}
if(os != null) {
try {
os.close();
} catch (IOException e) {
LOGGER.error(e.getMessage());
}
}
if (dos != null) {
try {
dos.close();
} catch (IOException e) {
LOGGER.error(e.getMessage());
}
}
}
答案 0 :(得分:13)
执行以下操作以获取tomcat7的pid(例如1234)
ps aux |grep tomcat7
然后再做
cat /proc/1234/limits
阅读以下
Max open files 16384 16384 files
这些是Tomcat允许的最大打开文件数。要增加它,请按照以下说明进行操作
答案 1 :(得分:9)
通过将以下内容添加到/etc/security/limits.conf
来了解您可以更改已打开文件的限制可能很有用:
* soft nofile 2048 # Set the limit according to your needs
* hard nofile 2048
然后,您可以在shell上使用sysctl -p
重新加载配置。检查this article。
为了完整起见,您可以使用以下代码验证打开文件的当前限制:ulimit -n
答案 2 :(得分:6)
这将显示该进程的所有打开文件。
ls -l /proc/tomcatPID/fd
这将显示打开文件的数量。
ls -l /proc/tomcatPID/fd | wc -l
要增加打开文件限制更新/etc/Secirity/limits.conf
。
检查没有特定于tomcat的打开文件:
硬限制:su - tomcat -c 'ulimit -Hn' -s '/bin/bash'
软限制:su - tomcat -c 'ulimit -Sn' -s '/bin/bash'
您可以使用玉米作业运行以下脚本,以了解打开文件的详细信息。
=============================
#!/bin/bash
PID=$(ps -ef|grep tomcat6|grep -v grep |awk '{print $2}')
value=$(ls -l /proc/$PID/fd | wc -l)
echo `date`@$PID:$value >> /usr/local/filecount.txt
if [ $value -gt 2000 ];
then
printf "\n\n\n\n\n" >> /usr/local/files_report.txt
echo "-------------------------------`date`--Starting Session----------------------" >> /usr/local/files_report.txt
openfiles=$(ls -l /proc/$PID/fd | awk '{print NR,$11 "" >> "/usr/local/files_report.txt"}')
echo "--------------------`date`---Ending Session ------------------------------" >> /usr/local/files_report.txt
fi
=================
答案 3 :(得分:4)
@gaboroncancio发布的答案基本上是正确的,但他对如何使设置生效的建议并不完全正确。 sysctl -p
将重新加载/etc/sysctl.conf
,或者您传入的任何文件作为参数。但是,sysctl
命令无法识别/etc/security/limits.conf
。
要重新加载/etc/security/limits.conf
,您只需要注销并重新登录。
答案 4 :(得分:0)
如果此代码来自网络操作(套接字),我不确定Java XxxxxStrem与OS文件限制有1:1的关系(或者根本没有关系)。也许需要一些研究,异常消息有错误的文字?经常在软件中。
我的直觉说,我们不理解异常2,代码中没有任何东西(或者没有问题的配置)有关系。
当软件错误(泄漏)成为主要问题时扩展OS文件限制是不好的策略,因为你很好理解