oracle转mysql语句转换

这篇具有很好参考价值的文章主要介绍了oracle转mysql语句转换。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

场景:Oracle数据库 -> TD数据库(腾讯),而TD数据库实质上是对mysql数据库的封装拓展。因此,需要对项目中oracle的sql进行适配性转化,即在sql查询结果不变的前提下,转换成符合mysql语法的sql语句。
tips:其实TD数据库8.0以上版本的语法大部分是可以完美兼容oracle语法的。但是我们的版本是5.7,因此许多函数和语法是不兼容oracle。如果你的td数据库版本在8以上,就不用往下看了。

一、oracle函数 -> mysql函数

1. decode -> case when


-- oracle: decode
select decode(1, 1, 'out1', 2, 'out2') from dual; -- out1
select decode(2, 1, 'out1', 2, 'out2') from dual; -- out2
select decode(3, 1, 'out1', 2, 'out2') from dual; -- null

--mysql: 1. case [col_name] when
select case 1 when 1 then 'out1' when 2 then 'out2' end; -- out1
select case 2 when 1 then 'out1' when 2 then 'out2' end; -- out2
select case 3 when 1 then 'out1' when 2 then 'out2' end; -- null

-- mysql: 2. case when [expr]
select case when 1 > 0  then 'out0' end; -- out0

12的区别在于,1case后只能跟列,而2case when后是可以跟表达式的,适合复杂情况

2. NVL(exp1,exp2) -> ifnull

-- oracle: nvl(param, val)
select nvl(null, 1) from dual; -- 1
select nvl('', 1) from dual;   -- 1
select nvl('0', 1) from dual;  -- 0

-- mysql: ifnull(param, val)
select ifnull(null,1);-- 1
select ifnull('',1);  -- ''
select ifnull('0',1); -- 0

nvl中null''都输出1,是因为oracle中,空值''即为null
select '' from dual;  -- null
而ifnull中''输出'',是与oracle不同的,因此在进行函数替换的时候,要注意,
如果条件值不为''时,才与nvl等价。如数字类型不会为''

如果param可能为''时,需要用case when等价替换,例如:
select case param when '' then 'empty' when null then 'nul' else 'val end';

3. NVL2(exp1,exp2,exp3) -> if(expr1,expr2,expr3)

-- oracle nvl2(exp1,exp2,exp3) 
select nvl2(1, '123', 'null') from dual;  -- 123
select nvl2('', '123', 'null') from dual;  -- null
select nvl2(null, '123', 'null') from dual;  -- null

--mysql if(param,notNullVal,nullVal) 
select if(1, 123, 'null');-- 123
select if('', 123, 'null');-- null
select if(null, 123, 'null');-- null

此时看起来是等同的,那就大错特错了。
接下来看数据库实际情况:

-- oacle: int_col_0 是int类型 值为0 
select nvl2(int_col_0, 123, null) from demo where id = '1'; -- 123
-- mysql: int_col_0 是int类型 值为0 
select if(int_col_0, 123, null) from demo where id = '1'; -- null
这时候两个函数的结果就不一样了。
原因如下:if(param,notNullVal,nullVal)函数中,如果param是varchar类型,当param
值为''时,会输出后值;如果param时int类型,当param值为0时,会输出后值;当param是
decimal类型,当param为0.0时,会输出后者。
得出结论:当param的值为其数据类型的默认值时,if()函数会走后者。
因此,在param数据类型为varchar时,与nvl2函数是等价的。

4. TO_NUMBER(exp) -> CAST(value AS datatype)

-- 这个没什么讲的,可以直接用,在mysql中cast函数用于将数据类型强转为其它数据类型
类似于java。
select CAST('123' AS decimal(5,1) ); -- 123.0

5. to_char
a. to_char(date, dateFormatType) -> date_format(date, dateFormatType)

-- oracle: to_char(date, dateFormatType)
select to_char(sysdate, 'yyyymmdd') from dual; -- 20230222

-- mysql: date_format(date, dateFormatType)
select date_format(now(), '%Y%m%d'); -- 20230222

b. to_char(str) -> CAST(value AS char)


-- oracle to_char
select to_char(sysdate) from dual; -- [varchar2] 22-2月 -23

--mysql: cast(value AS char)
select cast(now() as char ); -- [varcahr]2023-02-22 09:10:02

6. TRUNC() -> DATE()

-- oracle trunc()
SELECT TRUNC(sysdate) from DUAL; -- [DATE]2023-02-21 00:00:00
SELECT TRUNC(to_date('20210101','yyyymmdd')) from DUAL; -- [DATE]2021-01-01 00:00:00

-- mysql date()
SELECT DATE(NOW()); -- [DATE]2023-02-21
SELECT DATE('20210201'); -- [DATE]2021-02-01

