Spring Boot的项目在直接关闭Tomcat8后,Tomcat8报内存溢出的4个警告(目前解决两个)

直接停止Tomcat8后提示的警告异常如下图所示,共有四个警告	:

在这里插入图片描述

  •  从图中可以看出,共有四个警告异常:
     第一个:数据库链接未释放,被强制反注册
     第二个~第四个:应用关闭,但是线程未释放,导致报出了可能会有内存溢出的警告
    

解决方法:
第一个:参考了网上诸多大神的博客,添加如下监听器,可解决第一个数据库链接未释放问题,此解决方法在我的项目中可以起到作用;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Configuration;

import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import java.sql.Driver;
import java.sql.DriverManager;
import java.sql.SQLException;
import java.util.Enumeration;

@Configuration
public class UnregisterDataSourceListener implements ServletContextListener {

    Logger logger = LoggerFactory.getLogger(UnregisterDataSourceListener.class);

    @Override
    public void contextInitialized(ServletContextEvent sce) {
        // On Application Startup, please…

        // Usually I'll make a singleton in here, set up my pool, etc.
    }

    @Override
    public void contextDestroyed(ServletContextEvent sce) {
      // This manually deregisters JDBC driver, which prevents Tomcat 7 from complaining about memory leaks wrto this class
        Enumeration<Driver> drivers = DriverManager.getDrivers();
        while (drivers.hasMoreElements()) {
            Driver driver = drivers.nextElement();
            try {
                DriverManager.deregisterDriver(driver);
                logger.debug("deregistering jdbc driver:{}", driver);
            } catch (SQLException e) {
                e.printStackTrace();
                logger.debug("Error deregistering driver:{}", driver);
                logger.debug("Error message:{}", e.getMessage());
            }

        }
    }

}

第二个~第四个:
主要是由于我的项目中在启动时需要加载企业微信的配置,获取Token,故启动了一个线程获取,最后线程未被回收,修改了一个监听器,改为实现ServletContextListener后又减少了一个警告:

import com.supplychain.controller.wechat.TokenThread;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Configuration;

import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;

/**
 * token失效管理
 * @author Hangele
 * @since 2018-09-17
 */
@Configuration
public class InitAppIdAndSecret implements ServletContextListener {

    private static Logger logger = LoggerFactory.getLogger(InitAppIdAndSecret.class);

    public void contextInitialized(ServletContextEvent e) {
        logger.info("contextInitialized starting");
        // 启动定时获取access_token的线程,暂时先拿掉,调试微信时再开放。
        String os = System.getProperty("os.name");
        logger.info(" 当前操作系统环境为:{}", os);
        if (os.toLowerCase().startsWith("win")) {
            logger.info("windows环境,access_token的线程不启动");
        } else {
            logger.info("启动定时获取access_token的线程............");
            new Thread(new TokenThread()).start();
        }
        logger.info("contextInitialized ends");
    }

    public void contextDestroyed(ServletContextEvent e) {
        logger.info("contextDestroyed starting");
        System.gc();
        logger.info("contextDestroyed ends");
    }
}

在释放上下文时,调用了GC但是好像并没有起到作用,还是要另寻办法,先记录至此!

修改后,效果如下图所示:
在这里插入图片描述

暂时去掉了两个,线程未释放目前还未解决,不知道GC啥时候会回收…

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章