Tomcat 9内存溢出:"http-apr-8080-Acceptor-0" java.lang.OutOfMemoryError: Direct buffer memory
Tomcat开启了APR模式,而APR模式会使用堆外内存,关于堆内存可从如下链接了解一下:http://blog.csdn.net/zhouhl_cn/article/details/6573213。
完整异常信息如下:
Exception in thread "http-apr-8080-Acceptor-0" java.lang.OutOfMemoryError: Direct buffer memory
at java.nio.Bits.reserveMemory(Bits.java:694)
at java.nio.DirectByteBuffer.<init>(DirectByteBuffer.java:123)
at java.nio.ByteBuffer.allocateDirect(ByteBuffer.java:311)
at org.apache.tomcat.util.net.SocketBufferHandler.<init>(SocketBufferHandler.java:38)
at org.apache.tomcat.util.net.AprEndpoint$AprSocketWrapper.<init>(AprEndpoint.java:2341)
at org.apache.tomcat.util.net.AprEndpoint.setSocketOptions(AprEndpoint.java:907)
at org.apache.tomcat.util.net.AprEndpoint.setSocketOptions(AprEndpoint.java:78)
at org.apache.tomcat.util.net.Acceptor.run(Acceptor.java:115)
at java.lang.Thread.run(Thread.java:748)
1. baidu/google半天,没找到太明确的解决办法。
2.网上提到,如果性能已经足够了,没有必要使用堆外内存,那么是否可以想办法禁用Tomcat使用堆外内存?查阅Tomcat官方文档的配置说明,没有找到。
3.下载相应版本的Tomcat 源码,根据异常信息找一下代码执行的流程,结果发现,代码中是写死的,一定会使用堆外内存,具体代码如下:
// org.apache.tomcat.util.net.AprEndpoint.java:2341行
public AprSocketWrapper(Long socket, AprEndpoint endpoint) {
super(socket, endpoint);
// TODO Make the socketWriteBuffer size configurable and align the
// SSL and app buffer size settings with NIO & NIO2.
if (endpoint.isSSLEnabled()) {
sslOutputBuffer = ByteBuffer.allocateDirect(SSL_OUTPUT_BUFFER_SIZE);
sslOutputBuffer.position(SSL_OUTPUT_BUFFER_SIZE);
} else {
sslOutputBuffer = null;
}
socketBufferHandler = new SocketBufferHandler(6 * 1500, 6 * 1500, true);
}
// org.apache.tomcat.util.net.SocketBufferHandler.java:38 public SocketBufferHandler(int readBufferSize, int writeBufferSize,
boolean direct) {
this.direct = direct;
if (direct) {// 使用堆外内存
readBuffer = ByteBuffer.allocateDirect(readBufferSize);
writeBuffer = ByteBuffer.allocateDirect(writeBufferSize);
} else {
readBuffer = ByteBuffer.allocate(readBufferSize);
writeBuffer = ByteBuffer.allocate(writeBufferSize);
}
}
4.既然没有办法禁用,是否可以调大?找到了如下参数,可以将此参数调大,此参数默认是64M。启动时命令行加上此参数即可。
-XX:MaxDirectMemorySize=512m
5.网上查询资料说,Full GC时,会将堆外内存回收,但是如果进程内存限制设的比较大,很长时间不会触发Full GC,就会导致这部分堆外内存持续不释放。所以,将进程内存调小一点,使用如下参数。
-Xms2000m -Xmx2000m
6.观察一下,是否还会出现此问题。
7.经观察后,发现问题还会出现,另外,又查了文档后发现-XX:MaxDirectMemorySize参数的默认值并不是64M,而是下面的规则:
- 如果有-XX:MaxDirectMemorySize参数显示指定了数值,则使用此参数指定的;
- 如果没有-XX:MaxDirectMemorySize,则看有没有-Xmx,如果有,则使用-Xmx的值;
- 如果-XX:MaxDirectMemorySize和-Xmx都没有,才会使用64M;
我们的程序中-Xmx设置的是2000M,所以理论上Direct Memory足够用了。
8.分析问题出现的时间点,每次必是00:00:00,排查代码,找到这个时间点进行的可疑的操作有两个,其中一个是定时任务(从业务上讲,这个定时任务已经没用了,但仍然在跑),另一个是log4j2的日志分割、压缩;
9.由于定时任务已经废弃,所以直接去掉此定时任务,再观察,发现问题出现的频率降低了,但并未完全消失,说明该定时任务仅是导致问题的部分原因;
10.修改log4j2.xml,取消对日志文件的压缩,并且,由原来的按天分割,改为按小时分割,并按天建立文件夹存储,文件如下:
<configuration debug="off" monitorInterval="1800">
<Properties>
<Property name="log-path">/data/logs/</Property>
</Properties>
<Appenders>
<Console name="Console" target="SYSTEM_OUT">
<PatternLayout pattern="%d{HH:mm:ss.SSS} %-5level %class{36}.%M()/%L - %msg%xEx%n"/>
</Console>
<RollingFile name="DailyRollingFile" fileName="${log-path}/cell.log" filePattern="${log-path}/$${date:yyyy-MM-dd}/cell-%d{HH}.log">
<Policies>
<TimeBasedTriggeringPolicy interval="1" modulate="true"/>
</Policies>
<DefaultRolloverStrategy>
<Delete basePath="${log-path}" maxDepth="2">
<IfFileName glob="*/*.log" />
<IfLastModified age="30d" />
</Delete>
</DefaultRolloverStrategy>
<PatternLayout pattern="[%p] %t %d{HH:mm:ss,SSS} %c{1}:%L(%M) %m%n"/>
</RollingFile>
</Appenders>
<Loggers>
<root level="INFO">
<appender-ref ref="DailyRollingFile"/>
</root>
</Loggers>
</configuration>
11.持续观察多天后,问题未再出现,至此问题解决。
Tomcat 9内存溢出:"http-apr-8080-Acceptor-0" java.lang.OutOfMemoryError: Direct buffer memory的更多相关文章
- myeclipse执行tomcat报错Exception in thread "main" java.lang.OutOfMemoryError: PermGen space
将myeclipse所配置的tomcat的jdk进行设置:-Xms512m -Xmx512m -XX:MaxNewSize=512m -XX:MaxPermSize=512m,例如以下图:
- Java内存溢出java.lang.OutOfMemoryError: PermGen space
今天把以前的一个项目部署在tomcat,启动没问题.因为用到了webservice,当调用webservice中的方法时一直报内存溢出异常 Exception in thread "http ...
- Tomcat内存溢出解决java.lang.OutOfMemoryError: PermGen space
背景:把两个项目同时部署在tomcat,启动快好的时候,报java.lang.OutOfMemoryError: PermGen space 原因:因为两个项目的jar包太多,JVM把里面的class ...
- Tomcat – java.lang.OutOfMemoryError: PermGen space Cause and Solution
Read more: http://javarevisited.blogspot.com/2012/01/tomcat-javalangoutofmemoryerror-permgen.html#ix ...
- eclipse:Tomcat设置jvm,解决java.lang.OutOfMemoryError: Java heap space 堆内存溢出
eclipse 有启动参数里设置jvm大小,因为eclipse运行时自己也需要jvm,所以eclipse.ini里设置的jvm大小不是具体某个程序运行时所用jvm的大小,这和具体程序运行的jvm大小无 ...
- Tomcat报内存溢出
1.错误描述 严重:Exception occurred during processing request:null java.lang.reflect.InvocationTar ...
- window下tomcat的内存溢出问题
打开注册表:https://jingyan.baidu.com/article/49ad8bce09d6085835d8fa63.html Tomcat 内存溢出对应解决方式 Windows平台,使用 ...
- eclipse启动tomcat出现内存溢出错误 java.lang.OutOfMemoryError: PermGen space
发布工程后,启动tomcat出现如下内存溢出错误: java.lang.OutOfMemoryError: PermGen space ... java.lang.OutOfMemoryError: ...
- Tomcat之——内存溢出设置JAVA_OPTS
答案1设置Tomcat启动的初始内存其初始空间(即-Xms)是物理内存的1/64,最大空间(-Xmx)是物理内存的1/4.可以利用JVM提供的-Xmn -Xms -Xmx等选项可进行设置三.实例,以下 ...
随机推荐
- 37 有n个人围成一圈,顺序排号,从第一个人开始报数(从1到3报数),凡报到3的人退出圈子,问最后留下的是原来第几号那位.
题目:有n个人围成一圈,顺序排号,从第一个人开始报数(从1到3报数),凡报到3的人退出圈子,问最后留下的是原来第几号那位. public class _037NumberOff { public st ...
- 大前端涉猎之前后端交互总结3:使用PHP进行表单数据删除与查询
1 首先:获取用户 id,根据id 删除指定的数据. 在链接完数据库之后,通过$_GET 超全局变量,获取删除的书内容 $id = $_GET['id']; 2 准备delete SQL语句: $sq ...
- javascript总结35:DOM之给a注册点击事件, 阻止a标签的默认行为
给a注册点击事件时,有默认行为,阻止默认行为的方式: retrun false <!DOCTYPE html> <html lang="zh-CN"> &l ...
- 两个SSH2间免密码登录
SSH2免密码登录OpenSSHhttp://blog.csdn.net/aquester/article/details/23836299 OpenSSH免密码登录SSH2http://blog.c ...
- ObjC正则表达式验证
试过ObjC的regkit这个框架. 也用过内置的正则表达式验证. 最后发现有个非常简单的方法就可以做到验证正则表达式.那就是NSPredicte这个类提供的方法. 这里有验证邮箱地址的正则为例: N ...
- 一个简单的编译tex的Makefile
tex编译成pdf通常要经过以下步骤:tex-->dvi-->ps-->pdf.如果修改了tex文件想看一下效果,就要把命令重新敲一遍.虽然就几行命令,反复敲还是很烦人的.最直接的办 ...
- 深入理解java虚拟机(八)类加载过程详解
类从被加载到虚拟机内存开始,到卸载出内存为止,它的整个生命周期包括:加载(Loading).验证(Verification).准备(Preparation).解析(Resolution).初始化(In ...
- 9、Semantic-UI之标题
9.1 定义基础的标题样式 在Semantic-UI中定义了5种标题样式,h1~h5. 示例:基础样式定义 <h1 class="ui header">一级标题&l ...
- python文件操作os模块
Python 统计某一文件夹下文件数量 使用python pathlib模块 from pathlib import Path dir_path = ' ' print(len(list(Path( ...
- ResorceGovernor--基础和Demo
资源调控器分为三部分:1:资源池,将资源CPU/MEMORY划分到不同的载体上2:负载组,承载负载并将负载映射到不同的资源池3: 分类函数,将不同回话映射到不同的负载组08提供两种预定义的系统资源池1 ...