阿里面试:Java开发中,应如何避免OOM
Java内存管理:避免OOM的10个实用小技巧
引言
在Java开发中,OutOfMemoryError(OOM)错误一直是令开发者头疼的问题,也是Java面试中出现核心频率很高的问题。
那么我们究竟怎么样才能够有效正确的管理内存,日常开发中究竟要注意哪些核心技巧来避免OOM错误。
本文将带大家一起学习10个避免OOM的实用小技巧,让大家在工作中能够有的放矢,避免OOM错误的飞来横祸。
正文
1、 合理配置JVM内存参数
应用上线前,设置合理的JVM启动参数是避免OOM的第一步。
通过调整堆内存、栈内存和Metaspace的大小,可以有效地管理内存资源。
以4G内存为例,应用上线时可以参考如下配置:
// 示例:设置JVM的启动参数
// -Xms1024m 设置初始堆大小为1024MB
// -Xmx2048m 设置最大堆大小为2048MB
// -XX:NewSize=512m 设置新生代大小为512MB
// -XX:MaxNewSize=1024m 设置新生代最大大小为1024MB
// -XX:MetaspaceSize=256m 设置Metaspace的初始空间大小为256MB
// -XX:MaxMetaspaceSize=512m 设置Metaspace的最大空间大小为512MB
2、 使用轻量级对象
在开发过程中,尽可能的使用轻量级对象,减少内存消耗。
例如,使用原始数据类型代替包装类,使用StringBuffer/StringBuilder代替String进行字符串操作。
// 使用原始数据类型代替包装类
int i = 10;
// 使用StringBuilder进行字符串拼接
StringBuilder sb = new StringBuilder();
sb.append("Hello");
sb.append(" ");
sb.append("World");
String result = sb.toString();`
3、 对象池技术
对于频繁创建和销毁的对象,可以考虑使用对象池技术,以减少GC的压力。
// 使用Commons Pool2实现对象池
// 定义一个简单的对象池工厂
public class MyObjectPoolFactory extends BasePooledObjectFactory<MyObject> {
@Override
public MyObject create() throws Exception {
return new MyObject();
}
@Override
public PooledObject<MyObject> wrap(MyObject obj) {
return new DefaultPooledObject<>(obj);
}
}
// 使用对象池
ObjectPool<MyObject> pool = new GenericObjectPool<>(new MyObjectPoolFactory());
MyObject obj = null;
try {
obj = pool.borrowObject();
// 使用对象...
} finally {
pool.returnObject(obj);
}```
**4、** **优化数据结构选择**
根据应用场景合理选择数据结构,例如,在频繁读取操作中使用ArrayList,在频繁插入删除操作中使用LinkedList。
```java
// 在频繁读取操作中使用ArrayList
List<String> arrayList = new ArrayList<>();
arrayList.add("Java");
arrayList.add("Python");
String element = arrayList.get(0);
// 在频繁插入删除操作中使用LinkedList
List<String> linkedList = new LinkedList<>();
linkedList.add("Java");
linkedList.add("Python");
linkedList.remove(0);
5、 避免创建不必要的对象
尽量复用已有对象,避免无谓的对象创建,特别是在循环或频繁调用的方法中。
// 避免在循环中创建对象
String result = "";
for(int i = 0; i < 100; i++) {
// 错误示范:在循环体内创建StringBuilder对象
// 正确做法是将StringBuilder的创建放到循环体外
StringBuilder sb = new StringBuilder(result);
sb.append(i);
result = sb.toString();
}
6、 及时释放不再使用的对象
确保不再使用的对象能够被GC及时回收,例如,将对象引用设置为null,关闭流等。
// 将对象引用设置为null
Object obj = new Object();
// 使用对象...
obj = null; // 明确标记obj不再使用
// 关闭流
FileInputStream fis = null;
try {
fis = new FileInputStream("test.txt");
// 使用流...
} finally {
if(fis != null) {
fis.close();
}
}
7、 使用软引用和弱引用管理内存
对于可回收的对象,使用软引用(SoftReference)或弱引用(WeakReference),以便在JVM内存不足时能被回收。
// 使用软引用
SoftReference<Object> softRef = new SoftReference<>(new Object());
// 使用弱引用
WeakReference<Object> weakRef = new WeakReference<>(new Object());
8、 合理使用缓存
合理设计缓存策略,避免缓存占用过多内存。可以使用第三方缓存库如Ehcache,Guava Cache等,并设置合理的过期策略。
// 使用Guava Cache
Cache<String, Object> cache = CacheBuilder.newBuilder()
.maximumSize(1000)
.expireAfterWrite(10, TimeUnit.MINUTES)
.build();
// 向缓存中添加对象
cache.put("key", new Object());
// 从缓存中获取对象
Object obj = cache.getIfPresent("key");
9、 监控和分析内存使用
使用JVM提供的工具(如jvisualvm, jconsole)监控和分析应用的内存使用情况,及时发现并解决内存问题。
10、 优化GC策略
根据应用的实际情况,调整和优化GC策略,减少GC的执行时间,提升系统的性能。
本文总结
监控和分析内存使用
使用JVM提供的工具(如jvisualvm, jconsole)监控和分析应用的内存使用情况,及时发现并解决内存问题。
10、 优化GC策略
根据应用的实际情况,调整和优化GC策略,减少GC的执行时间,提升系统的性能。
本文总结
避免OOM错误并非难事,关键在于对Java内存管理有深入的理解和正确的实践。通过以上10个实用小技巧的应用,可以有效地管理和优化Java应用的内存使用,避免内存溢出的问题。务必记得,持续的监控、分析和优化是保持应用稳定运行的关键。
最后说一句(求关注,求赞,别白嫖我)
最近无意间获得一份阿里大佬写的刷题笔记,一下子打通了我的任督二脉,进大厂原来没那么难。
这是大佬写的, 7701页的BAT大佬写的刷题笔记,让我offer拿到手软
本文,已收录于,我的技术网站 aijiangsir.com,有大厂完整面经,工作技术,架构师成长之路,等经验分享
求一键三连:点赞、分享、收藏
点赞对我真的非常重要!在线求赞,加个关注我会非常感激!
阿里面试:Java开发中,应如何避免OOM的更多相关文章
- 编写高质量代码:改善Java程序的151个建议(第一章:JAVA开发中通用的方法和准则)
编写高质量代码:改善Java程序的151个建议(第一章:JAVA开发中通用的方法和准则) 目录 建议1: 不要在常量和变量中出现易混淆的字母 建议2: 莫让常量蜕变成变量 建议3: 三元操作符的类型务 ...
- Java开发中代码规范有哪些?
Java开发中所要遵守的编码规范大体上有如下7点.命名规范.注释规范.缩进排版规范.文件名规范.声明规范.语句规范以及编程规范. 1.命名规范 (1)所有的标示符都只能用ASCⅡ字母(A-Z或a-z) ...
- 轻松理解 Java开发中的依赖注入(DI)和控制反转(IOC)
前言 关于这个话题, 网上有很多文章,这里, 我希望通过最简单的话语与大家分享. 依赖注入和控制反转两个概念让很多初学这迷惑, 觉得玄之又玄,高深莫测. 这里想先说明两点: 依赖注入和控制反转不是高级 ...
- paip.java 开发中web server的选择jboss resin tomcat比较..
paip.java 开发中web server的选择jboss resin tomcat比较.. 作者Attilax 艾龙, EMAIL:1466519819@qq.com 来源:attilax的专 ...
- Java开发中常见的危险信号(中)
本文来源于我在InfoQ中文站原创的文章,原文地址是:http://www.infoq.com/cn/news/2013/12/common-red-flags-in-java-1 Dustin Ma ...
- Java开发中文件读取方式总结
JAVA开发中,免不了要读文件操作,读取文件,首先就需要获取文件的路径. 路径分为绝对路径和相对路径. 在文件系统中,绝对路径都是以盘符开始的,例如C:\abc\1.txt. 什么是相对路径呢?相对路 ...
- java开发中遇到的问题及解决方法(持续更新)
摘自 http://blog.csdn.net/pony12/article/details/38456261 java开发中遇到的问题及解决方法(持续更新) 工作中,以C/C++开发为主,难免与其他 ...
- Java开发中常见的危险信号(上)
本文来源于我在InfoQ中文站原创的文章,原文地址是:http://www.infoq.com/cn/news/2013/12/common-red-flags-in-java-1 Dustin Ma ...
- 完整java开发中JDBC连接数据库代码和步骤[申明:来源于网络]
完整java开发中JDBC连接数据库代码和步骤[申明:来源于网络] 地址:http://blog.csdn.net/qq_35101189/article/details/53729720?ref=m ...
- Java 开发中的对象拷贝
前言 在 Java 开发中,很多时候需要将两个属性基本相同的对象进行属性复制,比如 DO 转 VO等等. 本文主要介绍自己实现的简易拷贝工具类与 Spring 提供的属性拷贝的对比. Spring 提 ...
随机推荐
- [转帖]tcp、http和socket的区别
https://www.jianshu.com/p/88d69454bdde 一:tcp协议 tcp协议属于传输层协议(UDP也属于传输层协议,但是UDP协议是无状态的).建立一个TCP连接需要三次握 ...
- Chrony 的学习与使用
Chrony 的学习与使用 背景 之前捯饬 ntp 发现很麻烦, 经常容易弄错了. 昨天处理文件精确时间时 想到了时间同步. 发现只有自己总结的ntpdate 但是还没有 chronyd相关的总结 本 ...
- [转帖]TiKV & TiFlash 加速复杂业务查询丨TiFlash 应用实践
返回全部 边城元元案例实践2022-08-02 复杂业务查询对于传统的关系型数据库来说是一种考验,而通过 TiKV 行存与 TiFlash 的列存结合使用就能很好地应对.本文根据 TUG 用户边城元元 ...
- [转帖]jmeter线程组与循环次数的区别
在压测的时候,有些接口需要携带登录信息,但是我们只想登录一次,然后其他接口进行多用户压测,此时你会怎么办?用仅一次控制器实现吗?下面我们来看看用仅一次控制器能不能实现 压测时jmeter中的线程数是模 ...
- 人大金仓学习之一_kwr的简单学习
人大金仓学习之一_kwr的简单学习 摘要 周末在家想着学习一下数据库相关的内容. 网上找了不少资料, 想着直接在本地机器上面进行一下安装与验证 理论上linux上面应该更加简单. windows 上面 ...
- [转帖]《Linux性能优化实战》笔记(24)—— 动态追踪 DTrace
使用 perf 对系统内核线程进行分析时,内核线程依然还在正常运行中,所以这种方法也被称为动态追踪技术.动态追踪技术通过探针机制来采集内核或者应用程序的运行信息,从而可以不用修改内核和应用程序的代码就 ...
- [转帖]Day63_Kafka(一)
第一讲 Kafka基础操作 课程大纲 课程内容 学习效果 掌握目标 Kafka简介 消息队列 掌握 Kafka简介 Kafka分布式环境 Kafka操作 Kafka shell 掌握 Kafka ap ...
- ARM下KVM虚拟化的损耗验证--redis
ARM下KVM虚拟化的损耗验证 摘要 看Windows 上面的 Workstation的虚拟机的 网络层的延迟特别高. 突然想之前统计都是直接在本地验证的, 只考虑了虚拟化CPU的性能损耗 没有考虑虚 ...
- [转帖]jmeter压力测试
使用jmeter 进行并发压力测试. 首先需要安装好jmeter,下面以widows操作平台为例: 1.确保电脑安装并配置好java环境:具体怎么下载和配置请自行百度: 2.登录jmeter官网htt ...
- [知乎]2019-nCov的致死率问题
https://www.zhihu.com/question/369630554/answer/998649507 知乎 dr.李的文章 跟自己一开始的理解很相似.. 个人也认为死亡率会高于2% 武汉 ...