immutable 与 stable 函数的差异
- Stable 函数不能修改数据库,单个Query中所有行给定同样的参数确保返回相同的结果。这种稳定级别允许优化器将多次函数调用转换为一次。在索引扫描的条件中使用这种函数是可行的,因为索引扫描只计算一次比较值(comparison value),而不是每行都计算一次。
- Immutable 函数不能修改数据库,在任何情况下,只要输入参数相同,返回结果就相同。这种级别的函数,优化器可以提前进行计算,在查询过程中作为常量参数。比如:SELECT...WHERE x=2+2 可以简化为SELECT...WHERE x=4。
以下以例子说明二者的差异。
一、KingbaseES
1、准备数据
create table t1(id1 integer,id2 integer);
insert into t1 select generate_series(1,10000000),generate_series(1,10000000); test=# \timing on
Timing is on.
test=# select count(*) from t1;
count
----------
10000000
(1 row) Time: 681.445 ms
2、创建immutable 和 stable 函数
create or replace function f001()
returns bigint
immutable
language sql
as
$$ select count(*) from t1 $$ ; create or replace function f002()
returns bigint
stable
language sql
as
$$ select count(*) from t1 $$ ;
3、单独explain 函数
可以看到对于 immutable 函数,在 explain 时,实际会去执行的;而stable 函数,explain 时则不会实际执行。
test=# explain select f001();
QUERY PLAN
------------------------------------------
Result (cost=0.00..0.01 rows=1 width=8)
(1 row) Time: 450.572 ms
test=# explain select f002();
QUERY PLAN
------------------------------------------
Result (cost=0.00..0.26 rows=1 width=8)
(1 row) Time: 0.641 ms test=# select f001();
f001
----------
10000000
(1 row) Time: 448.720 ms
test=# select f002();
f002
----------
10000000
(1 row) Time: 426.745 ms
4、例子一
可以看到 immutable 函数执行时间主要花在planning上,也就是在制定执行计划前,就已经取得函数的值;而 stable 函数,则在语句解析和执行时,都要执行函数,而且,针对语句的访问的每个tuple,都要执行一次函数调用。
test=# explain analyze select * from (select * from t1 limit 10) a where a.id1=f001();
QUERY PLAN
---------------------------------------------------------------------------------------------------------------------
Subquery Scan on a (cost=0.00..0.27 rows=1 width=8) (actual time=0.012..0.013 rows=0 loops=1)
Filter: (a.id1 = '10000000'::bigint)
Rows Removed by Filter: 10
-> Limit (cost=0.00..0.15 rows=10 width=8) (actual time=0.009..0.010 rows=10 loops=1)
-> Seq Scan on t1 (cost=0.00..148609.21 rows=10007621 width=8) (actual time=0.008..0.009 rows=10 loops=1)
Planning Time: 413.963 ms
Execution Time: 0.026 ms
(7 rows) test=# explain analyze select * from (select * from t1 limit 10) a where a.id1=f002();
QUERY PLAN
---------------------------------------------------------------------------------------------------------------------
Subquery Scan on a (cost=0.00..2.77 rows=1 width=8) (actual time=3691.788..3691.788 rows=0 loops=1)
Filter: (a.id1 = f002())
Rows Removed by Filter: 10
-> Limit (cost=0.00..0.15 rows=10 width=8) (actual time=0.012..0.028 rows=10 loops=1)
-> Seq Scan on t1 (cost=0.00..148609.21 rows=10007621 width=8) (actual time=0.011..0.021 rows=10 loops=1)
Planning Time: 364.233 ms
Execution Time: 3691.807 ms
(7 rows) Time: 4056.907 ms (00:04.057) test=# explain analyze select * from (select * from t1 where 1=2) a where a.id1=f002();
QUERY PLAN
------------------------------------------------------------------------------------------
Result (cost=0.00..2675533.51 rows=1 width=8) (actual time=0.001..0.001 rows=0 loops=1)
One-Time Filter: false
-> Seq Scan on t1 (cost=0.00..2675533.51 rows=1 width=8) (never executed)
Filter: (id1 = f002())
Planning Time: 490.720 ms
Execution Time: 0.017 ms
(6 rows) test=# explain analyze select * from (select * from t1 limit 1) a where a.id1=f002();
QUERY PLAN
--------------------------------------------------------------------------------------------------------------------
Subquery Scan on a (cost=0.00..0.28 rows=1 width=8) (actual time=390.833..390.834 rows=0 loops=1)
Filter: (a.id1 = f002())
Rows Removed by Filter: 1
-> Limit (cost=0.00..0.01 rows=1 width=8) (actual time=0.024..0.053 rows=1 loops=1)
-> Seq Scan on t1 (cost=0.00..148609.21 rows=10007621 width=8) (actual time=0.023..0.023 rows=1 loops=1)
Planning Time: 426.693 ms
Execution Time: 390.852 ms
(7 rows)
5、例子二
sysdate 函数 为 stable 时:sysdate 函数在同一事务内多次执行返回的结果都相同,跨事务则每次结果不同。
test=# begin
test-# for i in 1..5 loop
test-# raise notice '%', sysdate();
test-# perform sys_sleep(1);
test-# commit;
test-# end loop;
test-# end;
test-# /
NOTICE: 2022-06-29 19:53:18
NOTICE: 2022-06-29 19:53:19
NOTICE: 2022-06-29 19:53:20
NOTICE: 2022-06-29 19:53:21
NOTICE: 2022-06-29 19:53:22
ANONYMOUS BLOCK
Time: 5011.031 ms (00:05.011) test=# begin
test-# for i in 1..5 loop
test-# raise notice '%', sysdate();
test-# perform sys_sleep(1);
test-# end loop;
test-# end;
test-# /
NOTICE: 2022-06-29 19:54:14
NOTICE: 2022-06-29 19:54:14
NOTICE: 2022-06-29 19:54:14
NOTICE: 2022-06-29 19:54:14
NOTICE: 2022-06-29 19:54:14
ANONYMOUS BLOCK
Time: 5005.724 ms (00:05.006)
sysdate 函数为 immutable 时:不管是否跨事务,sysdate 函数结果都相同。
test=# alter function sysdate immutable;
ALTER FUNCTION
Time: 6.276 ms
test=# begin
test-# for i in 1..5 loop
test-# raise notice '%', sysdate();
test-# perform sys_sleep(1);
test-# commit;
test-# end loop;
test-# end;
test-# /
NOTICE: 2022-06-29 19:54:58
NOTICE: 2022-06-29 19:54:58
NOTICE: 2022-06-29 19:54:58
NOTICE: 2022-06-29 19:54:58
NOTICE: 2022-06-29 19:54:58
ANONYMOUS BLOCK
Time: 5007.899 ms (00:05.008)
test=# begin
test-# for i in 1..5 loop
test-# raise notice '%', sysdate();
test-# perform sys_sleep(1);
test-# end loop;
test-# end;
test-# /
NOTICE: 2022-06-29 19:55:11
NOTICE: 2022-06-29 19:55:11
NOTICE: 2022-06-29 19:55:11
NOTICE: 2022-06-29 19:55:11
NOTICE: 2022-06-29 19:55:11
ANONYMOUS BLOCK
Time: 5007.694 ms (00:05.008)
二、Oracle
1、创建函数
create or replace function f001
return integer
deterministic
as
cnt integer;
begin
for i in 1..10 loop
select count(*) into cnt from t1 ;
end loop;
return cnt;
end ; create or replace function f002
return integer
as
cnt integer;
begin
for i in 1..10 loop
select count(*) into cnt from t1 ;
end loop;
return cnt;
end ;
2、单独explain 函数
可以看到两个函数都不会执行,不管是 deterministic,还是 volatile
SQL> explain plan for select f001() from dual; Explained. Elapsed: 00:00:00.00
SQL> explain plan for select f002() from dual; Explained. Elapsed: 00:00:00.00
3、实际执行
deterministic 只需执行一次,但是在SQL执行时才执行函数调用,而非explain时;volatile 解析时,不需要调用函数,而针对每个tuple 都必须要调用一次,如果没有记录,则无需调用。
SQL> select * from (select * from t1 where rownum<11) where id1=f001(); no rows selected Elapsed: 00:00:00.48 SQL> select * from (select * from t1 where rownum<11) where id1=f002(); no rows selected Elapsed: 00:00:05.01 SQL> select * from (select * from t1 where 1=2) where id1=f002(); no rows selected Elapsed: 00:00:00.00
immutable 与 stable 函数的差异的更多相关文章
- PHP 日期计算函数【差异天数】
function count_days($a,$b){ $a_dt=getdate($a); $b_dt=getdate($b); $a_new=mktime(12,0,0,$a_dt['mon'], ...
- 十八般武艺玩转GaussDB(DWS)性能调优:SQL改写
摘要:本文将系统介绍在GaussDB(DWS)系统中影响性能的坏味道SQL及SQL模式,帮助大家能够从原理层面尽快识别这些坏味道SQL,在调优过程中及时发现问题,进行整改. 数据库的应用中,充斥着坏味 ...
- Oracle 与 PostgreSQL 函数行为的差异引发性能差异
对于Oracle,对于数据修改的操作通过存储过程处理,而对于函数一般不进行数据修改操作.同时,函数可以通过 Select 进行调用,而存储过程则不行. 一.对于volatile 函数的行为 1.Ora ...
- KingbaseES 与Oracle 函数稳定性对于性能影响差异比较
一.函数的属性 KingbaseES 函数在定义时有三种稳定性级别:volatile.stable 和 immutable.默认情况下,创建函数的稳定性为volatile.以下是这三种函数的区别: V ...
- PG 函数的易变性(Function Volatility Categories)
此概念的接触是在做分区表的时候碰到的,分区表按时间字段分区,在查询时当where条件中时间为now()或者current_time()等时是无法查询的,即使进行格式转换也不行,只有是时间格式如‘201 ...
- CREATE FUNCTION - 定义一个新函数
SYNOPSIS CREATE [ OR REPLACE ] FUNCTION name ( [ argtype [, ...] ] ) RETURNS rettype { LANGUAGE lang ...
- KingbaseES 函数稳定性与SQL性能
背景:客户现场的一次艰苦的调优过程(https://www.cnblogs.com/kingbase/p/16015834.html),让我觉得非常有必要让数据库用户了解函数的不同稳定性属性,及其对于 ...
- 《zw版·delphi与halcon系列原创教程》zw版_THOperatorSetX控件函数列表 v11中文增强版
<zw版·delphi与halcon系列原创教程>zw版_THOperatorSetX控件函数列表v11中文增强版 Halcon虽然庞大,光HALCONXLib_TLB.pas文件,源码就 ...
- 《zw版·delphi与halcon系列原创教程》zw版_THImagex控件函数列表
<zw版·delphi与halcon系列原创教程>zw版_THImagex控件函数列表 Halcon虽然庞大,光HALCONXLib_TLB.pas文件,源码就要7w多行,但核心控件就是两 ...
随机推荐
- SAP IDOC-Segment E1EDP19 Document Item Object Identification
PO创建时,通过IDOC EDI 接口自动创建SO 案例. BD54 配置逻辑系统 SCC4 给集团分配逻辑系统 SM59 新建RFC 链接 WE21 创建IDOC 处理端口 we20 创建合作伙伴 ...
- SpringCloud Alibaba整合Sentinel
SpringCloud Alibaba整合Sentinel Sentinel 控制台 1. 概述 Sentinel 提供一个轻量级的开源控制台,它提供机器发现以及健康情况管理.监控(单机和集群),规则 ...
- linux系统调优工具
系统调优思路 性能优化就是找到系统处理中的瓶颈以及去除这些的过程,性能优化其实是对 OS 各子系统达到一种平衡的定义.具体步骤如下: 1. 系统的运行状况: CPU -> MEM -> D ...
- Spring框架系列(7) - Spring IOC实现原理详解之IOC初始化流程
上文,我们看了IOC设计要点和设计结构:紧接着这篇,我们可以看下源码的实现了:Spring如何实现将资源配置(以xml配置为例)通过加载,解析,生成BeanDefination并注册到IoC容器中的. ...
- 更强的 JsonPath 兼容性及性能测试之2022版(Snack3,Fastjson2,jayway.jsonpath)
2022年了,重新做了一份json path的兼容性与性能测试.三个市面上流行框架比较性测试. 免责声明:可能测试得方式不对而造成不科学的结果(另外,机器不同结果会有不同),可以留言指出来.以下测试数 ...
- python+tkinter 简单的登录窗口demo
一个简单的登录窗口布局,可以用于日常快速搭建一个简单的窗口类. from tkinter import * import tkinter.messagebox class LoginUi: def _ ...
- jdbc 12: 模糊查询
jdbc连接mysql,进行模糊查询 package com.examples.jdbc.o11_模糊查询; import com.examples.jdbc.utils.DBUtils; impor ...
- ZooKeeper3.4.10集群安装配置-Docker
一. 服务器规划 主机 IP 端口 备注 b-mid-24 172.16.0.24 2181, 2888, 3888 2181:对cline端提供服务 3888:选举leader使用 2888:集群内 ...
- 移动web开发01
pc端的支持情况.IE9以下的版本就会全军覆没.移动端就不会出现版本支持问题. 因为第一个孩子是p,但是他又是在span里面选,所以根本选不出来,报错.改成第二个孩子才可以选出来. 这样就可以选出sp ...
- 解析MySQL存储过程的游标执行过程
GreatSQL社区原创内容未经授权不得随意使用,转载请联系小编并注明来源. 内容提纲 一.测试环境搭建 二.执行过程解析 三.注意事项 一.测试环境搭建 首先创建一张表,并插入几行数据字段: CRE ...