欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺析MySQL中主從延遲問(wèn)題的原因與解決方法

 更新時(shí)間:2024年02月28日 09:44:35   作者:DaveCui  
這篇文章主要帶大家從一個(gè)主從延遲問(wèn)題開(kāi)始回顧主從復(fù)制原理,并思考主從延遲造成的原因和解決方案,文中的示例代碼講解詳細(xì),感興趣的可以了解下

從一個(gè)主從延遲問(wèn)題開(kāi)始回顧主從復(fù)制原理,并思考主從延遲造成的原因和解決方案。當(dāng)然,作為底層開(kāi)發(fā),最后還是只能快準(zhǔn)狠的通過(guò)一個(gè)簡(jiǎn)單粗暴的等待方案進(jìn)行應(yīng)對(duì)。

事情的起因

事情要從我寫(xiě)下這樣的代碼開(kāi)始

// 獲取當(dāng)前數(shù)據(jù)庫(kù)中未使用的數(shù)據(jù)轉(zhuǎn)為正在使用的狀態(tài)
int updateUsing = fateDataDao.update(FateDataStatusEnum.UNUSED.getCode(),FateDataStatusEnum.USING.getCode());
log.info("update UNUSED to USING:{}",updateUsing);
// 獲取正在使用狀態(tài)的數(shù)據(jù)
List<FateData> fateDataList = fateService.queryStatus(FateDataStatusEnum.USING.getCode());
log.info("queryStatus USING:{}",fateDataList.size());

這部分邏輯清晰簡(jiǎn)單明了,把UNUSED狀態(tài)的數(shù)據(jù)更新為USING狀態(tài),然后查詢(xún)?nèi)〕?strong>USING狀態(tài)的數(shù)據(jù)。

按照一個(gè)正常的邏輯來(lái)說(shuō)updateUsing的數(shù)量和fateDataList.size()的數(shù)量應(yīng)該一樣,但是,他不正常。

我在測(cè)試環(huán)境小數(shù)據(jù)量測(cè)試時(shí),這段代碼邏輯完全無(wú)誤。但是上了灰度環(huán)境進(jìn)行大量數(shù)據(jù)的測(cè)試就出現(xiàn)了這樣的問(wèn)題。

此時(shí),我?guī)е苫蠛筒唤?,將目光投向百度?/p>

首先我認(rèn)為問(wèn)題可能好似,MYSQL更新返回的是查詢(xún)到的行數(shù),而不是受影響的行數(shù)。

但是負(fù)責(zé)MYSQL的同事和我說(shuō)MYSQL已經(jīng)配置了返回受影響行數(shù),并告訴我應(yīng)該是主從延遲問(wèn)題,沒(méi)辦法解決,看看業(yè)務(wù)能不能改下吧。

這時(shí),我才反應(yīng)過(guò)來(lái)當(dāng)時(shí)粗略了解的主從延遲問(wèn)題,我已經(jīng)忘的差不多了。

什么是主從復(fù)制

要了解主從延遲,首先就要知道什么是主從復(fù)制。

MySQL的主從復(fù)制(Master-Slave Replication)是一種數(shù)據(jù)庫(kù)復(fù)制技術(shù),用于解決數(shù)據(jù)備份、讀寫(xiě)分離、負(fù)載均衡以及故障恢復(fù)等問(wèn)題。

主從復(fù)制的基本原理是將一個(gè)數(shù)據(jù)庫(kù)實(shí)例(主服務(wù)器)的數(shù)據(jù)復(fù)制到另一個(gè)或多個(gè)數(shù)據(jù)庫(kù)實(shí)例(從服務(wù)器),使得從服務(wù)器的數(shù)據(jù)與主服務(wù)器保持同步。

主從復(fù)制的基本工作流程

  • 從服務(wù)器連接到主服務(wù)器,生成兩個(gè)線程,一個(gè)I/O線程,一個(gè)SQL線程
  • 主服務(wù)器記錄所有的數(shù)據(jù)更改(INSERT、UPDATE、DELETE),同時(shí)會(huì)生成一個(gè) log dump 線程,用來(lái)給從庫(kù) i/o線程傳binlog。
  • 主服務(wù)器會(huì)生成一個(gè) log dump 線程將binlog寫(xiě)到relay log(中繼日志) 文件中
  • 從服務(wù)器的SQL 線程會(huì)讀取relay log文件中的日志,并解析成具體操作,來(lái)實(shí)現(xiàn)主從的操作一致,而最終數(shù)據(jù)一致;

