Postgresql源码(108)不同类型insert在parse阶段的差异分析

news2024/12/24 11:38:57

0 概述

分析三种类型的insert在parse的各个阶段的差异:

insert into TAB_IS SELECT * FROM STUDENT a WHERE a.sno > ANY (SELECT b.sno from STUDENT b);
insert into TAB_IS values(10, 'AAA');
insert into TAB_IS values(20, 'CCC'),(30, 'DDD'),(40, 'EEE');

不同insert的计划树type

# T_NestLoopState
insert into TAB_IS SELECT * FROM STUDENT a WHERE a.sno > ANY (SELECT b.sno from STUDENT b);
# T_ResultState
insert into TAB_IS values(10, 'AAA');
# T_ValuesScanState
insert into TAB_IS values(20, 'CCC'),(30, 'DDD'),(40, 'EEE');
# T_FunctionScanState
insert into TAB_IS select i, 'QQQ', i % 10 from generate_series(1, 1000) t(i);
# T_ProjectSetState
insert into TAB_IS values(generate_series(1,10), 'DDD', 1);

1 语义分析差异

下面三种SQL在语义分析结果来看有什么区别?

insert into TAB_IS SELECT * FROM STUDENT a WHERE a.sno > ANY (SELECT b.sno from STUDENT b);
insert into TAB_IS values(10, 'AAA');
insert into TAB_IS values(20, 'CCC'),(30, 'DDD'),(40, 'EEE');

语义分析结果来看,insert语句都会构造插入表和数据表两张表(RangeTblEntry),数据表可能是值构造出来的,或者是select查询出来的。

核心流程都是构造数据表的RangeTblEntry。


代码位置:

transformInsertStmt
	SelectStmt *selectStmt = (SelectStmt *) stmt->selectStmt;

	// 如果selectStmt非空,表示存在select子句
	if (selectStmt == NULL)
		...						// 普通insert
	else if (isGeneralSelect)
		...						// 带select子句
	else if (list_length(selectStmt->valuesLists) > 1)
		...						// 多values

1 insert select语义分析结果

pg_analyze_and_rewrite_fixedparams

insert into TAB_IS SELECT * FROM STUDENT a WHERE a.sno > ANY (SELECT b.sno from STUDENT b);语义分析结果
在这里插入图片描述

2 insert values语义分析结果

insert into TAB_IS values(10, 'AAA');语义分析结果
在这里插入图片描述

3 insert values values语义分析结果

insert into TAB_IS values(20, 'CCC'),(30, 'DDD'),(40, 'EEE');语义分析结果
在这里插入图片描述

2 优化结果差异

一定存在ModifyTable节点,因为这是一个写表操作,也就是会进入ExecModifyTable函数。

ExecModifyTable函数loop下层节点每次拿一条数据,然后执行insert操作。知道下层节点没数据为止。

从ExecModifyTable节点的lefttree可以知道具体是哪种insert。
在这里插入图片描述

3 执行阶段

从执行阶段来看,下面三种SQL有什么区别?

insert into TAB_IS SELECT * FROM STUDENT a WHERE a.sno > ANY (SELECT b.sno from STUDENT b);
                                           QUERY PLAN
-------------------------------------------------------------------------------------------------
 Insert on tab_is  (cost=0.15..208.42 rows=0 width=0)
   ->  Nested Loop Semi Join  (cost=0.15..208.42 rows=367 width=46)
         ->  Seq Scan on student a  (cost=0.00..21.00 rows=1100 width=46)
         ->  Index Only Scan using student_pkey on student b  (cost=0.15..6.62 rows=367 width=4)
               Index Cond: (sno < a.sno)


insert into TAB_IS values(10, 'AAA');
                     QUERY PLAN
----------------------------------------------------
 Insert on tab_is  (cost=0.00..0.01 rows=0 width=0)
   ->  Result  (cost=0.00..0.01 rows=1 width=46)


insert into TAB_IS values(20, 'CCC'),(30, 'DDD'),(40, 'EEE');
                             QUERY PLAN
--------------------------------------------------------------------
 Insert on tab_is  (cost=0.00..0.04 rows=0 width=0)
   ->  Values Scan on "*VALUES*"  (cost=0.00..0.04 rows=3 width=46)

执行阶段没什么区别,都是走ExecModifyTable内部循环搞定。

  1. 每次从lefttree中执行一把拿到一条,subplanstate = outerPlanState(node);context.planSlot = ExecProcNode(subplanstate);
  2. 根据operation类型(insert)执行具体insert操作ExecInsert,比较简单,中间会有slot到tuple的转换。执行器的元组都是包装在slot中的。现在PG的存储引擎提供了AM接口,代码更清晰了。

(执行器层ExecInsert→存储层入口table_tuple_insert)

