今日見小王同志眉頭微皺,心想這兄臺(tái)必然遇到難題,遂問其故。果不其然,他在通過日志表統(tǒng)計(jì)用戶使用情況時(shí)創(chuàng)建試圖屢屢失敗。
我以前也沒有做過類似的SQL,但又想這實(shí)現(xiàn)總該不難,于是拿來分析,情況如下:
表1-日志表,表結(jié)構(gòu)如下:
ID,F_LOGIN,MTime,ManageName 這ID是主鍵(ID在我看來都是主鍵,下文不再贅述),F_LOGIN是用戶的登陸名縮寫,MTime是用戶的操作時(shí)間,ManageName是用戶操作的模塊名稱
表2-用戶表,結(jié)構(gòu)如下:
ID,F_ORDER,F_LOGIN,F_USERNAME,F_DEPTNAME...,F_ORDER是用戶的順序號(hào),F(xiàn)_LOGIN是用戶的登陸名縮寫,F_USERNAME是用戶的中文名,F_DEPTNAME是用戶所在單位的名稱
表3-部門表,結(jié)構(gòu)如下:
ID,F_DEPTORDER,F_DEPTNAME F_DEPTORDER是部門順序號(hào),F(xiàn)_DEPTNAME是部門名稱。
好了,就是這么三個(gè)表,客戶要求根據(jù)統(tǒng)計(jì)用戶對(duì)每個(gè)模塊的使用次數(shù),并要求按照部門順序進(jìn)行排序,并且統(tǒng)計(jì)結(jié)果排除管理帳號(hào)admin:
怎么辦? 看到小王以前的視圖是:
SELECT 用戶表.F_DEPTNAME, COUNT(*)
AS count, 部門表.F_ORDER
FROM 日志表 INNER JOIN
用戶表 ON
日志表.F_login = 用戶表.F_LOGIN INNER JOIN
部門表 ON
用戶表.F_DEPTNAME = 部門表.F_DEPATNAME
WHERE (日志表.F_login <> 'admin')
GROUP BY 用戶表.F_DEPTNAME,
部門表.F_NO
ORDER BY 部門表.F_NO
郁悶,這試圖看起來沒什么問題啊,但是一運(yùn)行問題就來了:
考,如果部門A的用戶都沒有使用,也就是日志表里沒有記錄,那么視圖里根本就不會(huì)顯示該單位,但是很明顯這樣不對(duì),我們需要沒有使用的單位顯示次數(shù)為0嘛,
我想辦法不是明擺著的嘛,把"INNER JOIN 部門表"改為"RIGHT JOIN"部門表不就ok了么,好,改變:
SELECT 用戶表.F_DEPTNAME, COUNT(*)
AS count, 部門表.F_ORDER
FROM 日志表 INNER JOIN
用戶表 ON
日志表.F_login = 用戶表.F_LOGIN RIGHT JOIN
部門表 ON
用戶表.F_DEPTNAME = 部門表.F_DEPATNAME
WHERE (日志表.F_login <> 'admin')
GROUP BY 用戶表.F_DEPTNAME,
部門表.F_NO
ORDER BY 部門表.F_NO
運(yùn)行,又郁悶,怎么還是沒有出現(xiàn),抓耳撓腮半晌弄不明白,心想反正老子最不怕的就是困難(最怕的是美女放電^_^),我一句一句來,調(diào)試、調(diào)試,終于發(fā)現(xiàn)問題所在:
"WHERE (日志表.F_login <> 'admin')"
當(dāng)Right join以后,沒有操作的部門會(huì)在視圖留下一條記錄,而這條記錄只包含部門表的信息,用戶表和日志表均為NULL,NULL是沒有辦法和'admin'比較的,也就是說NULL <> 'admin' 返回的是false,怎么辦?調(diào)整視圖join的次序,如下:
SELECT 用戶表.F_DEPTNAME, COUNT(*)
AS count, 部門表.F_ORDER
FROM 用戶表 INNER JOIN
部門表 ON
用戶表.F_DEPTNAME = 部門表.F_DEPATNAME LEFT JOIN
日志表 ON
日志表.F_login = 用戶表.F_LOGIN
WHERE (用戶表.F_login <> 'admin')
GROUP BY 用戶表.F_DEPTNAME,
部門表.F_NO
ORDER BY 部門表.F_NO
這樣不管怎么變,這所有用戶和部門都是有的,而且admin也過濾的,但是....不對(duì)啊,怎么沒有用戶的單位使用次數(shù)都很大啊,哦,原來是我用的count(*)
有問題,肯定得用sum函數(shù)啦。查查聯(lián)機(jī)叢書,最后定稿如下:
SELECT 用戶表.F_DEPTNAME,
SUM(CASE WHEN 統(tǒng)計(jì)表.F_login IS NULL THEN 0 ELSE 1 END) as count,
部門表.F_ORDER
FROM 用戶表 INNER JOIN
部門表 ON
用戶表.F_DEPTNAME = 部門表.F_DEPATNAME LEFT JOIN
日志表 ON
日志表.F_login = 用戶表.F_LOGIN
WHERE (用戶表.F_login <> 'admin')
GROUP BY 用戶表.F_DEPTNAME,
部門表.F_NO
ORDER BY 部門表.F_NO
終于搞定了,萬歲!!不過CASE的使用也分兩種,一種是簡單CASE函數(shù),一種是CASE搜索函數(shù),聯(lián)機(jī)從書中關(guān)于when_expression 和Boolean_expression 寫的很籠統(tǒng),我的理解when_expression就是一個(gè)值,而Boolean_expression是一個(gè)判斷,嗯,寫這個(gè)破東西也婆婆媽媽的寫了半個(gè)小時(shí),到此收筆。