Tomcat开启了APR模式,而APR模式会使用堆外内存,关于堆内存可从如下链接了解一下:http://blog.csdn.net/zhouhl_cn/article/details/6573213。

完整异常信息如下:

  1. Exception in thread "http-apr-8080-Acceptor-0" java.lang.OutOfMemoryError: Direct buffer memory
  2. at java.nio.Bits.reserveMemory(Bits.java:694)
  3. at java.nio.DirectByteBuffer.<init>(DirectByteBuffer.java:123)
  4. at java.nio.ByteBuffer.allocateDirect(ByteBuffer.java:311)
  5. at org.apache.tomcat.util.net.SocketBufferHandler.<init>(SocketBufferHandler.java:38)
  6. at org.apache.tomcat.util.net.AprEndpoint$AprSocketWrapper.<init>(AprEndpoint.java:2341)
  7. at org.apache.tomcat.util.net.AprEndpoint.setSocketOptions(AprEndpoint.java:907)
  8. at org.apache.tomcat.util.net.AprEndpoint.setSocketOptions(AprEndpoint.java:78)
  9. at org.apache.tomcat.util.net.Acceptor.run(Acceptor.java:115)
  10. at java.lang.Thread.run(Thread.java:748)

1. baidu/google半天,没找到太明确的解决办法。

2.网上提到,如果性能已经足够了,没有必要使用堆外内存,那么是否可以想办法禁用Tomcat使用堆外内存?查阅Tomcat官方文档的配置说明,没有找到。

3.下载相应版本的Tomcat 源码,根据异常信息找一下代码执行的流程,结果发现,代码中是写死的,一定会使用堆外内存,具体代码如下:

  1. // org.apache.tomcat.util.net.AprEndpoint.java:2341行
  2.  
  3. public AprSocketWrapper(Long socket, AprEndpoint endpoint) {
  4. super(socket, endpoint);
  5.  
  6. // TODO Make the socketWriteBuffer size configurable and align the
  7. // SSL and app buffer size settings with NIO & NIO2.
  8. if (endpoint.isSSLEnabled()) {
  9. sslOutputBuffer = ByteBuffer.allocateDirect(SSL_OUTPUT_BUFFER_SIZE);
  10. sslOutputBuffer.position(SSL_OUTPUT_BUFFER_SIZE);
  11. } else {
  12. sslOutputBuffer = null;
  13. }
  14.  
  15. socketBufferHandler = new SocketBufferHandler(6 * 1500, 6 * 1500, true);
  16. }
  1. // org.apache.tomcat.util.net.SocketBufferHandler.java:38
  2.  
  3. public SocketBufferHandler(int readBufferSize, int writeBufferSize,
  4. boolean direct) {
  5. this.direct = direct;
  6. if (direct) {// 使用堆外内存
  7. readBuffer = ByteBuffer.allocateDirect(readBufferSize);
  8. writeBuffer = ByteBuffer.allocateDirect(writeBufferSize);
  9. } else {
  10. readBuffer = ByteBuffer.allocate(readBufferSize);
  11. writeBuffer = ByteBuffer.allocate(writeBufferSize);
  12. }
  13. }

4.既然没有办法禁用,是否可以调大?找到了如下参数,可以将此参数调大,此参数默认是64M。启动时命令行加上此参数即可。

  1. -XX:MaxDirectMemorySize=512m

5.网上查询资料说,Full GC时,会将堆外内存回收,但是如果进程内存限制设的比较大,很长时间不会触发Full GC,就会导致这部分堆外内存持续不释放。所以,将进程内存调小一点,使用如下参数。

  1. -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,取消对日志文件的压缩,并且,由原来的按天分割,改为按小时分割,并按天建立文件夹存储,文件如下:

  1. <configuration debug="off" monitorInterval="1800">
  2. <Properties>
  3. <Property name="log-path">/data/logs/</Property>
  4. </Properties>
  5. <Appenders>
  6. <Console name="Console" target="SYSTEM_OUT">
  7. <PatternLayout pattern="%d{HH:mm:ss.SSS} %-5level %class{36}.%M()/%L - %msg%xEx%n"/>
  8. </Console>
  9. <RollingFile name="DailyRollingFile" fileName="${log-path}/cell.log" filePattern="${log-path}/$${date:yyyy-MM-dd}/cell-%d{HH}.log">
  10. <Policies>
  11. <TimeBasedTriggeringPolicy interval="1" modulate="true"/>
  12. </Policies>
  13. <DefaultRolloverStrategy>
  14. <Delete basePath="${log-path}" maxDepth="2">
  15. <IfFileName glob="*/*.log" />
  16. <IfLastModified age="30d" />
  17. </Delete>
  18. </DefaultRolloverStrategy>
  19. <PatternLayout pattern="[%p] %t %d{HH:mm:ss,SSS} %c{1}:%L(%M) %m%n"/>
  20. </RollingFile>
  21. </Appenders>
  22. <Loggers>
  23. <root level="INFO">
  24. <appender-ref ref="DailyRollingFile"/>
  25. </root>
  26. </Loggers>
  27. </configuration>

11.持续观察多天后,问题未再出现,至此问题解决。