PortalRun
	PortalRunMulti
		ProcessQuery
			CreateQueryDesc
			ExecutorStart
			ExecutorRun
				standard_ExecutorRun
					ExecutePlan
						ExecProcNode
							ExecProcNodeFirst
								ExecModifyTable 
                             -----> ExecProcNode(subplanstate)  --- 
						   /     	switch (operation)              \
						   \			case CMD_INSERT:            /
                             ------------- ExecInsert   <----------
											
										

ps. 测试数据

drop table student;
create table student(sno int primary key, sname varchar(10), ssex int);
insert into student values(1, 'stu1', 0);
insert into student values(2, 'stu2', 1);
insert into student values(3, 'stu3', 1);
insert into student values(4, 'stu4', 0);

drop table course;
create table course(cno int primary key, cname varchar(10), tno int);
insert into course values(10, 'meth', 1);
insert into course values(11, 'english', 2);

drop table teacher;
create table teacher(tno int primary key, tname varchar(10), tsex int);
insert into teacher values(1, 'te1', 1);
insert into teacher values(2, 'te2', 0);

drop table score;
create table score (sno int, cno int, degree int);
insert into score values (1, 10, 100);
insert into score values (1, 11, 89);
insert into score values (2, 10, 99);
insert into score values (2, 11, 90);
insert into score values (3, 10, 87);
insert into score values (3, 11, 20);
insert into score values (4, 10, 60);
insert into score values (4, 11, 70);


SELECT * FROM STUDENT a WHERE a.sno > ANY (SELECT b.sno from STUDENT b); 


drop table TAB_IS;
create table TAB_IS(sno int, sname varchar(10), ssex int);
       
insert into TAB_IS SELECT * FROM STUDENT a WHERE a.sno > ANY (SELECT b.sno from STUDENT b);
insert into TAB_IS values(10, 'AAA');
insert into TAB_IS values(20, 'CCC'),(30, 'DDD'),(40, 'EEE');


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/707931.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

代码随想录再战day4

24 两两交换链表的节点 力扣 思路&#xff1a; 还是看了carl哥的视频讲解才写出来。有点难搞 首先 还是老样子 需要一个dummyhead虚拟头节点。 然后核心就是 我们要操作后面两个节点的时候 一定要移动到 这两个节点的上一个节点。 &#xff08;来自代码随想录&#xff09; 然后…

EasyFlash在GD32F303CC上面的移植

记录学习的过程&#xff0c;如果在GD32F303CC上面移植EasyFlash。关于EasyFlash的相关介绍和源码&#xff0c;请参考&#xff1a;https://gitee.com/Armink/EasyFlash 或者 https://github.com/armink/EasyFlash 主要记录移植过程中需要注意的点&#xff0c;移植还是比较简单的…

6.6 极重要的复习,权限与指令间的关系

权限对于使用者账号很重要&#xff0c;因为他可以限制使用者能不能读取/创建/删除/修改文件或目录。 一、让使用者能进入某目录成为“可工作目录”的基本权限为何&#xff1a; 可使用的指令&#xff1a;例如 cd 等变换工作目录的指令&#xff1b; 目录所需权限&#xff1a;使…

liunx安装git

liunx安装git &#xff1a; 提示&#xff1a;记录自己装git 过程 执行下边命令安装 yum -y install git 安装完查看是否安装成功 git --version安装路径默认在/usr/libexe 愉快开始使用git

帆软 FineReport 绘制漏斗图

七一建党节&#xff0c;祝党生日快乐&#xff01; 夏日炎炎&#xff0c;周末在家&#xff0c;想起在用帆软做页面展示的时候&#xff0c;使用到了漏斗图&#xff0c;记录下来&#xff0c;方便查看。 以订单销量变化为例&#xff0c;分为五个阶段&#xff0c;商品浏览人数&#…

fastadmin给操作按钮添加权限菜单控制|新增权限控制菜单

1、在对应的控制器文件中&#xff0c;添加如下代码&#xff1a; $adminIds $this->getDataLimitAdminIds(); if (is_array($adminIds)) {if (!in_array($row[$this->dataLimitField], $adminIds)) {$this->error(__(You have no permission));} } 2、在对应的index…

Redis实战篇(三)

四.分布式锁 4.1.分布式锁概述 分布式锁&#xff1a;满足分布式系统或集群模式下多进程可见并且互斥的锁。分布式锁的核心思想就是让大家都使用同一把锁&#xff0c;只要大家使用的是同一把锁&#xff0c;那么我们就能锁住线程&#xff0c;不让线程并行&#xff0c;让程序串行…

通过简数实现全网文章采集的方法简介

简数是一款全网文章采集工具&#xff0c;旨在帮助站长通过设置关键词来进行全网文章采集。然而&#xff0c;使用简数可能会有一些相对复杂的操作&#xff0c;导致站长在学习和使用过程中面临较高的学习成本&#xff0c;因为其中涉及到许多专业术语和功能的作用不明确。鉴于这一…

