记一次ArrayList产生的线上OOM问题

本文记录了一次由于ArrayList的addAll方法导致的线上OOM问题。通过对ArrayList的内部原理分析,发现频繁扩容尤其是在循环中添加大量元素可能导致内存溢出。解决方案是预估所需容量或减少扩容操作。提醒开发者注意初始化ArrayList的容量,避免线上出现此类问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言:本以为(OutOfMemoryError)OOM问题会离我们很远,但在一次生产上线灰度的过程中就出现了Java.Lang.OutOfMemoryError:Java heap space异常,通过对线上日志的查看,最终定位到ArrayList#addAll方法中,出现这个问题的原因是:由于历史原因有个接口的响应时间经常超时,所以笔者对其进行了优化,之前使用的是ArrayList#add方法,笔者通过一系列修改后将add方法修改为了addAll方法,导致内存溢出。但具体是怎样产生的呢,下面对其详细分析。


ArrayList的内部原理

谈起ArrayList想必大家在日常中经常使用,用于存储一系列的元素。由于笔者在使用过程中出现了OOM异常,这里有必要对其内部原理进行简单的分析:

#1.ArrayList底层采用数组来存储数据,查找速度快,毕竟直接使用数组下标进行数据的查找。这里有一点特别重要其内部的数据存储结构为数组。

#2.数组:数组是一种线性表数据结构,它是一组连续的内存空间。注意:一组连续的内存空间,这就意味着在申请数组时如果不能满足连续的内存空间,哪怕是内存足够也会导致OOM问题。

