PostgreSQL巧用ctid等价改写SQL

PostgreSQL中的ctid即行号,有点类似oracle中的rowid,今天碰到一个SQL刚好利用到pg中ctid的特点得到了极大的优化。

原始SQL(语句里内容已修改):

bill@bill=>do language plpgsql  $$
bill$# declare 
bill$# v_cnt int;
bill$# begin
bill$# select count(*) into v_cnt from t1 ;
bill$# if v_cnt>0 
bill$# then  
bill$# create table t2 as select * from t1 limit 100;
bill$# else 
bill$# create table t3 as select * from t1 limit 10;
bill$# end if;
bill$# end;
bill$# $$;
DO
Time: 62.643 ms

我们先来分析下上面的SQL,其意思大致为:获取t1表中的记录数,判断是否大于0,如果大于0则创建表t2,否则创建表t3。

因此上面的代码就写成这样了,但是我们思考下其真正的需求就是如此吗?
我们可以换个说法:如果t1表有记录则创建表t2,没有则创建表t3。

两者有区别吗?其实区别还是很大的,前者可是强调获取记录数,我们是不是一定要遍历整个表得出一个记录数才知道是否大于0?
真正需求的理解可以让我们这样实现,只要从t1表中成功获取到第一条记录,就可以停止检索了,表示该表有记录了,难道事实不是这样?

因此我们可以将原先的sql从select count() from t1;改造为:
select count(
) from t1 where ctid=’(0,1)’;

那么上面的SQL整个就可以写成:

bill@bill=>do language plpgsql  $$
bill$# declare 
bill$# v_cnt int;
bill$# begin
bill$# select count(*) into v_cnt from t1 where ctid='(0,1)' ;
bill$# if v_cnt>0 
bill$# then  
bill$# create table t2 as select * from t1
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值