Unity - 搬砖日志 - UGUI合批优化 - Overlap(UI AABB 有重叠), Z != 0 照样合批的方案

文章目录 环境目的Screen Space - Overlay优化限制该方案起源 环境 Unity : 2020.3.37f1 Pipeline : BRP &#xff08;另一个项目在 2021.1.xx 的 LTS 下的 URP 管线同样如此&#xff0c;目测&#xff1a;因为 UGUI 不受渲染管线切换而变化&#xff09; 目的 便于索引&#…

IDEA启动Tomcat控制台乱码问题解决方法

最近项目需要用到Tomcat跑一个项目&#xff0c;但是开发Sprintboot项目久了&#xff0c;也很少用Tomcat&#xff0c;因为Springboot是集成Tomcat等中间件的&#xff0c;所以不需要下载Tomcat&#xff0c;好了&#xff0c;回归真题&#xff0c;本博客介绍跑Tomcat过程遇到的控制…

信道编码【编码、纠错检错】

差错控制的基本概念--数字信号在传输过程中&#xff0c;由于信道传输特性不理想及加性噪声的影响&#xff0c;不可避免地会发生错误。 可通过以下三方面的措施来减小误码率&#xff1a;1&#xff09;提高信道容量&#xff1b;2&#xff09;降低编码效率&#xff1b;3&#xff…

Binder系列--ServiceManager的启动

ServiceManager的启动 hongxi.zhu Android 13 主要流程&#xff1a; 1. 启动ServiceManager进程 ServiceManager是由init(pid 1)进程启动的 system/core/rootdir/init.rc on init......# Start essential services.start servicemanager //framework层使用start hwservic…

Linux | Ubuntu卸载QQ

Linux | Ubuntu卸载QQ 终端输入&#xff1a; dpkg -l| grep qq如下图&#xff0c;找到QQ文件&#xff1a; 删除命令&#xff1a; sudo apt-get --purge remove 文件名在终端输入&#xff1a; sudo apt-get --purge remove libqqwing2v5:amd64如下图删除成功

【Java程序设计实训】基于B/S架构的MyShop商城

MYSHOP商城 实验目的实验概述系统功能概述Myshop 商城概述系统开发分析功能列表系统用例图系统活动图 数据库设计运作界面展示用户管理模块新用户注册用户登录商城首页与用户退出 商品模块商品分页展示查看商品详情信息 购物车模块空购物车页面加入商品到购物车 订单模块提交订…

4.32UDP通信实现 4.33广播 4.34组播 4.35本地套接字通信

4.32UDP通信实现 ![在这 udp_client.c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <string.h> #include <arpa/inet.h>int main() {// 1.创建一个通信的socketint fd socket(PF_INET, SOCK_DGRAM, 0);if(fd -1) {…

springboot项目外卖管理 day08-缓存优化

文章目录 一、缓存优化问题说明环境搭建导入maven坐标配置yml文件设置序列化器&#xff0c;编写配置类 缓存短信验证码缓存菜品数据实现思路 SpringCacheSpring Cache介绍Spring Cache常用注解Spring Cache使用方式 缓存套餐数据实现思路 一、缓存优化 问题说明 环境搭建 导入…

【Framework】startService启动流程

前言 启动service有两种方式&#xff1a;startService和bindService。 这一篇先讲startService&#xff0c;读者如果只想看流程图&#xff0c;可以直接跳到总结。 1. ContextImpl 代码路径&#xff1a;frameworks\base\core\java\android\app\ContextImpl.java 1.1 startServ…

SHA-256算法及示例

1. 引言 SHA-256&#xff08;安全哈希算法&#xff0c;FIPS 182-2&#xff09;是密码学哈希函数&#xff0c;其摘要长度为256位。SHA-256为keyless哈希函数&#xff0c;即为MDC&#xff08;Manipulation Detection Code&#xff09;。【MAC消息认证码有key&#xff0c;不是key…

【数据库】外键的作用

前言 说到外键&#xff0c;一般就会牵扯出约束。不谈约束&#xff0c;起始外键就是一个普通的字段&#xff08;Column&#xff09;&#xff0c;起到一个关联的作用。 先把约束放一边&#xff0c;看看外键有哪些作用。 建立表中记录的一对一的关系 学生表&#xff1a; 手机表…

C++引用计数

文章目录 1. 什么是引用计数2. 引用计数的实现3. 示例代码 1. 什么是引用计数 引用计数&#xff08;reference count&#xff09;的核心思想是使用一个计数器来标识当前指针指向的对象被多少类的对象所使用&#xff08;即记录指针指向对象被引用的次数&#xff09;。它允许有多…