业务场景:
- 使用java8提供的ScriptEngineManager 加载并运行groovy脚本,每次运行从数据库中获取脚本来实现动态编译运行
运行一段时间后
- 初始用户量少,没感觉,偶尔一次内存溢出重启就好(话外音:搞不清楚状况先重启 总是没错的)
- 后来用户量越来越大,溢出的次数就越来越频繁
java.lang.OutOfMemoryError: GC overhead limit exceeded
通过导出headdump分析
- 发现非堆内存一直在持续增加,不能释放,网上查询的资料也说明了这一点,groovy确实涉及内存溢出问题
本地验证
- 循环调用代码片断
···
ScriptEngineManager factory = new ScriptEngineManager();
ScriptEngine engine = factory.getEngineByName(“groovy”);
String script = “sql = “”“sql\n” +
" select nick_name name,nick_name nickName,head_img,sex gender,phone,open_id,project_account account,update_date,4 createType \n” +
" from t_project_customer where phone is not null and update_date >= #{offset} order by update_date asc limit 100\n" +
“”""";
engine.eval(script)
··· - 3分钟内增加了10M的非堆内存

本文探讨了使用Java 8的ScriptEngineManager加载并运行Groovy脚本导致的内存溢出问题,特别是在高并发场景下非堆内存持续增长的原因及解决办法。
最低0.47元/天 解锁文章
1504





