decode

Oracle公司提供的函數

DECODE是Oracle公司獨家提供的功能(ps:informix資料庫也提供該函數),它是一個功能很強的函數。它雖然不是SQL的標準,但對於性能非常有用。

到目前,其他的資料庫供應商還不能提供類似DECODE的功能,甚至有的資料庫的供應商批評Oracle的SQL不標準。實際上,這種批評有些片面或不夠水平。就象有些馬車製造商抱怨亨利·福特的“馬車”不標準一樣。

函數


Decode:
(指令解碼)由於X86指令的長度不一致,必須用一個單元進行"翻譯",真正的內核按翻譯后要求來工作。
1 DECODE 中的if-then-else邏輯
在邏輯編程中,經常用到If – Then –Else 進行邏輯判斷。在DECODE的語法中,實際上就是這樣的邏輯處理過程。它的語法如下:
DECODE(value, if1, then1, if2,then2, if3,then3, . . . else )
Value 代表某個表的任何類型的任意列或一個通過計算所得的任何結果。當每個value值被測試,如果value的值為if1,Decode 函數的結果是then1;如果value等於if2,Decode函數結果是then2;等等。事實上,可以給出多個if/then 配對。如果value結果不等於給出的任何配對時,Decode 結果就返回else 。
需要注意的是,這裡的if、then及else 都可以是函數或計算表達式。
2 DECODE 的簡單例子
Oracle系統中就有許多數據字典是使用decode 思想設計的,比如記錄會話信息的V$SESSION數據字典視圖就是這樣。我們從《Oracle8i/9i Reference》資料中了解到,當用戶登錄成功后在V$SESSION中就有該用戶的相應記錄,但用戶所進行的命令操作在該視圖中只記錄命令的代碼(0—沒有任何操作,2—Insert…),而不是具體的命令關鍵字。因此,我們需要了解當前各個用戶的名字及他們所進行的操作時,要用下面命令才能得到詳細的結果:
select sid,serial#,username,
DECODE(command,
0,’None’,
2,’Insert’,
3,’Select’,
6,’Update’,
7,’Delete’,
8,’Drop’,
‘Other’) command
from v$session where username is not null;
3 DECODE實現表的轉置
資料庫中的表是由列和行構成的一個二維表。一般列在任何資料庫中都是有限的數量,而行的變化較大,如果表很大,行的數量可能大上千萬行。同一列的不同行可能有不同的值,而且不是預先定義的。
例:住房公積金報表置換實例:
1.各個單位在本地經辦行進行開戶,開戶就是將單位的基本信息和職工信息的進行登記;
2.每月各個單位的會計到經辦行交繳本單位的所有職工的住房公積金,系統記錄有每個職工的交繳明細並在每條記錄上記錄有經辦行的代碼;
3.每月、季、半年及年終都要求將經辦行 變為“列”給出個月的明細報表:
經辦行:城西區城東區
月份:
2001.01 xxxx1.xx xxxxx2.xx
2001.02 xxxx3.xx xxxxx4.xx
。 。 。 。 。 。
原來的數據順序是:
城西區2001.01 xxxxx1.xx
城東區2001.01 xxxxx2.xx
城西區2001.02 xxxxx3.xx
城東區2001.02 xxxxx4.xx
住房公積金系統記錄職工的每月交繳名細的pay_lst表結構是:
bank_code varchar2(6)NOT NULL, -- 經辦行代碼
acc_no varchar2(15) not null, -- 單位代碼(單位帳號)
emp_acc_no varchar2(20) not null, -- 職工帳號
tran_date date not null, -- 交繳日期
tran_val Number(7,2) not null, -- 交繳額
sys_date date default sysdate, --系統日期
oper_id varchar2(10) --操作員代碼
這樣的表結構,一般按照將經辦行作為行(row)進行統計是很容易的,但是如果希望將經辦行變為列(column)這樣的格式來輸出就有困難。如果用DECODE函數來處理則變得很簡單:
我們創建一個視圖來對目前的pay_lst表進行查詢。將經辦行代碼變為一些具體的經辦行名稱即可:
CREATE OR REPLACE VIEW bank_date_lst AS
Select to_char(tran_date,’yyyy .mm’),
SUM( DECODE ( bank_code,’001’, tran_val,0 )) 城西區,
SUM( DECODE ( bank_code,’002’, tran_val,0 )) 城南區
SUM( DECODE ( bank_code,’003’, tran_val,0 )) 城東區
FROM pay_lst
GROUP BY to_char(tran_date,’yyyy .mm’);
建立視圖后,可直接對該視圖進行查詢就可按照列顯示出結果。
簡單應用:
如果欄位a值為null,則替換為0
decode(a,Null,0,a) a欄位如果為null,則替換為0,如果不為null,則為a

比較


oracle中有decode函數,如下:
select sum(decode(sex,'男',1,0))男生數 from school;
統計男生數目,含義為:decode()中sex欄位為男時,用1代替,然後計算總和
mysql中沒有該函數,替代的有case與if函數

case函數

同樣統計男女生人數,語句如下:
SELECT school,COUNT(school) 人數,
SUM(CASE
WHEN sex='男'THEN 1
ELSE 0
END)
男生數,
SUM(CASE
WHEN sex='女'THEN 1
ELSE 0
END)
女生數
FROM sch
GROUP BY sex,school
CASE 可能是 SQL 中被誤用最多的關鍵字之一。雖然你可能以前用過這個關鍵字來創建欄位,但是它還具有更多用法。例如,你可以在 WHERE 子句中使用CASE。
首先讓我們看一下 CASE 的語法。在一般的 SELECT 中,其語法如下:
SELECT =
CASE
WHEN THEN
USE pubs
GO
SELECT
CASE
WHEN price IS NULL THEN 'Unpriced'
WHEN price < 10 THEN 'Bargain'
WHEN price BETWEEN 10 and 20 THEN 'Average'
ELSE 'Gift to impress relatives'
END AS Range,
Title
FROM titles
GROUP BY
CASE
WHEN price IS NULL THEN 'Unpriced'
WHEN price < 10 THEN 'Bargain'
WHEN price BETWEEN 10 and 20 THEN 'Average'
ELSE 'Gift to impress relatives'
END,
Title
ORDER BY
CASE
WHEN price IS NULL THEN 'Unpriced'
WHEN price < 10 THEN 'Bargain'
WHEN price BETWEEN 10 and 20 THEN 'Average'
ELSE 'Gift to impress relatives'
END,
Title
GO
注意,為了在 GROUP BY 塊中使用 CASE,查詢語句需要在 GROUP BY 塊中重複 SELECT 塊中的 CASE 塊。

if語句

同樣統計男女生人數,語句如下:
SELECT school,COUNT(school) 人數,SUM(IF(sex='男',1,0)) 男生數,SUM(IF(sex='女',1,0)) 女生數,
SUM(IF(pass=1,1,0)) pass1,SUM(IF(pass=0,1,0)) pass0,
SUM(IF(state=1,1,0)) state1,SUM(IF(state=0,1,0)) state0
FROM sch
GROUP BY sex,school
語法和decode類似。