|
我有一个名为“工作”的表。对于特定用户,作业可以是活动的,已归档的,过期的,待处理的或已关闭的。现在,每个页面请求都生成5个COUNT查询,为了进行优化,我试图将其减少为单个查询。到目前为止,这是我所拥有的,但这仅比5个单独的查询要快。请注意,我已经简化了每个子查询的条件,以使其更易于理解,但是完整查询的行为相同。5 @& y8 x( U9 G" Z5 C. w/ [8 n1 x
有没有一种方法可以在不使用无效子查询的情况下在同一查询中获得这5个计数?( K: ^6 D3 R3 t9 Y3 z
SELECT
0 N2 t; _. a% Q; E( D (SELECT count(*)3 e7 ~% w7 W: S+ `2 G
FROM "jobs"$ O- y* ~+ O2 g- y3 R
WHERE
7 q8 g# v! E6 J6 h1 m4 I; z6 R jobs.creator_id = 5 AND, H3 Z+ o. A' y$ h8 H
jobs.status_id NOT IN (8,3,11) /* 8,3,11 being 'inactive' related statuses */& v1 k2 r0 q. W
) AS active_count,
. P+ l! A( ?( J6 v. ~9 l (SELECT count(*)
: D2 [& A; I/ d c; z5 r FROM "jobs"
- c+ b$ e( D; D/ p* I: Q" Q WHERE
- C9 O' k: B- E jobs.creator_id = 5 AND
! D e6 o& l! E8 m6 O, o H jobs.due_date 继续进行另外2个子查询,但我认为您明白了。
0 B; z6 _9 N, c! ~7 ?; V是否有一种更简单的方法来收集此数据,因为它与Jobs表中的同一数据子集基本相差5个不同的COUNT?
0 P8 {+ E% u8 H! d/ g数据子集为“ creator_id =
, e/ K# Y2 j. ^7 ]3 l( c5”,此后每个计数基本上只是1-2个附加条件。请注意,现在我们正在使用Postgres,但可能会在不久的将来转向MySQL。因此,如果您可以提供与ANSI兼容的解决方案,我将不胜感激
. I' N- X8 \; j7 j% u
2 k. Q- C$ n7 {4 G* [( |, g9 _) y' x解决方案:
2 O3 ?2 Z: T( `( Y9 l, l: ?; `
# H8 ]' p h" q- N* ]2 g/ X
; z: m6 l. q" i' ~2 E' i
: c% A# ?( n: ~- k5 w- E 这是典型的解决方案。使用case语句来分解不同的条件。如果一条记录满足,则得到1,否则得到0。然后SUM对值执行a
: E: A! `& G- u, Z9 s5 V, m SELECT
0 z* I8 s- g3 G4 q. s SUM(active_count) active_count,
4 M( ]! t: K: o8 F; g+ p) H. Z SUM(overdue_count) overdue_count
" |3 S, A* V5 H9 [6 X2 l4 b SUM(due_today_count) due_today_count
2 f8 N; a: v4 R+ W$ W+ M FROM # {" y [2 |3 j/ m) v
(, H, w) v% R/ y
SELECT
# c( |. k. Z2 h# V% k/ l2 E CASE WHEN jobs.status_id NOT IN (8,3,11) THEN 1 ELSE 0 END active_count,
( [" `" A* a! m' H& Z CASE WHEN jobs.due_date UPDATE如前所述,当0记录作为t返回时,此结果将作为所有值中Null的单个结果。您有三个选择8 q3 u; r8 F- T! r1 Q) I) B
1)添加一个Haveing子句,以便您没有返回记录,而不是所有NULLS的结果4 C: ]% Q+ h* E9 o! M/ `8 C2 K- G. Y1 F
HAVING SUM(active_count) is not null
4 G y( t; E. N6 l* N/ y6 P+ h( O2)如果您希望返回所有零,则可以对所有总和加总# j; w# c! C6 V* ?. i0 e4 O
例如6 e2 T, F' q: G/ V, g* O5 x
SELECT
8 M% _3 I8 s" g; h) Q, Q COALESCE(SUM(active_count)) active_count,$ }* M9 v; |% m( \! E0 ]) e
COALESCE(SUM(overdue_count)) overdue_count
3 V4 u6 A& _0 C- C# B COALESCE(SUM(due_today_count)) due_today_count! j. ?" [5 t0 g3 S/ X/ X3 V+ Y: t
3)利用COUNT(NULL) = 0sbarro所展示的事实。您应该注意,非空值可以是任何数字,而不必为1
( j: m4 Y2 `/ C3 y例如
- N# W: U- g8 F7 M+ V: J+ Q SELECT
! M; `' z! u) h4 a/ L6 k COUNT(CASE WHEN
0 G( @2 |! i1 j0 z jobs.status_id NOT IN (8,3,11) THEN 'Manticores Rock' ELSE NULL
& X+ I+ M4 G- p. M' v END) as [active_count] |
|