流程圖如下:

主從復(fù)制解決的問(wèn)題

要用到主從復(fù)制的原因主要是為了高可用、高并發(fā)

  • 數(shù)據(jù)備份和恢復(fù): 從服務(wù)器可以用作主服務(wù)器的備份,當(dāng)主服務(wù)器發(fā)生故障時(shí),可以快速切換到從服務(wù)器進(jìn)行恢復(fù)。
  • 讀寫(xiě)分離: 主服務(wù)器負(fù)責(zé)寫(xiě)操作,而從服務(wù)器可以用于處理讀操作,從而分擔(dān)主服務(wù)器的負(fù)載,提高系統(tǒng)性能。
  • 負(fù)載均衡: 多個(gè)從服務(wù)器可以平均分擔(dān)讀請(qǐng)求,實(shí)現(xiàn)負(fù)載均衡,提高系統(tǒng)的可伸縮性。
  • 高可用性: 當(dāng)主服務(wù)器故障時(shí),可以快速切換到一個(gè)從服務(wù)器,保證系統(tǒng)的高可用性。

主從復(fù)制帶來(lái)的問(wèn)題

主從復(fù)制也會(huì)造成一些衍生出的問(wèn)題:

  • 數(shù)據(jù)一致性: 主從復(fù)制是異步的,存在一定的延遲,因此在進(jìn)行讀寫(xiě)分離時(shí),需要注意可能出現(xiàn)的數(shù)據(jù)一致性問(wèn)題。
  • 寫(xiě)操作集中: 所有寫(xiě)操作都集中在主服務(wù)器上,可能導(dǎo)致主服務(wù)器的負(fù)載較高。
  • 配置和維護(hù): 操作復(fù)雜,需要正確配置主從服務(wù)器,以及定期進(jìn)行監(jiān)控和維護(hù),確保系統(tǒng)正常運(yùn)行。

本次遇到的bug,主要就是數(shù)據(jù)一致性方面的問(wèn)題了。

主從延遲的原因

主庫(kù)使用單線程順序?qū)懭隻inlog,效率很高。然而從庫(kù)的SQL Thread線程需要對(duì)主庫(kù)的日志進(jìn)行隨機(jī)IO來(lái)重新執(zhí)行DML和DDL,效率較低,難以跟上主庫(kù)日志寫(xiě)入速度,因此產(chǎn)生了主從延遲。

另外,從庫(kù)SQL Thread也是單線程,當(dāng)主庫(kù)并發(fā)較高時(shí),產(chǎn)生大量DML,超過(guò)了從庫(kù)單線程能處理的速度,或者從庫(kù)中有大查詢(xún)語(yǔ)句產(chǎn)生鎖等待,也會(huì)導(dǎo)致從庫(kù)執(zhí)行延遲,無(wú)法跟上主庫(kù)的進(jìn)度

主從延遲的解決方案

從主從延遲的原因,我們定位出主要是主庫(kù)的高并發(fā)和從庫(kù)的SQL Thread效率低造成了這樣的問(wèn)題。

所以,在不增加機(jī)器的情況下的解決方案就是控制主庫(kù)的并發(fā)或者提升從庫(kù)的SQL Thread處理效率,例如MySQL 5.6 版本后,提供的一種多線程的方式。

簡(jiǎn)單粗暴的解決方案

當(dāng)然,對(duì)于我們公司的底層開(kāi)發(fā)來(lái)說(shuō),這種層次的設(shè)計(jì)需要更高層面的人來(lái)推動(dòng),而且也需要更長(zhǎng)的時(shí)間才能處理。

所以這里貼出我自己的解決方案。Thread.sleep時(shí)間請(qǐng)自行控制。

