怎样写出可在各种数据库间移植的SQL

简介: 国际标准没有对这些函数做规定,这种 SQL 语句就会和数据库相关而无法移植了,而这些函数在应用开发中还特别常见。Hibernate 能将包括这些函数的标准 HQL 语句转换成不同数据库的 SQL,但 HQL 支持的函数太少,碰到不认识的函数还是要分别注册,这就丧失了可移植性;而且 HQL 能生成的 SQL 本身也比较简单,覆盖面太窄了。

任意的SQL语句都可移植是做不到的。因为各种数据库的功能并不一样,某些数据库的语法在另一种数据库中根本就没有对应的实现机制,这时当然就不可能移植了。

如果我们不用数据库的特殊功能,只在国际标准的范围内,也就是只使用所有数据库的功能交集,其实也相当丰富了,这总该能移植了吧。

基本的情况确实是可以,比如

select * from employee where age>50
select area,sum(amount) from orders group by area

在所有数据库中都能正常执行。

但是,还有很多SQL函数在各种数据库中的写法不一样,特别是字符串和日期相关的,比如

MySQL: year( x )
Oracle: extract( year from x )

国际标准没有对这些函数做规定,这种SQL语句就会和数据库相关而无法移植了,而这些函数在应用开发中还特别常见。

Hibernate能将包括这些函数的标准HQL语句转换成不同数据库的SQL,但HQL支持的函数太少,碰到不认识的函数还是要分别注册,这就丧失了可移植性;而且HQL能生成的SQL本身也比较简单,覆盖面太窄了。

转换SQL语句中的不同的函数,esProc SPL是个更好的解决方案。

esProc SPL是个Java写的开源软件,在这里 https://github.com/SPLWare/esProc。

SPL中有个sqltranslate函数,能够使用了某些“标准”的函数语法书写的SQL翻译成各种数据库的函数。比如:

sql = "select * from Orders where year(OrderDate)=2000"
sql.sqltranslate("MySQL") -> "select * from Orders where year(OrderDate)=2000"
sql.sqltranslate("Oracle") -> "select * from Orders where extract( year from OrderDate )=2000"

sqltranslate中已经预定义了很多数据库中很多函数的写法,比如:

sql = "select * from Orders where ADDDAYS(OrderDate,3)>ShipDate"
sql.sqltranslate("MySQL") -> "select * from Orders where (OrderDate + INTERVAL 3 DAY)>ShipDate"
sql.sqltranslate("Oracle") -> "select * from Orders where (OrderDate + NUMTODSINTERVAL(3, "DAY")>ShipDate"
sql.sqltranslate("DB2") -> "select * from Orders where (OrderDate + 3 DAYS )>ShipDate"

如果碰到了新的函数或新的数据库,esProc还允许程序员自己在配置表中增加。

esProc提供了JDBC接口,很容易被集成进Java应用以实现SQL移植。这里有完整的使用方法:SPL:跨数据库移植 SQL

esProc当然并不是仅仅为了移植SQL而设计的,SPL是功能强大的结构数据处理语言,转换SQL只是SPL顺便实现的一点点小功能而已,SPL更多的作用在于替代SQL实现复杂的查询运算。

当计算需求复杂时我们会用到某些数据库特有的语法或函数。比如Oracle有个KEEP函数可以方便地计算每组的第一条/最后一条记录,其它很多数据库都没有。如果SQL语句中用到了这个函数,就不能再用上面的简单办法移植到其它数据库上了。这种情况下,我们可以只用基本的SQL读取数据,而把更复杂的运算交给SPL来做,SPL是完全不依赖于数据库的,这样写出来的代码就可以继续拥有可移植性了。

select Area, max(Amount) KEEP( dens_rank first order by extract( month from OrderDate) )
from Orders where extract( year from OrderDate)=2000 group by Area

这条Oracle的SQL语句可以用SPL改成这样:





A
B


1
'select OrderDate, Area, Amount from Orders where year(OrderDate)=2000
/读数SQL


2
=A1.sqltranslate("Oracle")
/转成目标数据库的SQL


3
=db.query@x(A2)
/读出数据


4
=A3.group(Area;~.minp@a(month(OrderDate)).max(Amount))
/用SPL计算目标结果


