`
Dreamer_good
  • 浏览: 2218 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Oracle函数介绍:decode

阅读更多

1、Oracle函数介绍:decode

 

Sql代码 

select sum(DECODE(C810000125,'是',1,0))/COUNT(1) 合格率 FROM t581 

统计合格率,如果 C810000125这个字段为“是”结果1,不为是结果为0 

 

还可以这样写:

Sql代码 

select sum(case when C810000125 = '是' then 1 else 0 end)/COUNT(1) 合格率 FROM t581 

 

含义解释: 

decode(条件,值1,返回值1,值2,返回值2,...值n,返回值n,缺省值) 

该函数的含义如下: 

IF 条件=值1 THEN 

RETURN(翻译值1) 

ELSIF 条件=值2 THEN 

RETURN(翻译值2) 

...... 

ELSIF 条件=值n THEN 

RETURN(翻译值n) 

ELSE 

RETURN(缺省值) 

END IF 

 

decode(字段或字段的运算,值1,值2,值3) 

这个函数运行的结果是,当字段或字段的运算的值等于值1时,该函数返回值2,否则返回值3 

当然值1,值2,值3也可以是表达式,这个函数使得某些sql语句简单了许多 

 

使用方法: 

1、比较大小 

select decode(sign(变量1-变量2),-1,变量1,变量2) from dual; --取较小值 

sign()函数根据某个值是0、正数还是负数,分别返回0、1、-1 

例如: 

变量1=10,变量2=20 

则sign(变量1-变量2)返回-1,decode解码结果为“变量1”,达到了取较小值的目的。 

2、此函数用在SQL语句中,功能介绍如下: 

Decode函数与一系列嵌套的 IF-THEN-ELSE语句相似。base_exp与compare1,compare2等等依次进行比较。

如果base_exp和 第i个compare项匹配,就返回第i 个对应的value 。如果base_exp与任何的compare值都不匹配,则返回default。

每个compare值顺次求值,如果发现一个匹配,则剩下的 compare值(如果还有的话)就都不再求值。

一个为NULL的base_exp被认为和NULL compare值等价。如果需要的话,每一个compare值都被转换成和第一个compare 值相同的数据类型,这个数据类型也是返回值的类型。 

Decode函数在实际开发中非常的有用 

结合Lpad函数,如何使主键的值自动加1并在前面补0 

select LPAD(decode(count(记录编号),0,1,max(to_number(记录编号)+1)),14,'0') 记录编号 from tetdmis 

eg: 

select decode(dir,1,0,1) from a1_interval 

dir 的值是1变为0,是0则变为1 

比如我要查询某班男生和女生的数量分别是多少? 

通常我们这么写: 

select count(*) from 表 where 性别 = 男; 

select count(*) from 表 where 性别 = 女; 

要想显示到一起还要union一下,太麻烦了 

用decode呢,只需要一句话 

select sum(decode(性别,男,1,0)),sum(decode(性别,女,1,0)) from 表 

补充:同事遇到一个问题,分组的条件是动态变化的,比如:一组数据最多按A、B、C三种条件分组,但是根据情况这三个条件会动态的参与到分组中,有8中情况,例如:按null,按A,按B,按C,按A、B,按A、C,按B、C,按A、B、C。

Sql如下

select t.e

       ,t.f

      ,decode(param1,'A',t.a,null) as A

      ,decode(param2,'B',t.b,null) as B

,decode(param3,'C',t.c,null)as C

  from test t

 group by

       t.e,

       t.f,

       decode(param1,'A',t.a,null),

       decode(param2,'B',t.b,null),

decode(param3,'C',t.c,null)

 order by t.e

 

 

 

 

 

 

 

 

 

2、诡异的DECODE函数

今天同事遇到下面一个问题: 

order by decode(column_id,1,null,2,null,3,null,column_id);

有个问题,就是当列数大于10列时,column_id 的顺序成10,11,12,13,4,5,6,7,8,9了

 

这个排序的主要目的是让前3列排在后面,这3列的顺序无所谓。

 

对于小于10列的表是没问题的:

 

SQL> create table t(c1 number,c2 number,c3 number,c4number,c5 number);

 

表已创建。 

 

SQL> col column_name format a20

SQL> select column_name,column_id

  2  from  user_tab_columns

  3  where table_name='T'

  4  order bydecode(column_id,1,null,2,null,3,null,column_id);

 

 

COLUMN_NAME           COLUMN_ID

-------------------- ----------

C4                           4

C5                           5

C2                           2

C1                           1

C3                           3

 

但是当表的列数大于10的时候就会混乱了。 

SQL> select column_name,column_id,decode(column_id,1,null,2,null,3,null,column_id) sortcolumn

  2  from  user_tab_columns

  3  where table_name='T'

  4  order bydecode(column_id,1,null,2,null,3,null,column_id)

  5  /

 

 

COLUMN_NAME           COLUMN_IDSORTCOLUMN

-------------------- ---------- --------------------

C10                         10 10

C11                         11 11

C12                         12 12

C4                           4 4

C5                           5 5

C6                           6 6

C7                           7 7

C8                           8 8

C9                           9 9

C3                           3

C2                           2

C1                           1

 

已选择12行。

 

显然ORACLE把SORTCOLUMN列作为为字符类型排序了。

 

加个TO_NUMBER即可解决这个问题。 

SQL> select column_name,column_id,decode(column_id,1,null,2,null,3,null,column_id) sortcolumn

  2  from  user_tab_columns

  3  where table_name='T'

  4  order byto_number(decode(column_id,1,null,2,null,3,null,column_id));

 

COLUMN_NAME           COLUMN_IDSORTCOLUMN

-------------------- ---------- --------------------

C4                           4 4

C5                           5 5

C6                           6 6

C7                           7 7

C8                           8 8

C9                           9 9

C10                         10 10

C11                         11 11

C12                         12 12

C1                           1

C3                           3

C2                           2

 

已选择12行。

 

但是为什么会导致这个问题,DECODE函数为何返回了字符类型。

 

这个问题yangtingkun大师专门写个几篇文章介绍。

有兴趣的可以找找看看。

 

在这里我借花献佛简单稍微说一下:

对于NULL 类型,ORACLE的默认返回类型是VARCHAR。

对于DECODE函数 ORACLE返回的类型依赖于第一个值。

 

如下所示: 

SQL> CREATE TABLE A AS SELECTDECODE(DUMMY,'X',1,'Y','2',DUMMY) C1, <---由于第一个返回的值1是整数类型,因此整个表达式返回整数类型

  2  DECODE(DUMMY,'X','1','Y',2,DUMMY) C2 ,<---由于第一个返回的值'1'是字符类型,因此整个表达式返回CHAR类型

  3  DECODE(DUMMY,'X',NULL,'Y','HUATENG',DUMMY)C3 FROM DUAL; <---由于第一个返回的值是NULL,因此整个表达式返回CHAR类型

 

表已创建。

 

SQL> DESC A

 名称                                    是否为空? 类型

 ----------------------------------------- ------------------------------------

 C1                                               NUMBER

 C2                                               VARCHAR2(1)

 C3                                               VARCHAR2(7)

 

也正是因为DECODE函数的这种依赖于第一次的值类型作为返回类型,对于其他返回的值如果和第一个类型不匹配,可能会让你遇到很蛋疼的问题:

 

SQL> DESC A;

 名称                                    是否为空? 类型

 ----------------------------------------- ------------------------------------

 C1                                               NUMBER

 C2                                               VARCHAR2(1)

 C3                                               VARCHAR2(7)

 

SQL> INSERT INTO A VALUES(2,2,2);

 

已创建 1 行。

 

SQL> SELECT * FROM A;

 

        C1 C2 C3

---------- -- --------------

         1 1

         2 2  2

 

 

SQL> SELECT DECODE(C1,1,1,2,'E',C1) FROM A;

ERROR:

ORA-01722: 无效数字

 

未选定行

 

SQL> SELECT DECODE(C1,1,1,2,'E',C1) FROM A WHERE C1=1;

 

DECODE(C1,1,1,2,'E',C1)

-----------------------

                     1

 

SQL> SELECT DECODE(C1,1,1,2,'E',C1) FROM A WHERE C1=2;

SELECT DECODE(C1,1,1,2,'E',C1) FROM A WHERE C1=2

                      *

第 1 行出现错误:

ORA-01722: 无效数字

 

上面的问题主要是字符'E'无法转为整数类型导致的。

 

 

 

 

 

 

 

 

 

 

 

 

 

3、Decode函数返回类型的确定

今天在QQ上一个朋友发出问题,说min函数返回错误的取值。详细如下:一个数据表列类型为number(6,2),其中有三行记录,分别为0,0.6和1。用min获取最小值,得到0.6。

min是Oracle SQL的一个基础函数,理论上不会出现这样的Bug之类的。下面一起来模拟下实验环境。

1、环境构建

在实验数据库Oracle11g环境下,构建实验数据表t。填入实验数据。

SQL> create table t (num number(6,2));

Table created

SQL> insert into t values (0);

1 row inserted

SQL> insert into t values (0.6);

1 row inserted

SQL> insert into t values (1);

1 row inserted

SQL> commit;

Commit complete

SQL> select * from t;

    NUM

--------

   0.00

   0.60

1.00

实验那位兄弟的说法。

SQL> select min(num) from t;

 MIN(NUM)

----------

        0

SQL> select min(to_number(num)) from t;

MIN(TO_NUMBER(NUM))

-------------------

                 0

没有什么问题,详细问了一下,获取到了SQL结构如下。

SQL> select min(decode(num,-1,null,num)),min(num)from t;

MIN(DECODE(NUM,-1,NULL,NUM))              MIN(NUM)

---------------------------------------- ----------

.6                                               0

果然,诡异的现象发生了。

2、问题分析

一时间还是很唬人的,那么我们先抛开min函数,单独看数据列情况。抽丝剥茧吧。

SQL> select decode(num,-1,null,num),num from t;

DECODE(NUM,-1,NULL,NUM)                      NUM

---------------------------------------- --------

0                                           0.00

.6                                          0.60

1                                           1.00

这里只剩下一个decode函数的使用。从含义上看,当num为-1的时候,返回null值,否则就是原有的num值。但是有两个疑点,首先是0.60是如何转变为.6的呢?其次就是decode函数处理列的列对其方式,数字类型默认是右对齐,只有字符串是左对齐的。难道说经过decode函数处理之后,返回值变成了字符串?

那么,如果decode处理之后,变成了字符串的话,我们调整一个decode的结构,看看是否是由于处理变成字符串造成了问题。

SQL> selectmin(to_number(decode(num,-1,null,num))),min(num) from t;

MIN(TO_NUMBER(DECODE(NUM,-1,NU  MIN(NUM)

------------------------------ ----------

                            0         0

看来原因就在于decode函数使用处理之后,返回数据列是一个字符串类型。但是decode函数命名指定了num列,返回值是什么类型呢?

num是数字肯定没有什么问题?难道说疑点出现在null的返回值类型上?继续实验。

SQL> select decode(num,-1,num,num),num from t;

DECODE(NUM,-1,NUM,NUM)     NUM

---------------------- --------

                    0    0.00

                  0.6    0.60

                    1    1.00

果然,临时取消掉null,decode返回类型就正常。看来真是受到了null的影响。这个时候,笔者思考一个问题,Oracle Decode函数如何确定返回值类型列呢?

SQL> select decode(num,-1,'d',num),numfrom t;

DECODE(NUM,-1,'D',NUM)                       NUM

---------------------------------------- --------

0                                           0.00

.6                                          0.60

1                                           1.00

SQL> select decode(num,-1,'k',num),to_char(num)from t;

DECODE(NUM,-1,'K',NUM)                  TO_CHAR(NUM)

---------------------------------------- ----------------------------------------

0                                       0

.6                                      .6

1                                       1

上面的实验,让我们得出了和null值是相同的效果。这样,我们对decode有下面猜想:

ü       Oracle在调用decode函数的时候,是需要预先确定列的类型,因为毕竟出现在相同的列上;

ü       确定decode返回值类型,是依据参数中第一个条件返回类型。之后所有的返回类型都依据第一个类型进行强制类型转换;

ü       Oracle在第一个条件返回类型为null的时候,默认将其作为字符串处理;

如果三个假设成立,那么所有问题就得到解释。

那个朋友的SQL中,decode函数第一个可选返回值是null,Oracle识别返回类型为字符类型。之后对所有的其他返回值均使用了to_char方法类似的转换逻辑。

那么,往后想一步,如果Oracle decode函数真是依靠第一条件来确定列类型,其他列进行强行转换,那么如果出现不匹配的时候怎么办?

SQL> select decode(num,1,num,'k') from t;

select decode(num,1,num,'k') from t

ORA-01722:无效数字

这个案例中的decode函数,根据第一个前条件取值num是数字类型,那么其他所有都会被强制转换为数字类型。但是我们写定的其他条件取值是’k’,不能进行强制类型转换。于是报错无效数字。

3、问题解决

了解了问题decode的根源,剩下的就好解释了。min函数可以接受字符串和数字。在数字类型时,依据数字类型的比较规则,选择出0是最小值。当接受字符串时,使用的是二进制对比策略。其中.小数点的排序位最小。于是选择出.6作为结果也就不奇怪了。

解决问题的方法很多,笔者推荐的方式是对null进行数字化处理。让Oracle识别为数字类型。

SQL> select min(decode(num,-1,to_number(null),num)),min(num)from t;

MIN(DECODE(NUM,-1,TO_NUMBER(NU  MIN(NUM)

------------------------------ ----------

                            0         0

4、结论

通过这个案例,我们除了重新认识到decode的原理外,还有几个收获。

首先是要重视null值,null在Oracle中是一种很特殊的类型。在运算和函数调用中,都有很多特殊之处。遇到问题,要注意考虑null的因素,是我们解决问题的思路;

其次就是重视函数的本质。Decode是我们常见的函数,但是我们对一些细节缺乏思考研究。比如Decode返回值类型如何确定?这些都是细节,但是细节也反映了我们的能力和修行。

2
0
分享到:
评论

相关推荐

    关于oracle decode函数的用法

    关于oracle decode函数的用法

    Oracle 中 decode 函数用法

    decode(条件,值1,返回值1,值2,返回值2,…值n,返回值n,缺省值) 该函数的含义如下: IF 条件=值1 THEN  RETURN(翻译值1) ELSIF 条件=值2 THEN  RETURN(翻译值2)  ...... ELSIF 条件=值n THEN  RETURN(翻译值n) ...

    Oracle中Decode()函数使用技巧.doc

    Oracle中Decode()函数使用技巧

    oracle中decode()函数使用技巧

    oracle中decode()函数使用技巧 很有帮助的哦

    oracle函数大全

    oracle中的函数集合 SQL中的单记录函数、系统函数。Decode函数的语法结构如下: decode (expression, search_1, result_1) decode (expression, search_1, result_1, search_2, result_2) decode (expression, ...

    Oracle中Decode()函数使用技巧

    Oracle中Decode()函数使用技巧Oracle中Decode()函数使用技巧Oracle中Decode()函数使用技巧

    oracle_function_decode.patch

    博客:PostgreSQL的学习心得和知识总结(四十四)|语法级自上而下完美实现Oracle数据库DECODE函数的实现方案(GreenPlum & AntDB)

    Oracle中Decode()函数的有关用法

    Oracle中Decode()函数的有关用法Oracle中Decode()函数的有关用法

    oracle的decode函数

    DECODE函数相当于一条件语句(IF).它将输入数值与函数中的参数列表相比较,根据输入值返回一个对应值。

    Oracle DECODE函数语法使用介绍

    Oracle DECODE函数功能很强,下面就为您详细介绍Oracle DECODE函数的用法,希望可以让您对Oracle DECODE函数有更多的了解。 Oracle DECODE函数 Oracle DECODE函数是Oracle公司独家提供的功能,它是一个功能很强的...

    Oracle-Decode()函数和CASE语句的比较

    本文讲述了Oracle-Decode()函数和CASE语句的比较。

    ORACLE 列转行 DECODE函数用法

    NULL 博文链接:https://lisanlai.iteye.com/blog/793404

    oracle中decode函数的使用方法

    含义解释:decode(条件,值1,返回值1,值2,返回值2,…值n,返回值n,缺省值) 该函数的含义如下: 代码如下:IF 条件=值1 THEN RETURN(翻译值1)ELSIF 条件=值2 THEN RETURN(翻译值2) ……ELSIF 条件=值n THEN RETURN...

    oracle函数大全.doc

    ORACLE函数大全 ________________________________________ 作者:[本站编辑] 来源:[CSDN] 浏览:[ ] SQL中的单记录函数 1.ASCII 返回与指定的字符对应的十进制数; SQL&gt; select ascii('A') A,ascii('a') a,...

    SQL中 decode()函数简介

    DECODE函数,是ORACLE公司的SQL软件ORACLE PL/SQL所提供的特有函数计算方式,以其简洁的运算方式,可控的数据模型和灵活的格式转换而闻名。 今天看别人的SQL时看这里面还有decode()函数,以前从来没接触到,上网...

    Oracle中DECODE()函数的使用法

    DECODE()函数,它将输入数值与函数中的参数列表相比较,根据输入值返回一个对应值。区别于SQL的其它函数,DECODE函数还能识别和操作空值。本文详细介绍了DECODE函数的语法。

    使用Oracle的Decode函数进行多值判断

    Decode函数的语法结构如下: 代码如下:decode (expression, search_1, result_1)decode (expression, search_1, result_1, search_2, result_2)decode (expression, search_1, result_1, search_2, result_2, …., ...

    Oracle常用的函数大全详细介绍

    Oracle常用函数大全是我初学时总结的知识心得,希望能和大家一起分享,一起学习,共同进步!!!

    oracle函数和分组

    一些關于orcle內嵌的函數和分組,如日期,字答處理(to_char, decode()....)

    Oracle函数大全

    Oracle函数大全,推荐函数decode

Global site tag (gtag.js) - Google Analytics