Stavo lavorando con una query che ho scritto oggi ho dovuto cambiare il codice dalla WHERE
clausola per usare un filtro IN (elenco di cose) invece di usare qualcosa come
item_desc = 'item 1'
OR item_desc = 'item 2'
OR item_desc = 'item 3'
OR item_desc = 'item 4'
Quanto sopra ha funzionato per 15 minuti e non ha restituito nulla, tuttavia quanto segue mi ha dato il mio set di risultati in 1,5 minuti
item_desc IN (
'item 1'
,'item 2'
,'item 3'
,'item 4'
)
L'ho fatto in SQL e mi chiedo perché IN (elenco di elementi) abbia funzionato molto più velocemente dell'istruzione OR.
- EDIT - SQL Server 2008, mi scuso per non aver inserito questo bit di informazioni in primo luogo.
Ecco la Query nella sua interezza usando le OR
istruzioni:
DECLARE @SD DATETIME
DECLARE @ED DATETIME
SET @SD = '2013-06-01';
SET @ED = '2013-06-15';
-- COLUMN SELECTION
SELECT PV.PtNo_Num AS 'VISIT ID'
, PV.Med_Rec_No AS 'MRN'
, PV.vst_start_dtime AS 'ADMIT'
, PV.vst_end_dtime AS 'DISC'
, PV.Days_Stay AS 'LOS'
, PV.pt_type AS 'PT TYPE'
, PV.hosp_svc AS 'HOSP SVC'
, SO.ord_no AS 'ORDER NUMBER'
--, SO.ent_dtime AS 'ORDER ENTRY TIME'
--, DATEDIFF(HOUR,PV.vst_start_dtime,SO.ent_dtime) AS 'ADM TO ENTRY HOURS'
, SO.svc_desc AS 'ORDER DESCRIPTION'
, OSM.ord_sts AS 'ORDER STATUS'
, SOS.prcs_dtime AS 'ORDER STATUS TIME'
, DATEDIFF(DAY,PV.vst_start_dtime,SOS.prcs_dtime) AS 'ADM TO ORD STS IN DAYS'
-- DB(S) USED
FROM smsdss.BMH_PLM_PtAcct_V PV
JOIN smsmir.sr_ord SO
ON PV.PtNo_Num = SO.episode_no
JOIN smsmir.sr_ord_sts_hist SOS
ON SO.ord_no = SOS.ord_no
JOIN smsmir.ord_sts_modf_mstr OSM
ON SOS.hist_sts = OSM.ord_sts_modf_cd
-- FILTER(S)
WHERE PV.Adm_Date BETWEEN @SD AND @ED
AND SO.svc_cd = 'PCO_REMFOLEY'
OR SO.svc_cd = 'PCO_INSRTFOLEY'
OR SO.svc_cd = 'PCO_INSTFOLEY'
OR SO.svc_cd = 'PCO_URIMETER'
AND SO.ord_no NOT IN (
SELECT SO.ord_no
FRROM smsdss.BMH_PLM_PtAcct_V PV
JOIN smsmir.sr_ord SO
ON PV.PtNo_Num = SO.episode_no
JOIN smsmir.sr_ord_sts_hist SOS
ON SO.ord_no = SOS.ord_no
JOIN smsmir.ord_sts_modf_mstr OSM
ON SOS.hist_sts = OSM.ord_sts_modf_cd
WHERE OSM.ord_sts = 'DISCONTINUE'
AND SO.svc_cd = 'PCO_REMFOLEY'
OR SO.svc_cd = 'PCO_INSRTFOLEY'
OR SO.svc_cd = 'PCO_INSTFOLEY'
OR SO.svc_cd = 'PCO_URIMETER'
)
ORDER BY PV.PtNo_Num, SO.ord_no, SOS.prcs_dtime
Grazie,
OR
come si fa nella query effettiva sopra, si consente il corto circuito del motore. WHERE A AND B OR C
valuterà vero anche se A AND B sono falsi, se C è vero. Se dici WHERE A and B OR C OR D OR E OR F
come fai sopra, AND
puoi prendere in considerazione. La logica equivalente effettivo sarebbe incapsulare le OR
serie di cui sopra in modo parentesi sono trattati come set: WHERE A AND (B OR C OR D OR E)
. Ecco come IN
viene trattato un.
AND
viene gestita in precedenza OR
, quindi la query sopra è equivalente al WHERE (OSM.ord_sts = 'DISCONTINUE' AND SO.svc_cd = 'PCO_REMFOLEY') OR SO.svc_cd = 'PCO_INSRTFOLEY' OR SO.svc_cd = 'PCO_INSTFOLEY' OR SO.svc_cd = 'PCO_URIMETER'
che significa che se una delle ultime 3 condizioni è vera, sarà in grado di cortocircuitare il resto della valutazione.