7. to_date -> str_to_date

-- oracle: to_date
select to_date('20210101', 'yyyymmdd') from dual; -- 2021-01-01 00:00:00

-- mysql : str_to_date
select str_to_date('20210101', '%Y%m%d'); -- 2021-01-01

8. 常用时间格式转换

oracle  ->  mysql
YYYYmmdd -> %Y%m%d
YYYY-mm-dd -> %Y-%m-%d 
yyyy-mm-dd hh24:mi:ss -> %Y-%m-%d %H:%i:%s

9. 时间计算方式
a. 日期+/-1天

-- oracle 
select to_date('20210101', 'yyyymmdd') - 1 from dual; -- 2020-12-31 00:00:00

-- mysql
select str_to_date('20210101', '%Y%m%d') - 1 from dual; -- 20210100
从结果可以看出是错误的,只是单纯-1,并没有校验时间是否正确。

-- mysql adddate(date, [num])
select adddate(str_to_date('20210101', '%Y%m%d'), -1); -- 2020-12-31

-- mysql date_sub/date_add(date, interval [num] [day/month/year...])
select date_sub(str_to_date('20210101', '%Y%m%d'), interval 1 day ); -- 2020-12-31
select date_add(str_to_date('20210101', '%Y%m%d'), interval 1 day ); -- 2021-01-02

其中interval后可跟如下时间单位:
FRAC_SECOND
SECOND
MINUTE
HOUR
DAY
WEEK
MONTH
QUARTER
YEAR

b. 日期+/-1月/年

-- oracle  add_months(date, num)
select add_months(to_date('20210101','yyyymmdd'), 1) from dual; -- 2021-02-01 00:00:00

-- mysql date_add(date,, interval [num] month)
select date_add(str_to_date('20210101', '%Y%m%d'), interval 1 month ); --2021-02-01

-- mysql 
SELECT TIMESTAMPADD(SQL_TSI_MONTH, 1, str_to_date('20210101', '%Y%m%d')); -- 2021-02-01

tips:
mybatis中使用date_add时,一旦里卖弄interval关键字,总是会sql编译失败,
不知道是否和版本还是什么有关。

使用TIMESTAMPADD函数不会报错。

interval-type间隔类型参数如下:
SQL_TSI_FRAC_SECOND
SQL_TSI_SECOND
SQL_TSI_MINUTE
SQL_TSI_HOUR
SQL_TSI_DAY
SQL_TSI_WEEK
SQL_TSI_MONTH
SQL_TSI_QUARTER
SQL_TSI_YEAR

10. SUBSTR() -> SUBSTRING()

-- 截取字符串:
注意:
-- oracle
select SUBSTR('123456', 1, 3)、SUBSTR('123456', 0, 3) from dual; -- 输出都是123;

-- mysql
select SUBSTRING('123456', 1, 3); -- 输出123  
select  SUBSTRING('123456', 0, 3); --输出空值''

11. 拼接符号|| -> concat(str, str1)

-- mysql concat(str1,str2,...)
select concat('1','2','3','4','A'); -- 1234A

12. length() -> char_length()

-- oracle length()
select char_length(''); -- null
select LENGTH('123'), '123' from dual; -- 3
select LENGTH(0.00), 0.00 from dual; -- 1
select LENGTH(0), 0 from dual; -- 1
select LENGTH(0.000001), 123456 from dual; -- 7
select LENGTH(sysdate), sysdate from dual; -- 9 2023-02-22 01:23:39

-- mysql char_length
select char_length(''); -- 0
select char_length(123); -- 3
select char_length(0); -- 1
select char_length(0.0); -- 3
select char_length(0.000001); -- 8
select char_length(now()), now(); -- 19  2023-02-22 09:18:34

tips:
可以看出,oracle的length()对于字符串的长度计算结果是符合我们预期的,但是数字类型和并不会把
小数点的长度加上。且当为0.0000时,会自动转会为0再做长度计算,得出结果为1.时间格式
时掺长度我也不确定是怎么得出的。
mysql的char_length()结果就比较符合我们的期望。会把所有参数都转化为char类型后,再进行
长度计算。

注意:当字符串计算长度时,如果字符串为空'',oracle-length()的结果为null,mysql-
char_length的结果为0

二、oracle语法 -> mysql语法

有些语法也不是通用的,下面是经常见到的几种。
1. 临时表with as
解决方案1:换成子查询放到表调用处
解决方案2:在java中拆分,减少sql复杂度。

2. 左右连接(+)写法
解决方案:根据原sql where后的表连接条件,有(+)的换成left join,没有的换成inner join

3. 递归 connect by prior
解决方案:这块内容比较多,另开了一篇:[‘@实现mysql递归查询’]
4. merge语法
解决方案:sql拆成查询、更新、插入三个语句。在java中组合调用文章来源地址https://www.toymoban.com/news/detail-403329.html

