概述 :在DBA的日常工作中,经常遇到一些需要基于数据库当前状态的实用程序查询的实例。比如一个逻辑复制的目标表,主键ID列与生成数据的序列不同步,这将导致插入新行是,会有主键冲突。要纠正这个问题,需要设置序列可以生成的值,超过表中当前最大值。

解决此类问题可以使用动态DDL,虽然SQL 中的数据定义语言(DDL) 本身是非动态的,因为DDL需要有严格的解析规则,而且其中的子查询预定义的数据类型、表结构,以及基于已知关联列。

下面的案例,可以让 SQL语句随心所欲地执行动态 DDL,而不必每次都手动编写这些查询。

一、改变序列重启值的解决方案

希望通过查询显式地为序列提供RESTART值,重置一个序列,使其从它所关联的表的当前最大值之后开始。预想中,使用简单直接的DDL语句:

ALTER SEQUENCE tablename_id_seq RESTART (SELECT max(id)+1 FROM tablename);
ERROR: syntax error at or near "(", at character 38
STATEMENT: ALTER SEQUENCE tablename_id_seq RESTART (SELECT max(id) + 1 FROM tablename);

正如所看到的,KingbaseE语法不支持这种方法,因为这里需要的是实际值,而不是子查询。

1、使用ksql的变量替换

使用ksql变量,首先选择一个变量的值,然后将这个值替换到表达式中。

-- 使用\gset设置一个带有查询结果的变量

SELECT max(id) + 1 as id_max from tablename
\gset
-- substitute the variable in a new query
ALTER SEQUENCE tablename _id_seq RESTART :id_max ;

在实例中,使用\gset命令来得到max(id)+1的结果,并将其存储在会话中,供以后使用。然后使用:id_max语法将该变量插入到SQL语句中,将其直接传递给服务器。

2、使用ksql的 \gexec 命令

使用ksql进行动态SQL的另一种方法,是设计一个查询SELECT语句,返回希望运行的语句,然后使用\gexec命令执行。

SELECT 'ALTER SEQUENCE tablename_id_seq RESTART ' || max(id) + 1 as query FROM tablename;

\gexec query

ALTER SEQUENCE tablename_id_seq RESTART 100001

ALTER SEQUENCE

与变量替换相比,这种方法的一个优点是,它可以处理更复杂的语句和多个返回值,因此可以基于任意条件设计查询语句,并生成多个SQL语句;第一个方法实现仅限于每次返回单个行的查询。在使用\gexec对服务器执行SQL语句之前,这还可以预览将要运行的SQL语句,因此提供一定程度的安全性。

二、非ksql的动态SQL

ksql具有处理动态SQL的功能,但如何在第三方应用中,实现动态SQL的功能。现在使用基本类似的方法,编写sql语句块来生成查询,然后执行返回的SQL语句。这些方法与上面的ksql部分中的方法大致相关,因此适用于单个或多个动态语句。

1、匿名块

要使用服务器端动态SQL,我们需要使用format函数构造查询,并执行底层文本,就像我们自己发出底层查询一样。

BEGIN
EXECUTE format('ALTER SEQUENCE tablename_id_seq RESTART %s', (SELECT max(id) + 1 FROM tablename));
END;

2、创建函数 exec()

与匿名块的功能几乎相同,我们也可以创建一个简单的用户函数,简单地调用它的输入参数EXECUTE。

CREATE OR REPLACE FUNCTION exec(raw_query text) RETURNS text AS
BEGIN
EXECUTE raw_query;
RETURN raw_query;
END;
SELECT exec(format('ALTER SEQUENCE tablename_id_seq RESTART %s', (SELECT max(id) + 1 FROM tablename)));

函数方法基本上只是将查询移动到传入的参数中,但它带来更灵活的针对构造的查询列表,在SQL语句中调用此函数,可以选择依次运行每个 SQL。

限制

这些方法对动态SQL运行的限制,在显式事务块之外,运行的SQL命令受到限制。如果想在所有已知索引上运行REINDEX CONCURRENTLY,那么使用exec()函数为public模式中的所有索引执行REINDEX CONCURRENTLY语句:

SELECT exec(format('REINDEX INDEX CONCURRENTLY %I', relname))
FROM sys_class r
JOIN sys_namespace s ON r.relnamespace = s.oid
WHERE relkind = 'i' AND nspname = 'public' ; ERROR: REINDEX CONCURRENTLY cannot be executed from a function
CONTEXT: SQL statement "REINDEX INDEX CONCURRENTLY tablename_pkey"
PL/SQL function exec(text) line 3 at EXECUTE

从这里看到,由于REINDEX CONCURRENTLY需要管理自己的事务状态,因此它不能作为函数运行;在 KingbaseES 中,函数的内在事务运行时,是以函数的SQL语句完全成功或完全失败。

试一试命令 \gexec

SELECT format('REINDEX INDEX CONCURRENTLY %I', relname)
FROM sys_class r
JOIN sys_namespace s ON r.relnamespace = s.oid
WHERE relkind = 'i' AND nspname = 'public' \gexec

  

由于\gexec处理是由ksql完成的,因此生成的语句独立运行。

