在MySQL中,可以指定按照多個(gè)字段進(jìn)行排序。例如,可以使employee表按照d_id字段和age字段進(jìn)行排序。排序過(guò)程中,先按照d_id字段進(jìn)行排序,遇到d_id字段的值相等的情況時(shí),再把d_id值相等的記錄按照age字段進(jìn)行排序。
創(chuàng)新互聯(lián)公司專(zhuān)注于企業(yè)營(yíng)銷(xiāo)型網(wǎng)站、網(wǎng)站重做改版、邱縣網(wǎng)站定制設(shè)計(jì)、自適應(yīng)品牌網(wǎng)站建設(shè)、H5開(kāi)發(fā)、購(gòu)物商城網(wǎng)站建設(shè)、集團(tuán)公司官網(wǎng)建設(shè)、成都外貿(mào)網(wǎng)站建設(shè)公司、高端網(wǎng)站制作、響應(yīng)式網(wǎng)頁(yè)設(shè)計(jì)等建站業(yè)務(wù),價(jià)格優(yōu)惠性?xún)r(jià)比高,為邱縣等各大城市提供網(wǎng)站開(kāi)發(fā)制作服務(wù)。
查詢(xún)employee表中的所有記錄,按照d_id字段的降序方式和age字段的升序方式進(jìn)行排序。
前幾天工作上遇到一個(gè)問(wèn)題,在mysql數(shù)據(jù)查詢(xún)的時(shí)候,使用的是in條件,而結(jié)果需要根據(jù)in來(lái)進(jìn)行排序,當(dāng)時(shí)嘗試了幾次都沒(méi)成功,后來(lái)在度娘的幫助下,找到了解決辦法:
select * from table_name where id in (6,2,1,10,5,7..)
查詢(xún)的結(jié)果和in中的順序并不一致,也就是說(shuō)在批量查詢(xún)時(shí),mysql的查詢(xún)并不是按照in中的值得順序來(lái)查詢(xún)的。那怎么才能保證和in查詢(xún)中的順序相同呢?
查詢(xún)了資料發(fā)現(xiàn)有兩種方式可以對(duì)in查詢(xún)的結(jié)果進(jìn)行排序。一種是order by find_in_set,另外一種是order by substring_index
1、select * from table_name where id in (6,2,1,10,5,7..) order by find_in_set(id,'6,2,1,10,5,7');
2、select * from table_name where id in (6,2,1,10,5,7..)?order by substring_index('6,2,1,10,5,7',id,1);
對(duì)結(jié)果進(jìn)行排序操作的代價(jià)可能很高,因此可以通過(guò)避免排序或讓參與排序的數(shù)據(jù)行更少來(lái)優(yōu)化查詢(xún)性能。
當(dāng) MySQL 不能使用索引產(chǎn)生有序結(jié)果時(shí),它必須對(duì)數(shù)據(jù)行進(jìn)行排序。這有可能是在內(nèi)存中進(jìn)行也可能是在磁盤(pán)進(jìn)行,但 MySQL 始終將這個(gè)過(guò)程稱(chēng)之為 filesort,即便實(shí)際上并沒(méi)有使用一個(gè)文件。
如果用于排序的值可以一次性放入排序緩存中,MySQL 可以在內(nèi)存中使用快排算法進(jìn)行排序。如果 MySQL 不能在內(nèi)存中進(jìn)行排序,則會(huì)在磁盤(pán)中按塊逐塊排序。它對(duì)每個(gè)塊使用快排算法,然后在將這些排序好的塊合并到結(jié)果中。
有兩個(gè)文件排序(filesort)算法:
很難說(shuō)哪種算法更有效,對(duì)每個(gè)算法來(lái)說(shuō)都會(huì)有最優(yōu)和最壞案例。MySQL 在數(shù)據(jù)表全部列加上用于排序的列的大小不超過(guò) max_length_for_sort_data 時(shí)會(huì)使用單次遍歷算法??梢酝ㄟ^(guò)修改這個(gè)參數(shù)影響排序算法的選擇。
需要注意的是,MySQL 的 filesort使用的臨時(shí)存儲(chǔ)空間可能會(huì)超出你的預(yù)期,這是因?yàn)樗鼘?duì)每個(gè)排序元素都分配了固定大小的存儲(chǔ)空間。這些存儲(chǔ)空間要足夠大以便容下存儲(chǔ)最大的元素,而且 VARCHAR這類(lèi)字段使用的是對(duì)應(yīng)的最大長(zhǎng)度。而且,如果使用的是 UTF-8字符集,MuSQL 會(huì)對(duì)每個(gè)字符分配3個(gè)字節(jié)。結(jié)果是,我們會(huì)發(fā)現(xiàn)那些沒(méi)怎么優(yōu)化的查詢(xún)會(huì)導(dǎo)致磁盤(pán)上的臨時(shí)存儲(chǔ)空間是數(shù)據(jù)表自身存儲(chǔ)空間的好幾倍。
而在對(duì)聯(lián)合查詢(xún)進(jìn)行排序時(shí),MySQL 可能會(huì)在查詢(xún)執(zhí)行過(guò)程中執(zhí)行兩次文件排序。如果 ORDER BY 子句只是引用聯(lián)合查詢(xún)的第一張表,MySQL 可以先對(duì)這個(gè)表進(jìn)行文件排序,然后再處理聯(lián)合查詢(xún)。如果是這種情況,在 EXPLAIN 時(shí)會(huì)在 Extra 字段顯示“Using filesort”。而對(duì)于其他的排序情況——例如排序不是針對(duì)第一張表,或者是 ORDER BY 使用的列對(duì)應(yīng)了不止一個(gè)數(shù)據(jù)表,MySQL 必須使用臨時(shí)表緩存查詢(xún)結(jié)果,然而在聯(lián)合查詢(xún)完成后,再對(duì)臨時(shí)表進(jìn)行文件排序。在這種情況下,EXPLAIN 會(huì)在 Extra 字段顯示“Using temorary; Using filesort”。如果包含 LIMIT 約束的話,會(huì)發(fā)生在文件排序后,因此臨時(shí)表和文件排序的存儲(chǔ)空間可能非常大。
MySQL 5.6在只需要對(duì)數(shù)據(jù)行的子集(例如 LIMIT)進(jìn)行排序時(shí),引入了一個(gè)重大改進(jìn)。相對(duì)于對(duì)整個(gè)結(jié)果集進(jìn)行排序再返回部分?jǐn)?shù)據(jù),MySQL 有時(shí)候會(huì)在排序的時(shí)候直接丟棄掉不需要的數(shù)據(jù)行來(lái)提高效率。不管怎么樣,排序也需要小心使用,很可能會(huì)導(dǎo)致存儲(chǔ)占用的飆升最終導(dǎo)致系統(tǒng)負(fù)荷過(guò)大。