《卸甲笔记》-PostgreSQL和Oracle的SQL差异分析之五:函数的差异(二)

本文涉及的产品
云原生数据库 PolarDB MySQL 版,通用型 2核4GB 50GB
云原生数据库 PolarDB PostgreSQL 版,标准版 2核4GB 50GB
简介: PostgreSQL是世界上功能最强大的开源数据库,在国内得到了越来越多机构和开发者的青睐和应用。随着PostgreSQL的应用越来越广泛,Oracle向PostgreSQL数据库的数据迁移需求也越来越多。数据库之间数据迁移的时候,首先是迁移数据,然后就是SQL、存储过程、序列等程序中不同的数据库.

PostgreSQL是世界上功能最强大的开源数据库,在国内得到了越来越多机构和开发者的青睐和应用。随着PostgreSQL的应用越来越广泛,Oracle向PostgreSQL数据库的数据迁移需求也越来越多。数据库之间数据迁移的时候,首先是迁移数据,然后就是SQL、存储过程、序列等程序中不同的数据库中数据的使用方式的转换。下面根据自己的理解和测试,写了一些SQL以及数据库对象转换方面的文章,不足之处,尚请多多指教。

1、to_number函数

to_number用于将字符类型转换成数字。主要使用在显示格式的控制以及排序等地方。特别是排序的时候,因为按照字符排序和按照数字排序,结果是不同的。

Oracle的to_number函数接受两个参数。第一个参数是需要转换的字符串,第二个参数是要转换的格式。实际使用的时候,除非采用特殊的格式显示,否则一个参数就已经足够了。

PostgreSQL的to_number函数也接受两个参数。使用的时候,不能只使用一个参数。排序的时候, 格式字符串中需要设置转换的位数为该字段的最大位数。否则,只使用格式提供的转换位数来排序。比如格式提供了两位,那么就转换最前面的两位字符为数字,然后使用它来排序。

Oracle to_number函数

SQL> desc o_test;
 名称                                      是否为空? 类型
 ----------------------------------------- -------- ----------------------------
 ID                                                 NUMBER
 NAME                                               VARCHAR2(10)
 AGE                                                VARCHAR2(10)

SQL> select * from o_test;

        ID NAME       AGE
---------- ---------- ----------
           赵大       20
           钱二       9
           孙三       30
           李四       110

SQL> select * from o_test order by age;

        ID NAME       AGE
---------- ---------- ----------
           李四       110
           赵大       20
           孙三       30
           钱二       9

SQL> select * from o_test order by to_number(age);

        ID NAME       AGE
---------- ---------- ----------
           钱二       9
           赵大       20
           孙三       30
           李四       110

SQL> select id, name, to_number(age) age from o_test order by to_number(age);

        ID NAME              AGE
---------- ---------- ----------
           钱二                9
           赵大               20
           孙三               30
           李四              110

PostgreSQL to_number函数

postgres=# \d p_test;
                           数据表 "public.p_test"
 栏位 |         类型          |                    修饰词
------+-----------------------+----------------------------------------------
 id   | integer               | 非空 默认 nextval('p_test_id_seq'::regclass)
 name | character varying(10) |
 age  | character varying(10) |

postgres=# select * from p_test;
 id | name | age
----+------+-----
  1 | 赵大 | 20
  2 | 钱二 | 9
  3 | 孙三 | 30
  4 | 李四 | 110
(4 行记录)

postgres=# select * from p_test order by age;
 id | name | age
----+------+-----
  4 | 李四 | 110
  1 | 赵大 | 20
  3 | 孙三 | 30
  2 | 钱二 | 9
(4 行记录)

postgres=# select * from p_test order by to_number(age);
错误:  函数 to_number(character varying) 不存在
第1行select * from p_test order by to_number(age);
                                   ^
提示:  没有匹配指定名称和参数类型的函数. 您也许需要增加明确的类型转换.
postgres=# select * from p_test order by to_number(age,'9999999999');
 id | name | age
----+------+-----
  2 | 钱二 | 9
  1 | 赵大 | 20
  3 | 孙三 | 30
  4 | 李四 | 110
(4 行记录)

postgres=# select * from p_test order by to_number(age,'99');
 id | name | age
----+------+-----
  2 | 钱二 | 9
  4 | 李四 | 110
  1 | 赵大 | 20
  3 | 孙三 | 30
(4 行记录)

postgres=# select id,name, to_number(age,'9999999999') from p_test order by to_number(age,'9999999999');
 id | name | to_number
----+------+-----------
  2 | 钱二 |         9
  1 | 赵大 |        20
  3 | 孙三 |        30
  4 | 李四 |       110
(4 行记录)

2、decode函数

decode是Oracle固有的一个函数,用于条件判断。其格式为
decode(条件, 值1, 返回值1, 值2, 返回值2,... 值n, 返回值n, 缺省值) 。当条件等于值1的时候返回返回值1,······等于值n的时候返回返回值n。都不等于的时候返回缺省值。