Tomcat 9内存溢出:"http-apr-8080-Acceptor-0" java.lang.OutOfMemoryError: Direct buffer memory的更多相关文章

  1. myeclipse执行tomcat报错Exception in thread &quot;main&quot; java.lang.OutOfMemoryError: PermGen space

    将myeclipse所配置的tomcat的jdk进行设置:-Xms512m -Xmx512m -XX:MaxNewSize=512m -XX:MaxPermSize=512m,例如以下图:

  2. Java内存溢出java.lang.OutOfMemoryError: PermGen space

    今天把以前的一个项目部署在tomcat,启动没问题.因为用到了webservice,当调用webservice中的方法时一直报内存溢出异常 Exception in thread "http ...

  3. Tomcat内存溢出解决java.lang.OutOfMemoryError: PermGen space

    背景:把两个项目同时部署在tomcat,启动快好的时候,报java.lang.OutOfMemoryError: PermGen space 原因:因为两个项目的jar包太多,JVM把里面的class ...

  4. Tomcat – java.lang.OutOfMemoryError: PermGen space Cause and Solution

    Read more: http://javarevisited.blogspot.com/2012/01/tomcat-javalangoutofmemoryerror-permgen.html#ix ...

  5. eclipse:Tomcat设置jvm,解决java.lang.OutOfMemoryError: Java heap space 堆内存溢出

    eclipse 有启动参数里设置jvm大小,因为eclipse运行时自己也需要jvm,所以eclipse.ini里设置的jvm大小不是具体某个程序运行时所用jvm的大小,这和具体程序运行的jvm大小无 ...

  6. Tomcat报内存溢出

    1.错误描述 严重:Exception occurred during processing request:null          java.lang.reflect.InvocationTar ...

  7. window下tomcat的内存溢出问题

    打开注册表:https://jingyan.baidu.com/article/49ad8bce09d6085835d8fa63.html Tomcat 内存溢出对应解决方式 Windows平台,使用 ...

  8. eclipse启动tomcat出现内存溢出错误 java.lang.OutOfMemoryError: PermGen space

    发布工程后,启动tomcat出现如下内存溢出错误: java.lang.OutOfMemoryError: PermGen space ... java.lang.OutOfMemoryError: ...

  9. Tomcat之——内存溢出设置JAVA_OPTS

    答案1设置Tomcat启动的初始内存其初始空间(即-Xms)是物理内存的1/64,最大空间(-Xmx)是物理内存的1/4.可以利用JVM提供的-Xmn -Xms -Xmx等选项可进行设置三.实例,以下 ...

随机推荐

  1. GitLab服务器IP地址修改

    gitlab安装介绍:https://about.gitlab.com/downloads/#centos7 刚搭建好的gitlab在GitLab上新建一个项目test_gitlab,刚开始仓库地址是 ...

  2. HDU 6201 transaction transaction transaction (树形DP)

    题意:给定一棵树,每个点有一个点权,每条边也是,找一条路径,问你 T-S-sum,T表示路径的终点的权值,S表示路径始点的权值,sum表示从S到T的边权和. 析:把这一条路径拆开来看,那么就是必然是从 ...

  3. 使用GeoServer+OpenLayers发布和调用WMTS、Vector Tile矢量切片服务 | Publishing and Calling WMTS, Vector Tile Service Using GeoServer + OpenLayers

    Web GIS系列: 1.搭建简易Web GIS网站:使用GeoServer+PostgreSQL+PostGIS+OpenLayers3 2.使用GeoServer+QGIS发布WMTS服务 3.使 ...

  4. 设计模式20:Memento 备忘录模式(行为型模式)

    Memento 备忘录模式(行为型模式) 对象状态的回溯 对象状态的变化无端,如何回溯.恢复对象在某个点的状态? 动机(Motivation) 在软件构建过程中,某些对象的状态在转换过程中,可能由于某 ...

  5. Android小技巧

    一.android:clipChildren属性 效果图 看到这个图时你可以先想想如果是你,你怎么实现这个效果.马上想到用RelativeLayout?NO,NO,NO,,, 实现代码 <?xm ...

  6. CodeForces 408E Curious Array(组合数学+差分)

    You've got an array consisting of n integers: a[1], a[2], ..., a[n]. Moreover, there are m queries, ...

  7. PreTranslateMessage()函数捕获键盘按键消息

    01. PreTranslateMessage函数,常用于屏蔽MFC对话框中默认的Enter和ESC消息 函数原型:BOOL PreTranslateMessage(MSG* pMsg) 用法举例: ...

  8. atan2()如何转换为角度

    atan2()如何转换为角度 Math.atan2()函数返回点(x,y)和原点(0,0)之间直线的倾斜角.那么如何计算任意两点间直线的倾斜角呢?只需要将两点x,y坐标分别相减得到一个新的点(x2-x ...

  9. ZKEACMS 自定义表单的使用

    ZKEACMS Core 2.2 已经发布了,其中主要添加了自定义表单的功能.使用自定义表单的功能,您可以在几分钟内就创建一个表单,并用它来收集一些信息.导出收集的信息,就可以做一些统计分析. 创建表 ...

  10. CentOS下Docker与.netcore(五)之 三剑客之一Docker-swarm集群

    CentOS下Docker与.netcore(一) 之 安装 CentOS下Docker与.netcore(二) 之 Dockerfile CentOS下Docker与.netcore(三)之 三剑客 ...