(SPL代码写在格子里,这和普通程序语言很不像,参考这里:写在格子里的程序语言

SPL拥有所有SQL的运算能力(过滤、分组、连接等),并且都是自行实现的(不是翻译成SQL),不依赖于任何数据库,保证这个代码可以在任何数据库上正确执行,就可以移植了。

事实上,SPL提供的运算能力远比SQL更强大丰富(所以也不可能翻译成SQL执行)。有些即使可以用SQL写出来的复杂运算,用SPL也会更简单。比如我们经常举例的任务:计算一支股票最长连续上涨的天数,SQL要写成多层嵌套,冗长且难懂:

    select max(ContinuousDays) from (
        select count(*) ContinuousDays from (
            select sum(UpDownTag) over (order by TradeDate) NoRisingDays from (
                select TradeDate,case when Price>lag(price) over ( order by TradeDate) then 0 else 1 end UpDownTag from Stock ))
        group by NoRisingDays )

SPL提供了更多的基础函数,同样的计算逻辑,写起来要简单得多:

    Stock.sort(TradeDate).group@i(Price<Price[-1]).max(~.len())

这样的代码,是不是既开发简单又容易移植了?

相关文章
|
15天前
|
SQL Oracle 数据库
使用访问指导(SQL Access Advisor)优化数据库业务负载
本文介绍了Oracle的SQL访问指导(SQL Access Advisor)的应用场景及其使用方法。访问指导通过分析给定的工作负载,提供索引、物化视图和分区等方面的优化建议,帮助DBA提升数据库性能。具体步骤包括创建访问指导任务、创建工作负载、连接工作负载至访问指导、设置任务参数、运行访问指导、查看和应用优化建议。访问指导不仅针对单条SQL语句,还能综合考虑多条SQL语句的优化效果,为DBA提供全面的决策支持。
44 11
|
2月前
|
SQL 缓存 监控
大厂面试高频:4 大性能优化策略(数据库、SQL、JVM等)
本文详细解析了数据库、缓存、异步处理和Web性能优化四大策略,系统性能优化必知必备,大厂面试高频。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:4 大性能优化策略(数据库、SQL、JVM等)
|
29天前
|
SQL 关系型数据库 MySQL
MySQL导入.sql文件后数据库乱码问题
本文分析了导入.sql文件后数据库备注出现乱码的原因,包括字符集不匹配、备注内容编码问题及MySQL版本或配置问题,并提供了详细的解决步骤,如检查和统一字符集设置、修改客户端连接方式、检查MySQL配置等,确保导入过程顺利。
|
28天前
|
SQL 监控 安全
SQL Servers审核提高数据库安全性
SQL Server审核是一种追踪和审查SQL Server上所有活动的机制,旨在检测潜在威胁和漏洞,监控服务器设置的更改。审核日志记录安全问题和数据泄露的详细信息,帮助管理员追踪数据库中的特定活动,确保数据安全和合规性。SQL Server审核分为服务器级和数据库级,涵盖登录、配置变更和数据操作等事件。审核工具如EventLog Analyzer提供实时监控和即时告警,帮助快速响应安全事件。
|
2月前
|
SQL 关系型数据库 MySQL
体验使用DAS实现数据库SQL优化,完成任务可得羊羔绒加厚坐垫!
本实验介绍如何通过数据库自治服务DAS对RDS MySQL高可用实例进行SQL优化,包含购买RDS实例并创建数据库、数据导入、生成并优化慢SQL、执行优化后的SQL语句等实验步骤。完成任务,即可领取羊羔绒加厚坐垫,限量500个,先到先得。
172 12
|
2月前
|
SQL 存储 BI
gbase 8a 数据库 SQL合并类优化——不同数据统计周期合并为一条SQL语句
gbase 8a 数据库 SQL合并类优化——不同数据统计周期合并为一条SQL语句
|
2月前
|
SQL 数据库
gbase 8a 数据库 SQL优化案例-关联顺序优化
gbase 8a 数据库 SQL优化案例-关联顺序优化
|
2月前
|
SQL 存储 Linux
从配置源到数据库初始化一步步教你在CentOS 7.9上安装SQL Server 2019
【11月更文挑战第16天】本文介绍了在 CentOS 7.9 上安装 SQL Server 2019 的详细步骤,包括配置系统源、安装 SQL Server 2019 软件包以及数据库初始化,确保 SQL Server 正常运行。
|
2月前
|
SQL 存储 Linux
从配置源到数据库初始化一步步教你在CentOS 7.9上安装SQL Server 2019
【11月更文挑战第8天】本文介绍了在 CentOS 7.9 上安装 SQL Server 2019 的详细步骤,包括系统准备、配置安装源、安装 SQL Server 软件包、运行安装程序、初始化数据库以及配置远程连接。通过这些步骤,您可以顺利地在 CentOS 系统上部署和使用 SQL Server 2019。
|
2月前
|
SQL 存储 Linux
从配置源到数据库初始化一步步教你在CentOS 7.9上安装SQL Server 2019
【11月更文挑战第7天】本文介绍了在 CentOS 7.9 上安装 SQL Server 2019 的详细步骤,包括系统要求检查与准备、配置安装源、安装 SQL Server 2019、配置 SQL Server 以及数据库初始化(可选)。通过这些步骤,你可以成功安装并初步配置 SQL Server 2019,进行简单的数据库操作。