PostgreSQL中,decode函数使用来解码的,和encode函数相对。对于Oracle的decode函数,可以把它转换成case......when....的SQL语句,得到一样的效果。
Oracle也支持case....when。用法和PostgreSQL中类似。

Oracle

SQL> select * from o_test;

        ID NAME       AGE
---------- ---------- ----------
           赵大       20
           钱二       9
           孙三       30
           李四       110

SQL> select decode(age, '20', '赵大', '9', '钱二', '张三') testname from o_test;

TEST
----
赵大
钱二
张三
张三

SQL> select case age when '20' then '赵大' when '9' then '钱二' else '张三' end testname from o_test;

TEST
----
赵大
钱二
张三
张三

SQL> select case  when age = '20' then '赵大' when age= '9' then '钱二' else '张三' end testname from o_test;

TEST
----
赵大
钱二
张三
张三

PostgreSQL

postgres=# select * from p_test;
 id | name | age
----+------+-----
  1 | 赵大 | 20
  2 | 钱二 | 9
  3 | 孙三 | 30
  4 | 李四 | 110
(4 行记录)

postgres=#  select case age when '20' then '赵大' when '9' then '钱二' else '张三' end testname from p_test;
 testname
----------
 赵大
 钱二
 张三
 张三
(4 行记录)


postgres=#  select case when age='20' then '赵大' when age= '9' then '钱二' else '张三' end testname from p_test;
 testname
----------
 赵大
 钱二
 张三
 张三
(4 行记录)

postgres=# select encode('abcdefghijklmn', 'base64');
        encode
----------------------
 YWJjZGVmZ2hpamtsbW4=
(1 行记录)

postgres=# select decode('YWJjZGVmZ2hpamtsbW4=', 'base64');
             decode
--------------------------------
 \x6162636465666768696a6b6c6d6e
(1 行记录)

3、instr函数

Oracle的instr函数是查找一个字符串中,另一个字符串所在的位置。如果找不到则返回0。instr一共有四个参数。前两个分别表示源字符串和查找字符串,第三个表示开始位置(<0的时候表示从右望左找)。第四个表示第几次出现的值。
PostgreSQL中,可以使用position(substring in string) 函数来对应它。position函数没有Oracle的那么复杂,有些复杂的功能只能使用自定义函数来实现它。

Oracle instr

SQL> select instr('helloworld', 'l') from dual;

INSTR('HELLOWORLD','L')
-----------------------
                      3

SQL> select instr('helloworld', 'l', 5) from dual;

INSTR('HELLOWORLD','L',5)
-------------------------
                        9

SQL> select instr('helloworld', 'l', -5) from dual;

INSTR('HELLOWORLD','L',-5)
--------------------------
                         4

SQL> select instr('helloworld', 'l', 4, 2) from dual;

INSTR('HELLOWORLD','L',4,2)
---------------------------
                          9

PostgreSQL position

postgres=# select position('l' in 'helloworld');
 position
----------
        3
(1 行记录)

postgres=# select length(substring('helloworld', 1, 4)) + position('l' in substring('helloworld',5));
 ?column?
----------
        9
(1 行记录)

postgres=#  select instr('helloworld', 'l', -5, 1) ;
错误:  函数 instr(unknown, unknown, integer, integer) 不存在
第1行select instr('helloworld', 'l', -5, 1) ;
            ^
