【性能优化】纳尼?内存又溢出了?!是时候总结一波了!!
点击上方蓝色“冰河技术”,关注并选择“设为星标” 持之以恒,贵在坚持,每天进步一点点! 作者个人研发的在高并发场景下,提供的简单、稳定、可扩展的延迟消息队列框架,具有精准的定时任务和延迟队列处理功能。自开源半年多以来,已成功为十几家中小型企业提供了精准定时调度方案,经受住了生产环境的考验。为使更多童鞋受益,现给出开源框架地址: https://github.com/sunshinelyz/mykit-delay PS: 欢迎各位Star源码,也可以pr你牛逼哄哄的代码。 写在前面 相信小伙伴们在平时工作的过程中,或多或少都会遇到一个场景:内存溢出。如果你没有遇到过这个场景,那就说明你是个假的程序员。哈哈,开个玩笑,平时工作过程中,我们确实会遇到这个问题。今天,我就将平时工作过程中遇到的内存溢出情况做个简单的总结,以通俗易懂的代码案例的形式直观的分享给大家。希望能够为小伙伴们带来实质性的帮助。 案例介绍 这里,我将在平时工作过程中总结的内存溢出的情况,以代码案例的形式直观的分享给大家,希望能够为小伙伴们带来实质性的帮助。 接下来,我们就以代码案例的形式来分析各种内存溢出的情况。 定义主类结构 首先,我们创建一个类叫做BlowUpJVM,所有的案例实验都是基于这个类进行。 publicclassBlowUpJVM{} 栈深度溢出 publicstaticvoidtestStackOverFlow(){BlowUpJVM.testStackOverFlow();} 栈不断递归,而且没有处理,所以虚拟机栈就不断深入不断深入,栈深度就这样溢出了。 永久代内存溢出 publicstaticvoidtestPergemOutOfMemory1(){//方法一失败List<String>list=newArrayList<String>();while(true){list.add(UUID.randomUUID().toString().intern());}} 打算把String常量池堆满,没想到失败了,JDK1.7后常量池放到了堆里,也能进行垃圾回收了。 然后换种方式,使用cglib,用Class把老年代取堆满 publicstaticvoidtestPergemOutOfMemory2(){try{while(true){Enhancerenhancer=newEnhancer();enhancer.setSuperclass(OOM.class);enhancer.setUseCache(false);enhancer.setCallback(newMethodInterceptor(){@OverridepublicObjectintercept(Objectobj,Methodmethod,Object[]args,MethodProxyproxy)throwsThrowable{returnproxy.invokeSuper(obj,args);}});enhancer.create();}}catch(Exceptione){e.printStackTrace();}} 虚拟机成功内存溢出了,那JDK动态代理产生的类能不能溢出呢? publicstaticvoidtestPergemOutOfMemory3(){while(true){finalOOMoom=newOOM();Proxy.newProxyInstance(oom.getClass().getClassLoader(),oom.getClass().getInterfaces(),newInvocationHandler(){publicObjectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable{Objectresult=method.invoke(oom,args);returnresult;}});}} 事实表明,JDK动态代理差生的类不会造成内存溢出,原因是:JDK动态代理产生的类信息,不会放到永久代中,而是放在堆中。 本地方法栈溢出 publicstaticvoidtestNativeMethodOutOfMemory(){intj=0;while(true){Printer.println(j++);ExecutorServiceexecutors=Executors.newFixedThreadPool(50);inti=0;while(i++<10){executors.submit(newRunnable(){publicvoidrun(){}});}}} 这个的原理就是不断创建线程池,而每个线程池都创建10个线程,这些线程池都是在本地方法区的,久而久之,本地方法区就溢出了。 JVM栈内存溢出 publicstaticvoidtestStackOutOfMemory(){while(true){Threadthread=newThread(newRunnable(){publicvoidrun(){while(true){}}});thread.start();}} 线程的创建会直接在JVM栈中创建,但是本例子中,没看到内存溢出,主机先挂了,不是JVM挂了,真的是主机挂了,无论在mac还是在windows,都挂了。 温馨提示,这个真的会死机的。 堆溢出 publicstaticvoidtestOutOfHeapMemory(){List<StringBuffer>list=newArrayList<StringBuffer>();while(true){StringBufferB=newStringBuffer();for(inti=0;i<10000;i++){B.append(i);}list.add(B);}} 不断往堆中塞新增的StringBuffer对象,堆满了就直接溢出了。 重磅福利 微信搜一搜【冰河技术】微信公众号,关注这个有深度的程序员,每天阅读超硬核技术干货,公众号内回复【PDF】有我准备的一线大厂面试资料和我原创的超硬核PDF技术文档,以及我为大家精心准备的多套简历模板(不断更新中),希望大家都能找到心仪的工作,学习是一条时而郁郁寡欢,时而开怀大笑的路,加油。如果你通过努力成功进入到了心仪的公司,一定不要懈怠放松,职场成长和新技术学习一样,不进则退。如果有幸我们江湖再见! 另外,我开源的各个PDF,后续我都会持续更新和维护,感谢大家长期以来对冰河的支持!! 写在最后 如果你觉得冰河写的还不错,请微信搜索并关注「 冰河技术 」微信公众号,跟冰河学习高并发、分布式、微服务、大数据、互联网和云原生技术,「 冰河技术 」微信公众号更新了大量技术专题,每一篇技术文章干货满满!不少读者已经通过阅读「 冰河技术 」微信公众号文章,吊打面试官,成功跳槽到大厂;也有不少读者实现了技术上的飞跃,成为公司的技术骨干!如果你也想像他们一样提升自己的能力,实现技术能力的飞跃,进大厂,升职加薪,那就关注「 冰河技术 」微信公众号吧,每天更新超硬核技术干货,让你对如何提升技术能力不再迷茫! 留言区 本文分享自微信公众号 - 冰河技术(hacker-binghe)。如有侵权,请联系 support@oschina.cn 删除。本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。