`

postgreSQL存储过程写法示例

 
阅读更多
PostgreSQL的存储过程简单入门 http://blog.csdn.net/rachel_luo/article/details/8073458
存储过程事物 http://www.php100.com/manual/PostgreSQL8/tutorial-transactions.html
PL/pgSQL - SQL存储过程语言 https://wiki.postgresql.org/wiki/9.1%E7%AC%AC%E4%B8%89%E5%8D%81%E4%B9%9D%E7%AB%A0

postgreSQL存储过程写法示例http://blog.sina.com.cn/s/blog_448574810101f64u.html
结构
PL/pgSQL是一种块结构的语言,比较方便的是用pgAdmin III新建Function,填入一些参数就可以了。基本上是这样的:
CREATE OR REPLACE FUNCTION 函数名(参数1,[整型 int4, 整型数组 _int4, ...])
RETURNS 返回值类型 AS
$BODY$
DECLARE
变量声明
BEGIN
函数体
END;
$BODY$
LANGUAGE ‘plpgsql’ VOLATILE;

变量类型
除了postgresql内置的变量类型外,常用的还有 RECORD ,表示一条记录。
赋值
赋值和Pascal有点像:“变量 := 表达式;”
有些奇怪的是连接字符串的是“||”,比如 sql := ‘SELECT * FROM’ || table || ‘WHERE …’;
判断
判断又和VB有些像:
IF 条件 THEN

ELSEIF 条件 THEN

ELSE

END IF;
循环
循环有好几种写法:
WHILE expression LOOP
statements
END LOOP;
还有常用的一种是:(从1循环到9可以写成FOR i IN 1..9 LOOP)
FOR name IN [ REVERSE ] expression .. expression LOOP
statements
END LOOP;
其他
还有几个常用的函数:
SELECT INTO record …; 表示将select的结果赋给record变量(RECORD类型)
PERFORM query; 表示执行query并丢弃结果
EXECUTE sql; 表示执行sql语句,这条可以动态执行sql语句(特别是由参数传入构造sql语句的时候特别有用)

参数:
传递给函数的参数都是用 $1,$2,等等这样的标识符。有时候为了增强可读性,我们可以为 $n 参数名声明别名。然后通过这个别名或者数字标识符可以指向这个参数值。
有两种方法创建一个别名。最好的方法是用CREATE FUNCTION命令给予这个参数一个名字,例如:
   CREATE FUNCTION sales_tax(subtotal real) RETURNS real AS $$
   BEGIN
      RETURN subtotal * 0.06;
   END;
   $$ LANGUAGE plpgsql;

另一个方法是,在PostgreSQL 8.0之前唯一的方法,明确的用别名进行声明,用以下的语法进行声明:
   name ALIAS FOR $n;
这个风格的同一个例子看起来像下面这样 :
   CREATE FUNCTION sales_tax(real) RETURNS real AS $$
   DECLARE
       subtotal ALIAS FOR $1;
   BEGIN
       RETURN subtotal * 0.06;
   END;
   $$ LANGUAGE plpgsql;

注意:这两个例子不是完全一样的。在第一种情况,subtotal可以用sales_tax.subtotal进行引用,但是在第二种情况下不能这么做。(如果我们给这个内部块附加了一个标签,subtotal能够替代这个标签)
一些更多的例子:
   CREATE FUNCTION instr(varchar, integer) RETURNS integer AS $$
   DECLARE
        v_string ALIAS FOR $1;
        index ALIAS FOR $2;
   BEGIN
         -- some computations using v_string and index here
   END;
   $$ LANGUAGE plpgsql;

   CREATE FUNCTION concat_selected_fields(in_t sometablename) RETURNS text AS $$
   BEGIN
        RETURN in_t.f1 || in_t.f3 || in_t.f5 || in_t.f7;
   END;
   $$ LANGUAGE plpgsql

当一个PL/pgSQL函数用输出参数来进行声明时,给予这个输出参数$n名和一个任意的别名跟正常输入参数是同样的方法。即使这个输出参数以NULL开始时也是一个有效的变量,它应该在函数的执行过程中被分配。这个参数最好的值将被返回。例如,这个sales-tax例子也可以用这种方法完成:
   CREATE FUNCTION sales_tax(subtotal real, OUT tax real) AS $$
   BEGIN
        tax := subtotal * 0.06;
   END;
   $$ LANGUAGE plpgsql;

注意:我们省略了RETURNS real---我们可以将它包括在内,但它是多余的。
当返回多个值的时候输出参数将非常有用,一个简单的例子是:
   CREATE FUNCTION sum_n_product(x int, y int, OUT sum int, OUT prod int) AS $$
   BEGIN
       sum := x + y;
       prod := x * y;
   END;
   $$ LANGUAGE plpgsql;

如在Section 35.4.4中的讨论,这将为这个函数的结果创建一个匿名的记录类型。如果使用了RETURNS字句,那么必须给它指明RETURNS记录。
另外一种方法声明PL/pgSQL函数是用RETURNS TABLE,例如:
   CREATE FUNCTION extended_sales(p_itemno int)
   RETURNS TABLE(quantity int, total numeric) AS $$
   BEGIN
        RETURN QUERY SELECT quantity, quantity * price FROM sales
                WHERE itemno = p_itemno;
   END;
   $$ LANGUAGE plpgsql;

这跟声明一个或者多个OUT参数和制定RETURNS SETOF这些类型是同样的方法。
当返回的PL/pgSQL函数的类型被声明为一个多态类型(anyelement, anyarray, anynonarray, 或者anyenum),特殊参数$0将被创建。它的数据类型将实际的返回函数的类型,从实际的输入类型返回(见Section 35.2.5)。这运行这个函数访问这个实际的返回类型如Section 39.3.3显示的那样。$0初始值为空并且能够被函数修改,如果需要,它可以用于保留返回值,虽然这不是必须的。$0也可以被给予一个别名。例如,这个函数能在任意一个有+操作符的数据类型上工作:
   CREATE FUNCTION add_three_values(v1 anyelement, v2 anyelement, v3 anyelement)
   RETURNS anyelement AS $$
   DECLARE
       result ALIAS FOR $0;
   BEGIN
       result := v1 + v2 + v3;
       RETURN result;
   END;
   $$ LANGUAGE plpgsql

声明一个或者多个多态类型的输出参数也是同样的效果。这种情况下这个特殊的$0参数将不会被用到,这个输出参数本身也是同样的作用,例如:
   CREATE FUNCTION add_three_values(v1 anyelement, v2 anyelement, v3 anyelement,
                                OUT sum anyelement)
   AS $$
   BEGIN
       sum := v1 + v2 + v3;
   END;
   $$ LANGUAGE plpgsql;


39.3.2. 别名
   newname ALIAS FOR oldname;
这个ALIAS语法比以前的章节中介绍的更加普通:你可以为任意一个变量声明一个别名,不只是函数的参数。这实际的用途是用预定义的名字为变量定义不同的名字,如触发器过程中的NEW或者OLD。 例子:
   DECLARE
      prior ALIAS FOR old;
      updated ALIAS FOR new;

因此,ALIAS使同样的对象有两种不同的方式命名,如果不限制的使用,将会变得混乱。这种方法最好只用于覆盖预定义的名字。



最后,贴出解决上面这个问题的存储过程吧:
CREATE OR REPLACE FUNCTION message_deletes(ids "varchar", userid int8)
  RETURNS int4 AS
$BODY$
DECLARE
  r RECORD;
  del bool;
  num int4 := 0;
  sql "varchar";
BEGIN
  sql := 'select id,receiveuserid,senduserid,senddelete,receivedelete from message where id in (' || ids || ')';
  FOR r IN EXECUTE sql LOOP
    del := false;
    IF r.receiveuserid=userid and r.senduserid=userid THEN
      del := true;
    ELSEIF r.receiveuserid=userid THEN
      IF r.senddelete=false THEN
        update message set receivedelete=true where id = r.id;
      ELSE
        del := true;
      END IF;
    ELSEIF r.senduserid=userid THEN
      IF r.receivedelete=false THEN
        update message set senddelete=true where id = r.id;
      ELSE
        del := true;
      END IF;
    END IF;
    IF del THEN
      delete from message where id = r.id;
      num := num + 1;
    END IF;
  END LOOP;
  return num;
END;
$BODY$
  LANGUAGE 'plpgsql' VOLATILE;




下面的例子是要调用一个存储过程自动创建对应的一系列表:
CREATE OR REPLACE FUNCTION create_table_for_client(id int)
RETURNS integer AS
$BODY$
DECLARE
num int4 := 0;
sql "varchar";
BEGIN
sql := 'create table _' || id || '_company(id int, name text)';
EXECUTE sql;
sql := 'create table _' || id || '_employee(id int, name text)';EXECUTE sql;
sql := 'create table _' || id || '_sale_bill(id int, name text)';EXECUTE sql;
.......
return num;
END;
$BODY$ LANGUAGE plpgsql VOLATILE



自动创建序列
第一个例子
CREATE OR REPLACE FUNCTION auto_gen_seq() RETURNS bigint AS
$BODY$
DECLARE 
rd RECORD;
num int4 := 0;
sql "varchar";  
seq_sql varchar;
BEGIN    
  sql := 'SELECT tablename FROM pg_tables WHERE tablename NOT LIKE ''pg%'' AND tablename NOT LIKE ''sql_%'' ORDER BY tablename;';  
  FOR rd IN EXECUTE sql LOOP  
    seq_sql:='CREATE SEQUENCE SQ_'||rd.tablename||' START 1000000 CACHE 30;';     
      BEGIN
        EXECUTE seq_sql;  
        EXCEPTION
          WHEN TOO_MANY_ROWS THEN
             RAISE EXCEPTION 'employee % not unique', seq_sql;
          WHEN OTHERS THEN
             return -1;
      END;
    num := num + 1;    
  END LOOP;    
  return num;    
END;
$BODY$
LANGUAGE plpgsql VOLATILE NOT LEAKPROOF
COST 100;

调用:
select auto_gen_seq()


第二个例子
-- Function: auto_gen_seq(character)

-- DROP FUNCTION auto_gen_seq(character);

CREATE OR REPLACE FUNCTION auto_gen_seq("tbName" character)
  RETURNS character varying AS
$BODY$/*
	调用示例:

SELECT tablename as tableName, 
'sq_'||REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(tablename, 'tb_am_', ''), 'tb_sm_', ''), 'tb_pm_', ''), 'tb_pc_', ''), 'tb_ps_', ''), 'rh_', ''), 'TB_', ''), 'RH_', '' ), 'tb_', '') AS sqName
,auto_gen_seq(tablename||'') as successFlag,current_date,current_time FROM pg_tables WHERE tablename NOT LIKE 'pg%' AND tablename NOT LIKE 'sql_%' ORDER BY successFlag,tablename;

*/

DECLARE  
rd RECORD;  
seq_sql varchar;  
flag_str varchar;
sq_name varchar;
sq_datetime varchar;
BEGIN

  seq_sql:='create table _sequence_table ( id SERIAL not null, code VARCHAR(200) null, increment_num INT8 null, minvalue_num INT8 null, maxvalue_num INT8 null, start_num INT8 null, cache_num INT8 null, cycle_flag VARCHAR(100) null,create_datetime timestamp without time zone,constraint PK__SEQUENCE_TABLE primary key (id) );CREATE UNIQUE INDEX INDEX__sequence_table ON _sequence_table (code);';
      BEGIN  
         EXECUTE seq_sql;    
         EXCEPTION  
            WHEN OTHERS THEN  
                flag_str:='失败';  
      END;

  --sq_name:=replace(replace($1, 'TB_', ''), 'RH_', '');
  --sq_name:=replace(replace(sq_name, 'tb_', ''), 'rh_', '');
  sq_name:='sq_'||REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE($1, 'tb_am_', ''), 'tb_sm_', ''), 'tb_pm_', ''), 'tb_pc_', ''), 'tb_ps_', ''), 'rh_', ''), 'TB_', ''), 'RH_', '' ), 'tb_', '');
  /*
  seq_sql:='drop SEQUENCE '||sq_name||';';       
    BEGIN  
      EXECUTE seq_sql;    
      EXCEPTION  
         WHEN OTHERS THEN  
             flag_str:='失败';  
      END; 
  */  
  seq_sql:='CREATE SEQUENCE '||sq_name||' START 1000000 CACHE 30;';       
    BEGIN  
      EXECUTE seq_sql;    
      EXCEPTION  
         WHEN OTHERS THEN  
             return '失败,创建序列';  
      END;   

  sq_datetime:=to_timestamp(current_date||' '||current_time,'yyyy-mm-dd hh24:mi:ss') ;
  seq_sql:='INSERT INTO _sequence_table( code,increment_num,minvalue_num,start_num, cache_num,create_datetime) VALUES ( '''||sq_name||''',1,1000000,1000000, 30,'''||sq_datetime||''');';       
    BEGIN  
      EXECUTE seq_sql;    
      EXCEPTION  
         WHEN OTHERS THEN  
             return '失败,插入序列信息';  
      END; 
      
  return '成功';      
END;
$BODY$
  LANGUAGE plpgsql VOLATILE STRICT
  COST 100;
ALTER FUNCTION auto_gen_seq(character)
  OWNER TO postgres;

调用
SELECT tablename as tableName, 
'sq_'||REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(tablename, 'tb_am_', ''), 'tb_sm_', ''), 'tb_pm_', ''), 'tb_pc_', ''), 'tb_ps_', ''), 'rh_', ''), 'TB_', ''), 'RH_', '' ), 'tb_', '') AS sqName
,auto_gen_seq(tablename||'') as successFlag,current_date,current_time FROM pg_tables WHERE tablename NOT LIKE 'pg%' AND tablename NOT LIKE 'sql_%' ORDER BY successFlag,tablename;
分享到:
评论

相关推荐

    Postgresql存储过程

    Postgresql存储过程详解 Postgresql存储过程是指在Postgresql数据库中定义的一组SQL语句的集合,它可以完成复杂的操作,并且可以重复使用。Postgresql存储过程可以用来实现业务逻辑,减少数据库服务器的压力和网络...

    PostgreSQL 存储过程调试

    调试PostgreSQL存储过程首先需要一个支持调试功能的客户端工具,例如pgAdmin或psql。在pgAdmin中,你可以通过图形界面设置断点,查看变量值和调用堆栈。在命令行工具psql中,可以利用 `\set` 命令设置变量,以及`\...

    Postgresql存储过程.docx

    PostgreSQL 存储过程详解 PostgreSQL 存储过程是一种强大的功能,可以实现复杂的逻辑操作和数据处理。下面将详细介绍 PostgreSQL 存储过程的结构、变量类型、连接字符、控制结构等知识点。 一、存储过程结构 存储...

    关于PostGreSQL中的存储过程

    PostgreSQL 存储过程详解 PostgreSQL 是一个开源的数据库管理系统,它提供了强大的数据存储和管理功能。...本文详细介绍了 PostgreSQL 中的存储过程,并提供了一个使用函数来查询数据的示例代码。

    postgresql存储过程

    PostgreSQL存储过程是一种数据库编程的方法,它允许用户在数据库中定义和执行复杂的操作逻辑,而不仅仅是简单的查询。存储过程在数据库管理系统中起着至关重要的作用,它们可以提高性能,增加安全性,减少网络流量,...

    SpringBoot 2.6.4 MyBatis PostgreSQL JUnit5 代码示例

    这是一个完整的Java Web开发示例,涵盖了从依赖管理到实际操作数据库以及进行测试的全过程。 首先,`pom.xml`是项目的Maven配置文件,它定义了项目所需的依赖。在这个案例中,我们能看到SpringBoot的父依赖(parent...

    PostgreSQL存储过程用法实战详解

    通过这个实例,我们可以学习到以下几点关于PostgreSQL存储过程的知识: 1. 存储过程的创建和替换:使用`CREATE OR REPLACE FUNCTION`语法。 2. 动态SQL的使用:允许在运行时构建SQL语句。 3. 安全地处理变量:使用`...

    Mybatis调用PostgreSQL存储过程实现数组入参传递

    例如,以下是一个接受整数数组作为参数的PostgreSQL存储过程: ```sql CREATE OR REPLACE FUNCTION "public"."func_arr_update"(ids _int4) RETURNS "pg_catalog"."void" AS $BODY$ DECLARE scount INTEGER; ...

    让postgresql支持存储过程(函数)的调试.doc

    在PostgreSQL数据库系统中,存储过程(也称为函数)是执行特定任务的预编译代码块,它们可以提高性能并简化复杂查询。然而,当这些过程或函数出现错误时,调试变得至关重要。本文将深入探讨如何使PostgreSQL支持存储...

    postgreSQL的CURD和存储过程

    本文将深入探讨PostgreSQL中的CURD操作以及存储过程,并结合jdbc、hibernate、ibatis三种不同的Java数据访问技术进行讲解。 **一、CURD操作** 1. **创建(Create)**:在PostgreSQL中,创建表是最基本的操作。例如,...

    初识PostgreSQL存储过程

    下面是一个简单的 PostgreSQL 存储过程示例: ```sql CREATE OR REPLACE FUNCTION 函数名(参数1,[整型 int4, 整型数组 _int4, …]) RETURNS 返回值类型 AS $BODY$ DECLARE 变量声明 BEGIN 函数体 END; $BODY$ ...

    运行在PostgreSQL中的AdventureWorks示例数据库

    本机环境(Windows、Postgresql 17、pgAdmin4), 按以下几步操作 1. 将文件包解压至D盘根目录,示例路径:D:\AdventureWorks-for-postgres\*.csv 2. 使用pgAdmin4在Postgres中创建AdventureWorks空数据库 3. 右键点击...

    PostgreSQL存储二进制数据.htm

    PostgreSQL存储二进制数据.htm

    一套C#与PostgreSQL数据库完美结合的实例

    本套程序是在VS2005下C#开发,...在对PostgreSQL数据库进行操作时,用到了如何调用存储过程来完成各项操作。 将App_Data下的数据库文件导入到PostgreSQL 8.3版里面,库名为:HYGL 然后修改web.config中的配置即可使用

    Node.js-基于Koa2-CoffeeScript-PostgreSQL的服务器开发示例

    在本示例中,我们将...Koa2提供简洁的中间件架构,CoffeeScript使JavaScript更易读写,而PostgreSQL则为数据存储提供了强大且灵活的平台。通过这个示例,你可以学习到如何将这些技术整合在一起,实现自己的Web服务。

    浅析PostgreSQL事务处理机制

    此外,PostgreSQL还允许用户轻松地定制和扩展,包括自定义函数、操作符、数据类型、索引以及过程语言。它可以运行在多种操作系统上,如Linux、UNIX、Windows等,并提供了丰富的编程接口支持,例如C/C++、Java、.NET...

    Python 操作 PostgreSQL 数据库示例【连接、增删改查等】

    以下是一个简单的示例,展示了如何使用Python连接到PostgreSQL数据库: ```python import psycopg2 conn = psycopg2.connect( database="testdb", user="postgres", password="pass123", host="127.0.0.1", ...

Global site tag (gtag.js) - Google Analytics