提示:  没有匹配指定名称和参数类型的函数. 您也许需要增加明确的类型转换.
postgres=# CREATE FUNCTION instr(string varchar, string_to_search varchar,
postgres(# beg_index integer, occur_index integer)
postgres-# RETURNS integer AS 
$$

postgres$# DECLARE
postgres$# pos integer NOT NULL DEFAULT 0;
postgres$# occur_number integer NOT NULL DEFAULT 0;
postgres$# temp_str varchar;
postgres$# beg integer;
postgres$# i integer;
postgres$# length integer;
postgres$# ss_length integer;
postgres$# BEGIN
postgres$# IF beg_index > 0 THEN
postgres$# beg := beg_index;
postgres$# temp_str := substring(string FROM beg_index);
postgres$#   FOR i IN 1..occur_index LOOP
postgres$# pos := position(string_to_search IN temp_str);
postgres$#             IF i = 1 THEN
postgres$# beg := beg + pos - 1;
postgres$# ELSE
postgres$# beg := beg + pos;
postgres$# END IF;
postgres$#             temp_str := substring(string FROM beg + 1);
postgres$# END LOOP;
postgres$#         IF pos = 0 THEN
postgres$# RETURN 0;
postgres$# ELSE
postgres$# RETURN beg;
postgres$# END IF;
postgres$# ELSE
postgres$# ss_length := char_length(string_to_search);
postgres$# length := char_length(string);
postgres$# beg := length + beg_index - ss_length + 2;
postgres$#         WHILE beg > 0 LOOP
postgres$# temp_str := substring(string FROM beg FOR ss_length);
postgres$# pos := position(string_to_search IN temp_str);
postgres$#             IF pos > 0 THEN
postgres$# occur_number := occur_number + 1;
postgres$#                 IF occur_number = occur_index THEN
postgres$# RETURN beg;
postgres$# END IF;
postgres$# END IF;
postgres$#             beg := beg - 1;
postgres$# END LOOP;
postgres$#         RETURN 0;
postgres$# END IF;
postgres$#
postgres$# END;
postgres$# 
$$
 LANGUAGE plpgsql;
CREATE FUNCTION
postgres=#  select instr('helloworld', 'l', -5, 1) ;
 instr
-------
     4
(1 行记录)

postgres=# select instr('helloworld', 'l', 4, 2);
 instr
-------
     9
(1 行记录)
相关实践学习
使用PolarDB和ECS搭建门户网站
本场景主要介绍基于PolarDB和ECS实现搭建门户网站。
阿里云数据库产品家族及特性
阿里云智能数据库产品团队一直致力于不断健全产品体系,提升产品性能,打磨产品功能,从而帮助客户实现更加极致的弹性能力、具备更强的扩展能力、并利用云设施进一步降低企业成本。以云原生+分布式为核心技术抓手,打造以自研的在线事务型(OLTP)数据库Polar DB和在线分析型(OLAP)数据库Analytic DB为代表的新一代企业级云原生数据库产品体系, 结合NoSQL数据库、数据库生态工具、云原生智能化数据库管控平台,为阿里巴巴经济体以及各个行业的企业客户和开发者提供从公共云到混合云再到私有云的完整解决方案,提供基于云基础设施进行数据从处理、到存储、再到计算与分析的一体化解决方案。本节课带你了解阿里云数据库产品家族及特性。
相关文章
|
18天前
|
SQL Oracle 关系型数据库
[Oracle]面试官:你举例几个内置函数,并且说说如何使用内置函数作正则匹配
本文介绍了多种SQL内置函数,包括单行函数、非空判断函数、日期函数和正则表达式相关函数。每种函数都有详细的参数说明和使用示例,帮助读者更好地理解和应用这些函数。文章强调了字符串操作、数值处理、日期计算和正则表达式的使用方法,并提供了丰富的示例代码。作者建议读者通过自测来巩固学习成果。
15 1
[Oracle]面试官:你举例几个内置函数,并且说说如何使用内置函数作正则匹配
|
2月前
|
关系型数据库 Serverless 定位技术
PostgreSQL GIS函数判断两条线有交点的函数是什么?
PostgreSQL GIS函数判断两条线有交点的函数是什么?
247 60
|
23天前
|
Oracle 安全 关系型数据库
Oracle与GreatSQL差异:更改唯一索引列
【11月更文挑战第1天】本文介绍了在 Oracle 和 GreatSQL 中更改唯一索引列的方法及差异。Oracle 需要手动删除和重建索引,过程复杂且可能影响数据一致性;而 GreatSQL 可以自动维护索引,直接修改列值即可,操作简便且更安全。
|
7天前
|
SQL Oracle 关系型数据库
Oracle与GreatSQL差异:更改唯一索引列
【11月更文挑战第11天】本文介绍了在 Oracle 和 GreatSQL 中修改唯一索引列的操作。Oracle 需要先删除索引、修改列值,再重新创建索引,步骤较为繁琐。而 GreatSQL 在满足一定条件下支持在线 DDL 操作,可以直接修改列值,操作相对简单。两者都需要考虑数据完整性和表上的其他约束条件。
|
1月前
|
SQL Oracle 关系型数据库
SQL优化-使用联合索引和函数索引
在一次例行巡检中,发现一条使用 `to_char` 函数将日期转换为字符串的 SQL 语句 CPU 利用率很高。为了优化该语句,首先分析了 where 条件中各列的选择性,并创建了不同类型的索引,包括普通索引、函数索引和虚拟列索引。通过对比不同索引的执行计划,最终确定了使用复合索引(包含函数表达式)能够显著降低查询成本,提高执行效率。
|
1月前
|
SQL 数据库 数据库管理
数据库SQL函数应用技巧与方法
在数据库管理中,SQL函数是处理和分析数据的强大工具
|
1月前
|
SQL 监控 Oracle
Oracle SQL性能优化全面指南
在数据库管理领域,Oracle SQL性能优化是确保数据库高效运行和数据查询速度的关键
|
1月前
|
SQL 数据库 索引
SQL中COUNT函数结合条件使用的技巧与方法
在SQL查询中,COUNT函数是一个非常常用的聚合函数,用于计算表中满足特定条件的记录数
|
1月前
|
SQL 存储 Oracle
Oracle数据库SQL语句详解与应用指南
在数字化时代,数据库已成为各类企业和组织不可或缺的核心组件。Oracle数据库作为业界领先的数据库管理系统之一,广泛应用于各种业务场景。掌握Oracle数据库的SQL语句是数据库管理员、开发人员及运维人员的基本技能。本文将详细介绍Oracle数据库SQL语句的基本概念、语法、应用及最佳实践。一、Or
56 3
|
1月前
|
SQL Oracle 关系型数据库
Oracle SQL:了解执行计划和性能调优
Oracle SQL:了解执行计划和性能调优
53 1

热门文章

最新文章