Mycat心跳机制梳理

本文分析了一段关于Mycat后端连接池的心跳检测代码,指出了并发操作可能导致的连接重复使用和响应混乱问题,并讨论了MySQLDetector中频繁创建SQLJob的问题。作者提出了代码修改建议和对数据库连接重用性的思考,还分享了Java开发者的学习资源和提升技术的方法论。
摘要由CSDN通过智能技术生成

关键代码:

for (PhysicalDBPool node : nodes.values()) {

node.heartbeatCheck(heartPeriod);

}

PhysicalDatasource.heatBeatCheck

上述代码主要就是遍历ConQueue(Mycat后端连接池存放的地方),选出需要进行心跳的连接,然后执行conHeartBeatHandler.doHeartBeat。

重点关注一下检测checkIfNeedHertBeat方法:

private void checkIfNeedHeartBeat(

LinkedList heartBeatCons, ConQueue queue,

ConcurrentLinkedQueue checkLis,

long hearBeatTime, long hearBeatTime2) {

int maxConsInOneCheck = 10;

Iterator checkListItor = checkLis.iterator();

while (checkListItor.hasNext()) {

BackendConnection con = checkListItor.next(); // @1

if (con.isClosedOrQuit()) {

checkListItor.remove();

continue;

}

if (validSchema(con.getSchema())) {

if (con.getLastTime() < hearBeatTime

&& heartBeatCons.size() < maxConsInOneCheck) {

checkListItor.remove(); // @2

// Heart beat check

con.setBorrowed(true);

heartBeatCons.add(con); // @3

}

} else if (con.getLastTime() < hearBeatTime2) {

// not valid schema conntion should close for idle

// exceed 2*conHeartBeatPeriod

checkListItor.remove();

con.close(" heart beate idle ");

}

}

}

上面的逻辑是,获取ConQueue中的所有连接(ConcurrentLinkedQueue),然后获取一个遍历器,

进行遍历,然后判断该连接是否需要加入到心跳检测中,如果符合,就先移除,然后加入到待检测列表中。

上述这个步骤,乍一看没什么问题,但如果你与获取连接一起考虑,就会发现问题,我们知道,获取连接,也是从ConQueue的(ConcurrentLinkedQueue中poll一个,这样就会产生这样一个问题,对于一个连接c,有可能放入到检测列表中,并同时分配给其他线程,造成一个连接,在同一时间,会又做心跳检测,又执行其他SQL,造成结果混乱。

分析上述代码出现场景:

t1 线程运行到代码@1,t2线程获取数据库连接,使用poll()方法,获取一个连接,有可能就是获取的这个连接正好与t1代码@1的连接一样,

然后t1运行到代码@2,将移除,但并不会返回成功与否,然后运行代码@3,将连接放入到待检测列表中,此时一个连接,同时被多个线程使用,并且,最后一个使用,会改变连接的响应处理器,造成第二个线程的响应处理器会获得第一个线程的响应结果。

3)检测频率:dataNodeHeartbeatPeriod,默认值10s,见io.mycat.MycatServer 【验证数据节点是否正常】,也就是readhost,writerhost的存活状态检测。

关键代码:MySQLDetector

上述代码引起了我一个疑问,为什么heartbeat方法没调用一次,就要新建一个SQLJob对象,为什么嗯?

我们先重点看一下看一下该方法的调用链:

一个PhysicalDatasource对应一个readerhost或writerhost。

跟踪代码发现,PhysicalDatasource拥有一个(private DBHeartbeat heartbeat),一个MySQLHeartbeat拥有一个MySQLDetector,

那为什么执行MySQLDetector时,确需要将SQLJob声明为volatile,并没次调用都新建一个对象,估计是因为每次心跳检查都需要一个独立的

OneRawSQLQueryResultHandler resultHandler = new OneRawSQLQueryResultHandler( fetchColms, this);故这里每次都新建了SQLJob。

此处,应该有一个可优化 的点:由于心跳连接运行比较频繁,默认10s一次,故此处连接的重用性问题值得思考。

上述BUG,是由我公司生产环境下跑出的,因为项目组反映,一条查询连接,竟然返回了心跳检测的响应结果:

附上本次代码修改后的代码:

PhysicalDatasource.checkIfNeedHeartBeat

private void checkIfNeedHeartBeat(

LinkedList heartBeatCons, ConQueue queue,

ConcurrentLinkedQueue checkLis,

long hearBeatTime, long hearBeatTime2) {

int maxConsInOneCheck = 10;

Iterator checkListItor = checkLis.iterator();

while (checkListItor.hasNext()) {

BackendConnection con = checkListItor.next();

if (con.isClosedOrQuit()) {

checkListItor.remove();

continue;

}

if (validSchema(con.getSchema())) {

if (con.getLastTime() < hearBeatTime

&& heartBeatCons.size() < maxConsInOneCheck) {

if(checkLis.remove(con)) {

//如果移除成功,则放入到心跳连接中,如果移除失败,说明该连接已经被其他线程使用,忽略本次心跳检测

con.setBorrowed(true);

heartBeatCons.add(con);

}

// checkListItor.remove();

// // Heart beat check
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

如何快速更新自己的技术积累?

  • 在现有的项目里,深挖技术,比如用到netty可以把相关底层代码和要点都看起来。
  • 如果不知道目前的努力方向,就看自己的领导或公司里技术强的人在学什么。
  • 知道努力方向后不知道该怎么学,就到处去找相关资料然后练习。
  • 学习以后不知道有没有学成,则可以通过面试去检验。

我个人觉得面试也像是一场全新的征程,失败和胜利都是平常之事。所以,劝各位不要因为面试失败而灰心、丧失斗志。也不要因为面试通过而沾沾自喜,等待你的将是更美好的未来,继续加油!

以上面试专题的答小编案整理成面试文档了,文档里有答案详解,以及其他一些大厂面试题目

八年CRUD,疫情备战三个月,三面头条、四面阿里拿offer面经分享

八年CRUD,疫情备战三个月,三面头条、四面阿里拿offer面经分享

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!
不要因为面试通过而沾沾自喜,等待你的将是更美好的未来,继续加油!**

以上面试专题的答小编案整理成面试文档了,文档里有答案详解,以及其他一些大厂面试题目

[外链图片转存中…(img-ON037VIU-1712048091861)]

[外链图片转存中…(img-zO5mV60D-1712048091862)]

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值