国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 數據庫 > Oracle > 正文

Oracle--SQL行列轉換實戰

2024-08-29 13:38:28
字體:
來源:轉載
供稿:網友

  行列轉換實例
  表ttt有三個字段
  seq --序列
  jcxm --檢查項目
  zhi --值
  
  數據分別如下:
  seq   jcxm     zhi
  -------   --------     --------
  11     1    0.50
  11     2    0.21
  11     3    0.25
  12     1    0.24
  12     2    0.30
  12     3    0.22
  
  實現功能
  創建視圖時移動行值為列值
  
  create view v_view1
  as
  select seq,
  sum(decode(jcxm,1, zhi)) 檢測項目1,
  sum(decode(jcxm,2, zhi)) 檢測項目2,
  sum(decode(jcxm,3, zhi)) 檢測項目3
  from ttt
  group by seq;
  
  序號 檢測項目1  檢測項目2  檢測項目3
  11     0.50    0.21     0.25
  12     0.24    0.30     0.22
  
  技巧:
  用THEN中的0和1來進行統計(SUM)
  
  jcxm  zhi
  ----  ----
  a      1
  b      1
  a      3
  d      2
  e      4
  f      5
  a      5
  d      3
  d      6
  b      5
  c      4
  b      3
  求他的zhi既是1,也是3,也是5的jcxm
  方法一
  select jcxm
  from ttt
  group by jcxm
  having sum(decode(zhi,1,-1,3,-1,5,-1,0)) = -3
  方法二
  select jcxm from ttt
  group by jcxm having (sign(sum(decode(zhi,1,-1,0)))+
  sign(sum(decode(zhi,3,-1,0)))+sign(sum(decode(zhi,5,-1,0)))<=-3);
  
  ----------
  a
  b
  說明:
  sign()函數根據某個值是0、正數還是負數,分別返回0、1、-1
  所以可以用sign和decode來完成比較字段大小來區某個字段
  select decode(sign(字段1-字段2),-1,字段3,字段4) from dual;
  
  sign是一個對于寫分析SQL有很強大的功能
  下面我對sign進行一些總結:
  但屬性student取0和1以外的值,或者student取兩個以上的標法值,問題就不會這么簡單了
  解決辦法就是特征函數(abs(),sign())
  
  常用的特征算法
 ?。跘=B]=1-abs(sign(A-B))
 ?。跘?。紹]=abs(sign(A-B))
  [A<B]=1-sign(1+sign(A-B)) 不能用-sign(A-B):因為假如不滿足A ?。跘<=B]=sign(1-sign(A-B))
 ?。跘>B]=1-sign(1-sign(A-B))
 ?。跘>=B]=sign(1+sign(A-B)))
 ?。跱OTα]=1-d [α]
 ?。郐罙NDb ]=d [α]*d [b ] (6)
  [αOR b ]=sign(d [α]+d [b ])
  
  例如:
  A<B             Decode( Sign(A-B), -1, 1, 0 )
  A<=B             Decode( Sign(A-B), 1, 0, 1 )
  A>B             Decode( Sign(A-B), 1, 1, 0 )
  A>=B             Decode( Sign(A-B), -1, 0, 1 )
  A=B             Decode( A, B, 1, 0 )
  A between B and C   Decode( Sign(A-B), -1, 0, Decode(Sign(A-C), 1, 0, 1 ))
  A is null            Decode(A,null,1,0)
  A is not null         Decode(A,null,0,1)     A in (B1,B2,...,Bn) Decode(A,B1,1,B2,1,...,Bn,1,0)
  nor LogA          Decode( LogA, 0, 1, 0 )       (1-Sign(LogA))
  LogA and LogB      LogA * LogB
  LogA or LogB       LogA + LogB
  LogA xor LogB      Decode(Sign(LogA),Sign(LogB),0,1)
  Mod(Sign(LogA),Sign(LogB),2
  
  >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
  
  另外一個關于成績的分析例子
  
  SELECT
  SUM(CASE WHEN cj <
60 THEN 1 ELSE 0 END) as "not passed",
  SUM(CASE WHEN cj BETWEEN 60 AND 79 THEN 1 ELSE 0 END) as "passed",
  SUM(CASE WHEN cj BETWEEN 80 AND 89 THEN 1 ELSE 0 END) as "good",
  SUM(CASE WHEN cj >=90 THEN 1 ELSE 0 END) as "Excellent"
  FROM cjtable;
  
  decode用法2
  表、視圖結構轉化
  現有一個商品銷售表sale,表結構為:
  month    char(6)      --月份
  sell    number(10,2)    --月銷售金額
  
  現有數據為:
  200001  1000
  200002  1100
  200003  1200
  200004  1300
  200005  1400
  200006  1500
  200007  1600
  200101  1100
  200202  1200
  200301  1300
  
  想要轉化為以下結構的數據:
  year   char(4)        --年份
  ------------  ---------------------
  month1  number(10,2)   --1月銷售金額
  month2  number(10,2)   --2月銷售金額
  month3  number(10,2)   --3月銷售金額
  month4  number(10,2)   --4月銷售金額
  month5  number(10,2)   --5月銷售金額
  month6  number(10,2)   --6月銷售金額
  month7  number(10,2)   --7月銷售金額
  month8  number(10,2)   --8月銷售金額
  month9  number(10,2)   --9月銷售金額
  month10  number(10,2)   --10月銷售金額
  month11  number(10,2)   --11月銷售金額
  month12  number(10,2)   --12月銷售金額
  
  結構轉化的SQL語句為:
  create or replace view
  v_sale(year,month1,month2,month3,month4,month5,month6,month7,month8,month9,month10,month11,month12)
  as
  select
  substrb(month,1,4),
  sum(decode(substrb(month,5,2),'01',sell,0)),
  sum(decode(substrb(month,5,2),'02',sell,0)),
  sum(decode(substrb(month,5,2),'03',sell,0)),
  sum(decode(substrb(month,5,2),'04',sell,0)),
  sum(decode(substrb(month,5,2),'05',sell,0)),
  sum(decode(substrb(month,5,2),'06',sell,0)),
  sum(decode(substrb(month,5,2),'07',sell,0)),
  sum(decode(substrb(month,5,2),'08',sell,0)),
  sum(decode(substrb(month,5,2),'09',sell,0)),
  sum(decode(substrb(month,5,2),'10',sell,0)),
  sum(decode(substrb(month,5,2),'11',sell,0)),
  sum(decode(substrb(month,5,2),'12',sell,0))
  from sale
  group by substrb(month,1,4);
  
  體會:要用decode /group by/ order by/sign/sum來實現不同報表的生成
  >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
  CASE應用
  
  1    1    部門a    800    男
  2    2    部門b    900    女
  3    3    部門a    400    男
  4    4    部門d    1400    女
  5    5    部門e    1200    男
  6    6    部門f    500    男
  7    7    部門a    300    女
  8    8    部門d    1000    男
  9    9    部門d    1230    女
  10    10    部門b    2000    女
  11    11    部門c    2000    男
  12    12    部門b    1200    男
  
  SELECT jcxm as 部門,COUNT(seq) as 人數,
  SUM(CASE SEX WHEN 1 THEN 1 ELSE 0 END) as 男,
  SUM(CASE SEX WHEN 2 THEN 1 ELSE 0 END) as 女,
  SUM(CASE SIGN(zhi-800) WHEN -1 THEN 1 ELSE 0 END) as 小于800元,
  SUM((CASE SIGN(zhi-800)*SIGN(zhi-1000)          /*用*來實現<
和>功能*/
  WHEN -1 THEN 1 ELSE 0 END)+(CASE zhi
  WHEN 800 THEN 1 ELSE 0 END)) as 從800至999,     /*注重別名不能以數字開頭*/
  SUM((CASE SIGN(zhi-1000)*SIGN(zhi-1200)
  WHEN -1 THEN 1 ELSE 0 END)+(CASE zhi
  WHEN 1000 THEN 1 ELSE 0 END)) as 從1000元至1199元,
  SUM((CASE SIGN(zhi-1200) WHEN 1 THEN 1 ELSE 0 END)
  +(CASE zhi WHEN 1200 THEN 1 ELSE 0 END)) as 大于1200元
  FroM ttt
  GROUP BY jcxm
  
  部門名 人數  男  女  小于800元 從800至999 從1000元至1199元  大于1200元
  部門a 3    2  1    2    1      0        0
  部門b 3    1  2    0    1      0        2
  部門c 1    1  0    0    0      0        1
  部門d 3    1  2    0    0      1        2
  部門e 1    1  0    0    0      0        1
  部門f 1    1  0    1    0      0

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 蒲江县| 临猗县| 彭泽县| 常山县| 长白| 固始县| 张家口市| 泌阳县| 松潘县| 万州区| 兴仁县| 尼玛县| 安龙县| 汉川市| 孝昌县| 临汾市| 恩平市| 宁都县| 团风县| 沾益县| 东港市| 宾阳县| 读书| 万安县| 平武县| 同仁县| 栾城县| 海伦市| 汉寿县| 绥滨县| 逊克县| 平顺县| 和政县| 青海省| 西乌珠穆沁旗| 青铜峡市| 新蔡县| 乌海市| 石景山区| 海门市| 新竹县|