巧用KingbaseES中的动态DDL的更多相关文章

  1. 在Oracle中执行动态SQL的几种方法

    转载:在Oracle中执行动态SQL的几种方法 以下为内容留存: 在Oracle中执行动态SQL的几种方法 在一般的sql操作中,sql语句基本上都是固定的,如:SELECT t.empno,t.en ...

  2. 巧用FPGA中资源

    随着FPGA的广泛应用,所含的资源也越来越丰富,从基本的逻辑单元.DSP资源和RAM块,甚至CPU硬核都能集成在一块芯片中.在做FPGA设计时,如果针对FPGA中资源进行HDL代码编写,对设计的资源利 ...

  3. 深入理解javascript中的动态集合——NodeList、HTMLCollection和NamedNodeMap

    × 目录 [1]NodeList [2]HTMLCollection [3]NamedNodeMap[4]注意事项 前面的话 一说起动态集合,多数人可能都有所了解.但是,如果再深入些,有哪些动态集合, ...

  4. Quartz 在 Spring 中如何动态配置时间--转

    原文地址:http://www.iteye.com/topic/399980 在项目中有一个需求,需要灵活配置调度任务时间,并能自由启动或停止调度. 有关调度的实现我就第一就想到了Quartz这个开源 ...

  5. ZBrush中的动态网格该怎么进行运用

    DynaMesh是ZBrush最新的基础模型创建工具,该命令用于基本模型的起稿到中模的制作.使用DynaMesh完全不启用考虑模型的拓扑,可以从一个图形拉扯出整个模型的分支,本文将以一个实例简单介绍Z ...

  6. (待续)C#语言中的动态数组(ArrayList)模拟常用页面置换算法(FIFO、LRU、Optimal)

    目录 00 简介 01 算法概述 02 公用方法与变量解释 03 先进先出置换算法(FIFO) 04 最近最久未使用(LRU)算法 05 最佳置换算法(OPT) 00 简介 页面置换算法主要是记录内存 ...

  7. 使用Java中的动态代理实现数据库连接池

    2002 年 12 月 05 日 作者通过使用JAVA中的动态代理实现数据库连接池,使使用者可以以普通的jdbc连接的使用习惯来使用连接池. 数据库连接池在编写应用服务是经常需要用到的模块,太过频繁的 ...

  8. Linux中的动态库和静态库(.a/.la/.so/.o)

    Linux中的动态库和静态库(.a/.la/.so/.o) Linux中的动态库和静态库(.a/.la/.so/.o) C/C++程序编译的过程 .o文件(目标文件) 创建atoi.o 使用atoi. ...

  9. iOS开发UI篇—在UITableview的应用中使用动态单元格来完成app应用程序管理界面的搭建

    iOS开发UI篇—在UITableview的应用中使用动态单元格来完成app应用程序管理界面的搭建 一.实现效果 说明:该示例在storyboard中使用动态单元格来完成. 二.实现 1.项目文件结构 ...

随机推荐

  1. 记一次beego通过go get命令后找不到bee.exe的坑

    学习goweb开发,gin是个轻量级的框架.如果想要一个类如aspnetmvc帮我们搭建好了的goweb框架,beego值得去学习.否则gin下面需要动手构建好多代码.新手还是先学现成的节约时间成本. ...

  2. netty系列之:在netty中使用native传输协议

    目录 简介 native传输协议的依赖 netty本地传输协议的使用 总结 简介 对于IO来说,除了传统的block IO,使用最多的就是NIO了,通常我们在netty程序中最常用到的就是NIO,比如 ...

  3. 华为HMS Core携手超图为三维GIS注入新动能

    6月30日,在2022(第五届)GIS软件技术大会GIS基础软件新技术板块论坛上,华为联合超图推出了基于HMS Core 3D Engine开发的高保真三维GIS插件,通过3D渲染技术助力三维GIS实 ...

  4. 详解HashMap源码解析(下)

    上文详解HashMap源码解析(上)介绍了HashMap整体介绍了一下数据结构,主要属性字段,获取数组的索引下标,以及几个构造方法.本文重点讲解元素的添加.查找.扩容等主要方法. 添加元素 put(K ...

  5. 强化学习-学习笔记5 | AlphaGo

    本文不是论文阅读笔记,只是一个学习笔记,重在理解,在严谨程度上可能稍差. AlphaGo 论文指路: Mastering the game of Go with deep neural network ...

  6. labview从入门到出家8(进阶篇)--简单好用的状态机

    labview的状态机类似于一个软件框架的基本单元,好的软件框架和软件思路采用一个好的状态机,就如虎添翼了.这章给大家讲一个本人常用的一个状态机,基本上以前的项目都是建立在这个状态机上完成的,当然网上 ...

  7. StringBuilder的构造方法和append方法

    Java中StringBuilder的构造方法 StringBuilder构造方法: public StringBuilder(): 创建一个空白可变的字符串对象,不含有任何内容 public Str ...

  8. 常用类-jdk8之前的日期和API

    一.System静态方法 点击查看代码 package com.Tang.StringDay01; import org.junit.Test; public class DateTimeTest { ...

  9. 一文吃透如何部署kubernetes之Dashboard

    kubernetes Dashboard是什么? Dashboard是kubernetes的Web GUI,可用于在kubernetes集群上部署容器化应用,应用排错,管理集群本身及其附加的资源等,它 ...

  10. AtCoder Beginner Contest 249 F - Ignore Operations // 贪心 + 大根堆

    传送门:F - Keep Connect (atcoder.jp) 题意: 给定长度为N的操作(ti,yi). 给定初值为0的x,对其进行操作:当t为1时,将x替换为y:当t为2时,将x加上y. 最多 ...