#3.ArrayList的默认容量为10,超过10时,会进行扩容:int newCapacity = oldCapacity + (oldCapacity >> 1);相当于扩大为原来的1.5倍。其扩容函数如下:

 1  private void grow(int minCapacity) {
 2         // overflow-conscious code
 3         // 获得当前ArrayList的大小
 4         int oldCapacity = elementData.length;
 5         // 进行扩容,扩大为原来的1.5倍,那为什么不直接*1.5呢,因为位操作速度更快
 6         int newCapacity = oldCapacity + (oldCapacity >> 1);
 7         // minCapacity参数为扩容前确认的数组大小参数,将在下面进行分析
 8         // 如果新容量比minCapacity小,说明容量不够,则使用minCapacity
 9         if (newCapacity - minCapacity < 0)
10             newCapacity = minCapacity;
11         // 如果newCapacity大于最大ArrayList承受的最大值,则计算最大值    
<think>好的,用户想了解中科软Java面试常问的问题。我需要先回忆一下中科软的业务方向,可能他们主要做金融、保险、政府类的软件,所以面试问题可能涉及基础Java知识、数据库、框架以及项目经验。 首先,用户可能是准备面试的求职者,可能刚毕业或者有少量工作经验。他们需要具体的知识点,而不仅仅是泛泛而谈。所以需要分门别类,把问题归类,比如Java基础、数据库、框架、设计模式、系统设计、项目经验等。 接下来要考虑中科软可能重视哪些方面。比如,Java基础中的集合、多线程、JVM这些肯定是重点。数据库方面,事务、索引优化、SQL语句。框架的话,Spring、Spring MVC、Spring Boot、MyBatis这些常见框架的问题。设计模式可能问单例、工厂之类的。系统设计方面,高并发、缓存、分布式系统可能也会涉及。项目经验的话,会问项目中的具体问题解决,如何协作,遇到的挑战等。 还要注意用户提到的系统级指令,比如数学表达式用$...$,但这里可能用不上。不过要确保回答结构清晰,分点列出,便于用户理解。 另外,用户可能需要真实可靠的信息,所以需要参考常见的Java面试题,结合中科软可能的业务需求,整理出相关问题。比如金融行业可能对事务、数据一致性要求高,所以数据库事务隔离级别、Spring事务管理这些会是重点。 还要考虑是否有遗漏的部分,比如网络协议、算法题、Linux命令等,但可能中科软更偏向于Java后端开发,所以重点在Java和数据库。不过安全起见,可以补充一些其他可能的问题,比如HTTP和HTTPS的区别,或者基本的算法复杂度分析。 最后,确保回答逻辑清晰,逐步展开每个大类别下的具体问题,并给出简要解释,帮助用户理解为什么这些问题是重点,以及如何准备。这样用户不仅得到问题列表,还能知道如何应对。</think>中科软作为国内知名软件企业,Java岗位面试常围绕技术基础、项目经验和实际场景展开。以下为常见问题分类及典型题目,供针对性准备: --- ### 一、Java基础 1. **面向对象特性** - 多态的实现原理(重载 vs 重写) - 抽象类与接口的区别,JDK 8后接口的变化 - `equals()`与`==`的区别,`hashCode()`的作用 2. **集合框架** - `HashMap`底层结构(数组+链表/红黑树),扩容机制 - `ConcurrentHashMap`如何保证线程安全(JDK 7 vs JDK 8) - `ArrayList`与`LinkedList`的适用场景 3. **多线程与并发** - `synchronized`与`ReentrantLock`的对比 - `volatile`关键字的作用(可见性、禁止指令重排) - 线程池参数(核心线程数、阻塞队列、拒绝策略)及工作流程 4. **JVM相关** - 内存区域划分(堆、栈、方法区) - 垃圾回收算法(标-清除、G1的Region分区) - 类加载机制(双亲委派模型及打破场景) --- ### 二、数据库与SQL 1. **MySQL优化** - 索引失效的常见场景(如`LIKE '%xx'`、隐式类型转换) - Explain执行计划的关键字段解读(type、key、Extra) - 事务隔离级别(脏读、不可重复读、幻象读对应解决级别) 2. **SQL编写** - 多表连接查询(LEFT JOIN与INNER JOIN区别) - 分组统计(GROUP BY + HAVING) - 分页优化(避免`LIMIT`大偏移量) --- ### 三、主流框架 1. **Spring** - Bean生命周期(实例化、属性注入、初始化方法) - AOP实现原理(动态代理 vs CGLIB) - 声明式事务传播行为(如`PROPAGATION_REQUIRED`) 2. **Spring Boot** - 自动配置原理(`@EnableAutoConfiguration`与`spring.factories`) - 如何自定义Starter - Actuator监控端点使用 3. **MyBatis** - `#{}`与`${}`的区别(预编译 vs 字符串替换) - 一级缓存与二级缓存的作用域 - 动态SQL标签(`<if>`、`<foreach>`) --- ### 四、系统设计与场景题 1. **设计模式** - 单例模式的线程安全实现(双重检查锁、静态内部类) - 工厂模式在Spring中的应用(BeanFactory) - 代理模式与AOP的关系 2. **分布式场景** - 如何实现接口幂等性(Token机制、数据库唯一索引) - 缓存穿透/雪崩的解决方案(布隆过滤器、随机过期时间) - 分布式锁实现方式(Redis SETNX、ZooKeeper临时节点) --- ### 五、项目经验 1. **核心问题** - 介绍一个复杂度较高的模块,技术选型依据 - 如何排查线OOM问题(MAT分析堆转储) - 项目中的性能优化案例(如SQL慢查询优化) 2. **软技能** - 团队协作中如何解决技术分歧 - 需求延期时的应对策略 - 学习新技术的方法论 --- ### 六、附加考点 - **网络基础**:TCP三次握手、HTTP与HTTPS区别 - **Linux命令**:查看进程(`ps`)、日志检索(`grep`)、内存占用(`top`) - **算法题**:字符串操作、二叉树遍历(可能要求手写伪代码) --- **准备建议**: 1. 结合项目经历梳理技术亮点,用STAR法则(情境-任务-行动-结果)组织回答 2. 对简历中提到的技术栈(如Redis、Kafka)需深入掌握原理 3. 模拟场景题时,先明确需求再逐步拆解,体现逻辑性
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值