Oracle 与 PostgreSQL 函数行为的差异引发性能差异
对于Oracle,对于数据修改的操作通过存储过程处理,而对于函数一般不进行数据修改操作。同时,函数可以通过 Select 进行调用,而存储过程则不行。
一、对于volatile 函数的行为
1、Oracle 行为
创建函数:
create or replace function fun01 return number as
begin
insert into t1 values(1);
return 1;
end; create or replace function fun02 return number as
v_cnt number;
begin
select count(*) into v_cnt from t1;
return v_cnt;
end;
(1)、函数有修改操作,不允许select 调用
有修改操作,不允许 select 调用:
SQL> select fun01() from dual;
select fun01() from dual
*
ERROR at line 1:
ORA-14551: cannot perform a DML operation inside a query
ORA-06512: at "SYS.FUN01", line 3
没有修改操作,可以select 调用:
SQL> select fun02() from dual; FUN02()
----------
2
有修改操作的函数,只能通过call 调用
SQL> var a number;
SQL> call fun01() into :a; Call completed.
(2)、视图包含函数的情景
create or replace view v_t2 as select id, fun01() as t1_cnt from t2; test=# select count(*) from v_t2;
count
-------
2
(1 row)
经验证,虽然 fun01 含有 insert 操作,但实际函数(fun01)是没有调用的,也就是函数内的 insert 没有执行。这个个人理解应该是个不严谨的地方。
2、PostgreSQL 行为
创建函数:
create or replace function fun01() returns number as
$$
begin
insert into t1 values(1);
return 1;
end;
$$
language plpgsql;
(1)、函数有修改操作,可以通过Select 调用
test=# select count(*) from t1;
count
-------
3
(1 row) test=# select fun01();
fun01
-------
1
(1 row) test=# select count(*) from t1;
count
-------
4
(1 row)
(3)、视图包含函数的情景
create or replace view v_t2 as select id, fun01() as t1_cnt from t2; test=# select count(*) from v_t2;
count
-------
2
(1 row)
经验证,查询视图时,针对每一行,都会调用fun01 函数一次,也就是 insert 操作实际发生的。
二、对于Stable函数的行为
以上的例子中,由于函数含有insert 操作,因此,函数只能是 volatile。 如果函数是 stable or immutable,那又是何种现象了?
构建数据: t1 表的数据量比较大,count(*) 一次大概要2秒
test=# create table t1(id integer,name text);
CREATE TABLE
test=# create table t2(id integer);
CREATE TABLE insert into t1 select generate_series(1,50000000),'a';
insert into t2 select generate_series(1,10); test=# select count(*) from t1;
count
----------
50000000
(1 row) Time: 2059.901 ms (00:02.060)
创建函数与视图
create or replace function test_volatile(id integer)
returns bigint
volatile
language sql
as
$$ select count(*) from t1 $$ ;
/ create or replace function test_stable(id integer)
returns bigint
stable
language sql
as
$$ select count(*) from t1 $$ ;
/ create or replace function test_immutable(id integer)
returns bigint
immutable
language sql
as
$$ select count(*) from t1 $$ ;
/ create view v_t2_volatile as select id,test_volatile(1) from t2;
create view v_t2_stable as select id,test_stable(1) from t2;
create view v_t2_immutable as select id,test_immutable(1) from t2;
验证视图的访问效率:immutable or stable 类型的函数实际一次都没有执行。
test=# select count(*) from v_t2_immutable;
count
-------
10
(1 row) Time: 1.027 ms test=# select count(*) from v_t2_stable;
count
-------
10
(1 row) Time: 0.950 ms
test=# select count(*) from v_t2_volatile;
count
-------
10
(1 row) Time: 23231.599 ms (00:23.232)
三、性能“问题”
对于 select count(*) from v_t2 (视图),由于 oracle 并不执行函数 ,因此,性能肯定更快。而对于 postgresql,针对每条都要调用一次函数,性能上肯定更慢。
Oracle 与 PostgreSQL 函数行为的差异引发性能差异的更多相关文章
- MySQL&SQL server&Oracle&Access&PostgreSQL数据库sql注入详解
判断数据库的类型 当我们通过一些测试,发现存在SQL注入之后,首先要做的就是判断数据库的类型. 常用的数据库有MySQL.Access.SQLServer.Oracle.PostgreSQL.虽然绝大 ...
- oracle VS postgresql系列-行列转换
[需求]例如先有数据为 id | name ------+--------- | lottu | xuan | rax | ak | vincent 现在需要转换为 id | names ------ ...
- postgresql 函数 参数为复合类型
postgresql没有存储过程,但是函数功能很强大. 在近期开发的电商管理平台中,对于产品的类目管理,设计时有个属性字段,设为字符数组,但是EF不支持数组的操作,所以在添加和修改类目时,需要对属性的 ...
- ORACLE当中自定义函数性优化浅析
为什么函数影响性能 在SQL语句中,如果不合理的使用函数(Function)就会严重影响性能,其实这里想说的是PL/SQL中的自定义函数,反而对于一些内置函数而言,影响性能的可能性较小.那么为什么SQ ...
- [转载]ORACLE日期时间函数大全
ORACLE日期时间函数大全 TO_DATE格式(以时间:2007-11-02 13:45:25为例) Year: yy two digits 两位年 ...
- Oracle的pipelined函数实现高性能大数据处理
从Oracle 8开始,我们就可以从一个collection类型的数据集合中查询出数据,这个集合称之为"虚拟表".它的方法是"SELECT FROM TABLE(CAST ...
- 【转帖】从 Oracle 到 PostgreSQL ,某保险公司迁移实践 技术实践
从 Oracle 到 PostgreSQL ,某保险公司迁移实践 http://www.itpub.net/2019/11/08/4108/ 信泰人寿保险股份有限公司 摘要:去O一直是金融保险行业永恒 ...
- Oracle 与 postgreSQL 事务处理区别(多版本与undo区别)
2015年左右,因为工作需要用MongoDB.CouchBase这两种文档型数据库,时不时到这两个数据库官网上查资料.报BUG.时常可以在MongoDB官网上看到这样一些新闻,“某某企业成功将MySQ ...
- Sql获取表所有列名字段——select * 替换写法,Sqlserver、Oracle、PostgreSQL、Mysql
实际开发中经常用到select * from table,往往需要知道具体的字段,这个时候再去数据库中翻或者查看数据字典比较麻烦.为了方便,自己特意写了一个小函数f_selectall,针对SqlSe ...
随机推荐
- windows10 程序和功能没有Hyper-V选项
1.在电脑桌面新建Hyper-V.cmd文件,将如下代码添加到文件中 pushd "%~dp0" dir /b %SystemRoot%\servicing\Packages\*H ...
- GaussDB(for MySQL) :Partial Result Cache,通过缓存中间结果对算子进行加速
摘要:华为云数据库高级内核技术专家详解GaussDB(for MySQL)Partial Result Cache特性,如何通过缓存中间结果对算子进行加速? 本文分享自华为云社区<GaussDB ...
- 抓到Dubbo异步调用的小BUG,再送你一个贡献开源代码的机会
hello,大家好呀,我是小楼. 最近一个技术群有同学at我,问我是否熟悉Dubbo,这我熟啊~ 他说遇到了一个Dubbo异步调用的问题,怀疑是个BUG,提到BUG我可就不困了,说不定可以水,哦不.. ...
- 服务器与Ajax
前端相关的技术点 HTML 主要用来实现页面的排版布局 CSS 主要用来实现页面的样式美化 JavaScript 主要用来实现前端功能特效 Ajax基础知识铺垫 客户端与服务器 通信协议( ...
- Jetty 源码解析 - 流程
前言 公司实习分配给的任务是精简和优化 Jetty 框架,这里做简单的思路记录(比较乱),源码是 Jetty 7.x.x . 大体流程 Connector 接口的实现类 SelectChannelCo ...
- loguru备忘
loguru是个非常好用的三方日志管理包,简单易用,奈何老是记不住,在这记录一下吧 #coding:utf-8 ''' @version: python3.8 @author: 'eric' @lic ...
- Java开发学习(十二)----基于注解开发依赖注入
Spring为了使用注解简化开发,并没有提供构造函数注入.setter注入对应的注解,只提供了自动装配的注解实现. 1.环境准备 首先准备环境: 创建一个Maven项目 pom.xml添加Spring ...
- AtCoder Beginner Contest 249 E - RLE // 动态规划 + 前缀和优化
传送门:E - RLE (atcoder.jp) 题意: 求满足原长为N且转换后长度严格小于N条件的小写字母组成的字符串的数量,输出时对P取模. 其中,转换规则为,将连续相同的字串替换为"字 ...
- 「Python实用秘技09」更好用的函数运算缓存
本文完整示例代码及文件已上传至我的Github仓库https://github.com/CNFeffery/PythonPracticalSkills 这是我的系列文章「Python实用秘技」的第9期 ...
- Odoo14 Error: While parsing modifiers for page: for modifier "invisible": fieldValue.indexOf is not a function
1 Traceback: 2 Error: While parsing modifiers for page: for modifier "invisible": fieldVal ...