到了这里,关于oracle转mysql语句转换的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 数据库去重(MYSQL和ORACLE)

    一、数据库中的去重操作(删除数据库中重复记录的SQL语句)主要有三种方法 (1)、rowid方法 (2)、group by 方法 (3)、distinct方法 1、用rowid方法 根据Oracle带的rowid属性,可以进行判断是否存在重复语句; (1)、查出表1和表2中name相同的数据 Select * from table1 a Where rowid !=

    2024年02月09日
    浏览(49)
  • 查询数据库空间(mysql和oracle)

    1.查看数据库表空间文件: 2.查看所有表空间的总容量: 3.查看数据库表空间使用率 4.1.查看表空间总大小、使用率、剩余空间 4.2.查看表空间使用率(包含temp临时表空间) 5.查看具体表的占用空间大小 1.更改表空间的dbf数据文件分配空间大小 2. 为表空间新增一个数据文件(表空间

    2024年02月05日
    浏览(47)
  • DBeaver连接mysql、oracle数据库

    1) 下载DBeaver https://dbeaver.io/download/,并安装 2) 新建数据库连接    3)选择mysql驱动程序  4)填写连接设置内容 5)点击 “编辑驱动设置”,并填写相关信息  6)选择本地自己下载的驱动库jar包 (也可以点击下载驱动)  7)完成后,点击测试连接,可以查看是否连接成功

    2024年02月05日
    浏览(68)
  • Oracle数据库SQL*Plus命令行执行SQL语句时,中文乱码报错解决方法

    🎉欢迎来到Java学习路线专栏~Oracle数据库SQL*Plus命令行执行SQL语句时,中文乱码报错解决方法 ☆* o(≧▽≦)o *☆嗨~我是IT·陈寒🍹 ✨博客主页:IT·陈寒的博客 🎈该系列文章专栏:Java学习路线 📜其他专栏:Java学习路线 Java面试技巧 Java实战项目 AIGC人工智能 数据结构学习 🍹

    2024年01月22日
    浏览(57)
  • oracle转mysql语句转换

    场景:Oracle数据库 - TD数据库(腾讯),而TD数据库实质上是对mysql数据库的封装拓展。因此,需要对项目中oracle的sql进行适配性转化,即在sql查询结果不变的前提下,转换成符合mysql语法的sql语句。 tips:其实TD数据库8.0以上版本的语法大部分是可以完美兼容oracle语法的。但是

    2023年04月08日
    浏览(33)
  • 数据库问题记录(粗略版)oracle、mysql等主流数据库通用

    1. ORA-00918:未明确定义列 该问题情况大致为:select 所取列名错误、重复等问题。 2. “select * from temp where 1=0; ”的含义 布尔值为FALSE,只返回表结构,不返回数据。 举一反三: select * from temp where 10 , 布尔值为TRUE,返回所有数据记录; select * from temp where 1=0, 暂不清楚是何

    2024年02月07日
    浏览(50)
  • 从 Oracle 到 MySQL 数据库的迁移之旅

    目录 引言 一、前期准备工作 1.搭建新的MySQL数据库 2 .建立相应的数据表 2.1 数据库兼容性分析 2.1.1 字段类型兼容性分析 2.1.2 函数兼容性分析 2.1.3 是否使用存储过程?存储过程的个数?复杂度? 2.1.4 是否使用触发器?个数?使用的场景? 2.2 建表过程中其他需要注意的事项

    2024年04月11日
    浏览(54)
  • Mysql和Oracle数据库死锁查看以及解决

    一、Mysql数据库死锁排查 SQL : SQL: SQL: 示例: 备注:通过查看事务的trx_started(开始时间)来判断该事务的阻塞时间。 SQL  : 备注:线程ID指的是 1.1.3步骤中查询出来的 trx_mysql_thread_id。 SQL : 示例 : 字段解析 : SQL : 示例: SQL : 备注:多个session用逗号隔开。 View Code   造

    2024年01月20日
    浏览(46)
  • 数据库加密AES 适用 Mysql Oracle 以及java

    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 备忘录:加密key为:vU4C!~0_HQtlU_p@ 可采用任意长度为16位的字符串 数据库数据需要对敏感数据进行加密,比如手机号,身份证号以及银行卡号等进行加密。需要一套通用的加密方法可以在java项目以及or

    2024年02月15日
    浏览(50)
  • 查询Oracle和MySQL数据库中当前所有连接信息

    查询Oracle当前所有连接信息: 查询MySQL当前所有连接信息: 在这两个查询中,我为每个字段添加了中文别名,以提高查询结果的可读性

    2024年02月12日
    浏览(72)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包