// 獲取當(dāng)前數(shù)據(jù)庫(kù)中未使用的數(shù)據(jù)轉(zhuǎn)為正在使用的狀態(tài)
int updateUsing = fateDataDao.update(FateDataStatusEnum.UNUSED.getCode(),FateDataStatusEnum.USING.getCode());
LOGGER.info("update UNUSED to USING:{}",updateUsing);
// 獲取正在使用狀態(tài)的數(shù)據(jù)
List<FateData> fateDataList = fateService.queryStatus(FateDataStatusEnum.USING.getCode());
LOGGER.info("queryStatus USING:{}",fateDataList.size());
// 如果數(shù)據(jù)相等,直接略過(guò)。
if(updateUsing != fateDataList.size()){
    // updateUsing為0,但fateDataList不為空的情況。任務(wù)失敗,未更新
    if (updateUsing == 0){
        Cat.logEvent("updateTmpData","jobFailed:"+"updateUsing:"+updateUsing+"--fateDataList:"+fateDataList.size());
        transaction.setStatus(Transaction.SUCCESS);
        return response;
    }
    // 數(shù)據(jù)數(shù)目不相等,等待三秒相等再繼續(xù)
    boolean equalFlag = false;
    while(!equalFlag){
        // 等待主從延遲
        Thread.sleep(3000);
        fateDataList = fateService.queryStatus(FateDataStatusEnum.USING.getCode());
        Cat.logEvent("updateTmpData","equalFailed:"+"updateUsing:"+updateUsing+"--fateDataList:"+fateDataList.size());
        LOGGER.info("queryStatus USING:{}",fateDataList.size());
        equalFlag = true;
    }

    // 數(shù)據(jù)數(shù)目不相等,則需要數(shù)據(jù)不為空再繼續(xù)
    while(fateDataList.isEmpty()){
        // 等待主從延遲
        Thread.sleep(1000);
        fateDataList = fateService.queryStatus(FateDataStatusEnum.USING.getCode());
        Cat.logEvent("updateTmpData","queryFailed:"+"updateUsing:"+updateUsing+"--fateDataList:"+fateDataList.size());
        LOGGER.info("queryStatus USING:{}",fateDataList.size());
    }
}

知識(shí)補(bǔ)充

Mybatis使用<update>標(biāo)簽怎么返回影響行數(shù)

Mybatis使用<update>標(biāo)簽怎么返回影響行數(shù)

在MyBatis Plus中,使用<update>標(biāo)簽執(zhí)行更新操作時(shí),默認(rèn)情況下是不返回影響行數(shù)的。但是可以通過(guò)配置來(lái)實(shí)現(xiàn)返回影響行數(shù)的功能。
在MyBatis Plus的配置文件(通常是mybatis-plus-config.xml)中,可以添加如下配置項(xiàng):

<configuration>
  <settings>
    <setting name="returnAffectedCount" value="true"/>
  </settings>
</configuration>

通過(guò)設(shè)置returnAffectedCount為true,可以讓MyBatis Plus在執(zhí)行更新操作后返回影響行數(shù)。

另外,如果只需要獲取更新操作的影響行數(shù)而不需要返回具體的更新結(jié)果,可以使用MyBatis Plus提供的UpdateWrapper或者LambdaUpdateWrapper來(lái)進(jìn)行更新操作,并通過(guò)調(diào)用相應(yīng)的方法獲取影響行數(shù)。例如:

UpdateWrapper<User> updateWrapper = new UpdateWrapper<>();
updateWrapper.eq("age", 25);
int affectedCount = userMapper.update(null, updateWrapper);

在上述示例中,使用了UpdateWrapper.eq()方法指定了更新條件,并通過(guò)調(diào)用userMapper.update()方法執(zhí)行更新操作,并返回影響行數(shù)。

總結(jié)起來(lái),在MyBatis Plus中使用<update>標(biāo)簽執(zhí)行更新操作時(shí),默認(rèn)情況下是不返回影響行數(shù)的。但可以通過(guò)配置來(lái)實(shí)現(xiàn)返回影響行數(shù)的功能,或者使用UpdateWrapper或者LambdaUpdateWrapper來(lái)獲取影響行數(shù)。

到此這篇關(guān)于淺析MySQL中主從延遲問(wèn)題的原因與解決方法的文章就介紹到這了,更多相關(guān)MySQL主從延遲內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論