[英]MySQL - Combining two select statements into one result with LIMIT efficiently
對於約會應用程序,我有幾個表,我需要查詢單個輸出,兩個查詢的LIMIT 10組合。 此刻似乎很難做到,即使單獨查詢它們也不是問題,但LIMIT 10將無法工作,因為數字不准確(例如,不是LIMIT 5和LIMIT 5,一個查詢可能返回0行,而另外10個,視情況而定)。
members table
member_id | member_name
------------------------
1 Herb
2 Karen
3 Megan
dating_requests
request_id | member1 | member2 | request_time
----------------------------------------------------
1 1 2 2012-12-21 12:51:45
dating_alerts
alert_id | alerter_id | alertee_id | type | alert_time
-------------------------------------------------------
5 3 2 platonic 2012-12-21 10:25:32
dating_alerts_status
status_id | alert_id | alertee_id | viewed | viewed_time
-----------------------------------------------------------
4 5 2 0 0000-00-00 00:00:00
想象一下你是凱倫並且剛剛登錄,你應該看到這兩個項目:
1. Herb requested a date with you.
2. Megan wants a platonic relationship with you.
在一個LIMIT為10的查詢中。相反,這里有兩個需要組合的查詢:
1. Herb requested a date with you.
-> query = "SELECT dr.request_id, dr.member1, dr.member2, m.member_name
FROM dating_requests dr
JOIN members m ON dr.member1=m.member_id
WHERE dr.member2=:loggedin_id
ORDER BY dr.request_time LIMIT 5";
2. Megan wants a platonic relationship with you.
-> query = "SELECT da.alert_id, da.alerter_id, da.alertee_id, da.type,
da.alert_time, m.member_name
FROM dating_alerts da
JOIN dating_alerts_status das ON da.alert_id=das.alert_id
AND da.alertee_id=das.alertee_id
JOIN members m ON da.alerter_id=m.member_id
WHERE da.alertee_id=:loggedin_id AND da.type='platonic'
AND das.viewed='0' AND das.viewed_time<da.alert_time
ORDER BY da.alert_time LIMIT 5";
同樣,有時兩個表可能都是空的,或者一個表可能是空的,或者兩個表都是空的(LIMIT 10可以啟動)並按時間排序。 有關如何有效執行此任務的查詢的任何想法? 歡迎提出想法,建議,編鍾和優化。
您可以將多個查詢與UNION
,但UNION
是查詢具有相同的列數。 理想情況下,列不僅在數據類型中,而且在於它們的語義含義; 但是,MySQL並不關心語義,並且會通過強制轉換為更通用的東西來處理不同的數據類型 - 因此,如果有必要,您可以重載列以使每個表具有不同的含義,然后確定更高級代碼中的含義是什么(雖然我不建議這樣做)。
當列數不同時,或者想要從兩個查詢中實現更好/更少重載的數據對齊時,可以在SELECT
語句中插入虛擬文本列。 例如:
SELECT t.cola, t.colb, NULL, t.colc, NULL FROM t;
您甚至可以為第一個表保留一些列,為第二個表保留其他列,這樣它們在其他地方是NULL
(但請記住,列名來自第一個查詢,因此您可能希望確保它們都在那里命名) :
SELECT a, b, c, d, NULL AS e, NULL AS f, NULL AS g FROM t1
UNION ALL -- specify ALL because default is DISTINCT, which is wasted here
SELECT NULL, NULL, NULL, NULL, a, b, c FROM t2;
您可以嘗試以這種方式對齊您的兩個查詢,然后將它們與UNION
運算符組合; 通過將LIMIT
應用於UNION
,您即將實現目標:
(SELECT ...)
UNION
(SELECT ...)
LIMIT 10;
剩下的唯一問題是,如上所述,來自第一個表的10個或更多記錄將“推出”來自第二個表的任何記錄。 但是,我們可以在外部查詢中使用ORDER BY
來解決此問題。
把它們放在一起:
(
SELECT
dr.request_time AS event_time, m.member_name, -- shared columns
dr.request_id, dr.member1, dr.member2, -- request-only columns
NULL AS alert_id, NULL AS alerter_id, -- alert-only columns
NULL AS alertee_id, NULL AS type
FROM dating_requests dr JOIN members m ON dr.member1=m.member_id
WHERE dr.member2=:loggedin_id
ORDER BY event_time LIMIT 10 -- save ourselves performing excessive UNION
) UNION ALL (
SELECT
da.alert_time AS event_time, m.member_name, -- shared columns
NULL, NULL, NULL, -- request-only columns
da.alert_id, da.alerter_id, da.alertee_id, da.type -- alert-only columns
FROM
dating_alerts da
JOIN dating_alerts_status das USING (alert_id, alertee_id)
JOIN members m ON da.alerter_id=m.member_id
WHERE
da.alertee_id=:loggedin_id
AND da.type='platonic'
AND das.viewed='0'
AND das.viewed_time<da.alert_time
ORDER BY event_time LIMIT 10 -- save ourselves performing excessive UNION
)
ORDER BY event_time
LIMIT 10;
當然,現在由您決定在讀取結果集中的每條記錄時確定要處理的行類型(建議您對NULL
值測試request_id
和/或alert_id
;或者可以在結果中添加一個額外的列它明確說明每個記錄來自哪個表,但它應該是等效的,只要那些id
列是NOT NULL
)。
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.