mysql踩坑之limit與sum函數(shù)混合使用問題詳解
今天同事在同步完訂單數(shù)據(jù)后,由于訂單總金額和數(shù)據(jù)源的總金額存在差異,選擇使用LIMIT和SUM()函數(shù)計(jì)算當(dāng)前分頁的總金額來和對方比較特定訂單的總金額,卻發(fā)現(xiàn)計(jì)算出來的金額并不是分頁的訂單總金額,而是所有訂單的總金額。
數(shù)據(jù)庫版本為mysql 5.7,下面會(huì)用一個(gè)示例復(fù)盤遇到的問題。
問題復(fù)盤
本次復(fù)盤會(huì)用一個(gè)很簡單的訂單表作為示例。
數(shù)據(jù)準(zhǔn)備
訂單表建表語句如下(這里偷懶了,使用了自增ID,實(shí)際開發(fā)中不建議使用自增ID作為訂單ID)
CREATE TABLE `order` ( `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '訂單ID', `amount` decimal(10,2) NOT NULL COMMENT '訂單金額', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
插入金額為100的SQL如下(執(zhí)行10次即可)
INSERT INTO `order`(`amount`) VALUES (100);
所以總金額為10*100=1000。
問題SQL
使用limit對數(shù)據(jù)進(jìn)行分頁查詢,同時(shí)使用sum()函數(shù)計(jì)算出當(dāng)前分頁的總金額
SELECT SUM(`amount`) FROM `order` ORDER BY `id` LIMIT 5;
前面也提到了運(yùn)行的結(jié)果,期待的結(jié)果應(yīng)該為5*100=500,然而實(shí)際運(yùn)行的結(jié)果卻為1000.00(帶有小數(shù)點(diǎn)是因?yàn)閿?shù)據(jù)類型)
問題排查
其實(shí)如果對SELECT語句執(zhí)行順序有一定了解的朋友可以很快確定為什么返回的結(jié)果為所有的訂單總金額?下面我會(huì)就問題SQL的執(zhí)行書序來分析問題:
- FROM:FROM子句是最先執(zhí)行的,確定了查詢的是order這張表
- SELECT:SELECT子句是第二個(gè)執(zhí)行的子句,同時(shí)SUM()函數(shù)也在此時(shí)執(zhí)行了。
- ORDER BY:ORDER BY子句是第三個(gè)執(zhí)行的子句,其處理的結(jié)果只有一個(gè),就是訂單總金額
- LIMIT:LIMIT子句是最后執(zhí)行的,此時(shí)結(jié)果集中只有一個(gè)結(jié)果(訂單總金額)
補(bǔ)充內(nèi)容
這里補(bǔ)充一下SELECT語句執(zhí)行順序
- FROM <left_table>
- ON <join_condition>
- <join_type> JOIN <right_table>
- WHERE <where_condition>
- GROUP BY <group_by_list>
- HAVING <having_condition>
- SELECT
- DISTINCT <select_list>
- ORDER BY <order_by_condition>
- LIMIT <limit_number>
解決辦法
遇到需要統(tǒng)計(jì)分頁數(shù)據(jù)時(shí)(除了SUM()函數(shù)外,常見的COUNT()、AVG()、MAX()、MIN()函數(shù)也存在這個(gè)問題),可以選擇使用子查詢來處理(PS:這里不考慮內(nèi)存計(jì)算,針對的是使用數(shù)據(jù)庫解決這個(gè)問題)。上面的問題解決方案如下:
SELECT SUM(o.amount) FROM (SELECT `amount` FROM `order` ORDER BY `id` LIMIT 5) AS o;
運(yùn)行的返回值為500.00。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對本站的支持。
版權(quán)聲明:本站文章來源標(biāo)注為YINGSOO的內(nèi)容版權(quán)均為本站所有,歡迎引用、轉(zhuǎn)載,請保持原文完整并注明來源及原文鏈接。禁止復(fù)制或仿造本網(wǎng)站,禁止在非www.sddonglingsh.com所屬的服務(wù)器上建立鏡像,否則將依法追究法律責(zé)任。本站部分內(nèi)容來源于網(wǎng)友推薦、互聯(lián)網(wǎng)收集整理而來,僅供學(xué)習(xí)參考,不代表本站立場,如有內(nèi)容涉嫌侵權(quán),請聯(lián)系alex-e#qq.com處理。