实际项目中的一个OOM和事务的问题

问题描述

线上代码如下:

  1. 先加上一个数据库行锁

  2. 执行业务操作

  3. 最后释放这个行锁

@Transactional(rollbackFor = Exception.class)

void method(){

try{

add db lock

doBiz();

}finally{

release db lock

}

}

实际遇到了 doBiz() 导致OOM了,导致数据库行锁没有释放,结果其它用户执行method方法全部失败。

问题排查

首先如上代码因为一直以来都是好的,没有遇到问题,而报错是 **DUPLICATE KEY**错误(在add db lock方法中,拿个表设置了unique key,插入相同的会报错)。

另外因为线上是很多用户操作的。

所以:直接去arthas trace该方法,然后顺便去看doBiz()方法有没有重复加锁的逻辑,且doBiz()方法是耗时的;另外还去数据库看了是否锁加成功了,即数据库有没有记录。发现都没有

然后排查到的几个现象,这个时候还没有人准确看doBiz()的异常

  1. 联系dba查看可以看到很多加锁的sql操作在等待队列中

  2. arthas trace 加锁报错了 MySQLTransactionRollbackException: Lock wait timeout exceeded,try restarting transaction

  3. 部分可以看到释放锁但是报错Communications link failure。

所以是加锁失败,然后又一直重试,一直失败。(所以去数据库查询的时候没有找到记录,然后因为很简单的sql,所以怀疑锁没有释放,去找重复加锁和释放锁为什么没有执行,结果原因在这里)。另外发现了释放锁有问题,则机器肯定也有问题了。

随后又看了机器的各个log,发现了OOM异常。所以结论是:在一个事务中,业务代码超时且OOM异常了,导致事务一直没有提交,等待锁超时,然后又一直重试。

随后改写代码加日志,验证猜想。所以这也说明平时日常工作中重要地方加日志的重要性质(然后走了一次机器发布重启,当时主要是排查也没有想到这个)

@Transactional(rollbackFor = Exception.class)

void method(){

try{

add db lock

doBiz();

}catch(Exception e){

logger.error("xxx", e)

}finally{

logger.warn("release lock");

release db lock

}

}

测试发现有测试例子是没有执行finally的释放锁方法的


顺便也说下OOM的问题,是select了数据库中的大字段,大量内存占用处,一般这个大纪录字段是单个记录才需要查询的; 当扫描了所有记录,且也select这个字段导致出现OOM。所以spring mybatis 结合查询的时候,不要什么时候都select所有字段。

相关推荐
麦聪聊数据39 分钟前
Web 原生架构如何重塑企业级数据库协作流?
数据库·sql·低代码·架构
未来之窗软件服务40 分钟前
数据库优化提速(四)新加坡房产系统开发数据库表结构—仙盟创梦IDE
数据库·数据库优化·计算机软考
Goat恶霸詹姆斯2 小时前
mysql常用语句
数据库·mysql·oracle
大模型玩家七七2 小时前
梯度累积真的省显存吗?它换走的是什么成本
java·javascript·数据库·人工智能·深度学习
曾经的三心草2 小时前
redis-9-哨兵
数据库·redis·bootstrap
明哥说编程3 小时前
Dataverse自定义表查询优化:D365集成大数据量提速实战【索引配置】
数据库·查询优化·dataverse·dataverse自定义表·索引配置·d365集成·大数据量提速
xiaowu0803 小时前
C# 拆解 “显式接口实现 + 子类强类型扩展” 的设计思想
数据库·oracle
讯方洋哥3 小时前
HarmonyOS App开发——关系型数据库应用App开发
数据库·harmonyos
惊讶的猫4 小时前
Redis持久化介绍
数据库·redis·缓存
Apple_羊先森4 小时前
ORACLE数据库巡检SQL脚本--19、磁盘读次数最高的前5条SQL语句
数据库·sql·oracle