postgre存储过程简单实用方法 (过程语言: PL/pgSQL)
一,介绍常用的PL/pgSQL结构和语法:
1,结构
PL/pgSQL是一种块结构的语言,比较方便的是用pgAdmin III新建Function,填入一些参数就可以了。基本上是这样的:
CREATE OR REPLACE FUNCTION 函数名(参数1,[整型 int4, 整型数组 _int4, …])
RETURNS 返回值类型 AS
$BODY$
DECLARE
变量声明
BEGIN
函数体
END;
$BODY$
LANGUAGE ‘plpgsql’ VOLATILE;
2,变量类型 除了postgresql内置的变量类型外,常用的还有 RECORD ,表示一条记录。
赋值 :“变量 := 表达式;”
连接字符串的是“||”,比如 sql := ‘SELECT * FROM’ || table || ‘WHERE …’;
3,判断
IF 条件 THEN
…
ELSEIF 条件 THEN
…
ELSE
…
END IF;
4,循环 循环有好几种写法:
WHILE expression LOOP
statements
END LOOP;
还有常用的一种是:(从1循环到9可以写成FOR i IN 1..9 LOOP)
FOR name IN [ REVERSE ] expression .. expression LOOP
statements
END LOOP;
二 跟mysql对比较
1,postgre 中的limit不支持LIMIT #,# 这样的语法。
而是支持 LIMIT and OFFSET clauses 语法
mysql上面的两种方式都支持。
2,存储过程中在ibatis中的使用:
(1),mysql存储过程可以直接返回结果集,同时可以有out参数
例如:
存储过程:
CREATE PROCEDURE `test`
(IN _login VARCHAR(32),
IN _psw VARCHAR(32),
OUT _ret INTEGER(10),
OUT _id INTEGER(10),
OUT _name VARCHAR(32),
OUT _email VARCHAR(32),
OUT _phone VARCHAR(20),
OUT _active INTEGER(11))//同时返回多个结果集合
BEGIN
DECLARE CONTINUE HANDLER FOR NOT FOUND set _ret =-1;
set _ret = 0 ;
select id,name,email,phone,active
into _id,_name,_email,_phone,_active
from test
where tx_account.`loginname`=_login and tx_account.`password`=MD5(_psw) and active=1;
---------返回结果集-----
if _ret = 0 then
select a.id as id ,a.name as name,a.priority as priority
from test b left join test1 a on b.role=a.id
where b.account=_id;
end if;
END;
直接返回结果集
ibatis文件
<parameterMap id="testParameterMap" class="params">
<parameter property="loginname" jdbcType="VARCHAR" javaType="java.lang.String" mode="IN"/>
<parameter property="password" jdbcType="VARCHAR" javaType="java.lang.String" mode="IN"/>
<parameter property="ret" jdbcType="INTEGER" javaType="java.lang.Integer" mode="OUT"/>
<parameter property="id" jdbcType="INTEGER" javaType="java.lang.Integer" mode="OUT"/>
<parameter property="name" jdbcType="VARCHAR" javaType="java.lang.String" mode="OUT"/>
<parameter property="phone" jdbcType="VARCHAR" javaType="java.lang.String" mode="OUT"/>
<parameter property="email" jdbcType="VARCHAR" javaType="java.lang.String" mode="OUT"/>
<parameter property="active" jdbcType="INTEGER" javaType="java.lang.Integer" mode="OUT"/>
</parameterMap>
<procedure id="test" parameterMap="testMap" resultMap="AccountRoleResultMap">
{call test(?,?,?,?,?,?,?,?)}
</procedure>
dao 的实现
定义一个传参的map params ,
HashMap<String,Object> params = new HashMap<String,Object>();
//把需要的参数放到map中
params.put("id",account.getId());
params.put("ret",null);
params.put("loginname", null);
params.put("name", null);
params.put("email", null);
params.put("phone",null);
params.put("active", null);
定义一个list
List list=null;
list= (List)(getSqlMapClientTemplate().queryForList("test",params));
//上面这样操作就可以获得存储过程返回的结果集。
Object var;
var = params.get("ret");//从map 中获得制定的输出参数的值。
在mysql中不需要的ibatis的配置文件中,声明返回的结果集。
(2) postgre的函数返回结果集
在postgre中返回结果集一定要在ibatis中定义输出参数。
方法1:不能输出参数,使用直接返游标的方法
例如:
函数:
CREATE OR REPLACE FUNCTION test(IN _login VARCHAR(32))//只有输入参数
RETURNS
refcursor //制定返回类型为游标。
AS
$BODY$
declare video_cur refcursor;
BEGIN
open video_cur for
select id , title from test;
return video_cur ;//返回游标
END
$BODY$
LANGUAGE 'plpgsql' VOLATILE;
ALTER FUNCTION test(integer) OWNER TO postgres;
ibatis文件
<parameterMap id="testParameters" class="java.util.HashMap">
<parameter property="result" jdbcType="OTHER" javaType="java.sql.ResultSet" mode="OUT"/>//返回结果集
<parameter property="loginName" jdbcType="VARCHAR" javaType="java.lang.String" mode="IN"/>
<parameter property="loginPasswd" jdbcType="VARCHAR" javaType="java.lang.String" mode="IN"/>
</parameterMap>
<procedure id="test" resultMap="testResultMap" parameterMap="testParameters" >
{? = call test(?,?)}
</procedure>
上面的map文件描述了3个参数,按照调用方式: ? = call test(?, ?)的顺序,
第一个参数是返回结果集的,这里的jdbcType填写OTHER,javaType填写java.sql.ResultSet,
如果是ORACLE的存储过程通过游标返回结果集的话,jdbcType应该填写为ORACLECURSOR,
不过在PostgreSQL中不能用ORACLECURSOR,得用OTHER。
dao的实现:
定义map文件 parameters ;
List list;
HashMap<String, String> parameters = new HashMap<String, String>();
parameters.put("loginName", loginName);
parameters.put("loginPasswd", loginPasswd);
list=getSqlMapClientTemplate().queryForList("test", parameters);//这样来得到返回的结果集。
return list;
方法2: 同时返回多个结果,
函数:
CREATE OR REPLACE FUNCTION test(IN _login VARCHAR(32),
IN _psw VARCHAR(32),
OUT _ret INTEGER,
OUT _id INTEGER,
OUT _name VARCHAR(32),
OUT _email VARCHAR(32),
OUT _phone VARCHAR(20),
OUT _ref refcursor ---返回一个游标
)
RETURNS record
AS
$BODY$
declare video_cur refcursor;
BEGIN
select id,name,email,phone
into _id,_name,_email,_phone
from test
where tx_account.`loginname`=_login and tx_account.`password`=MD5(_psw) and active=1;
open _ref for
select id , title from test1;
END
$BODY$
LANGUAGE 'plpgsql' VOLATILE;
ALTER FUNCTION test(integer) OWNER TO postgres;
如果返回多个结果集,就要使用返回伪类型 record可以在输出参数中指定游标为其中一个out参数
ibatis文件
out 参数输出游标
<parameterMap id="ParameterMap" class="map" >
<parameter property="login " jdbcType="VARCHAR" javaType="java.lang.String" mode="IN"/>
<parameter property="password" jdbcType="VARCHAR" javaType="java.lang.String" mode="IN"/>
<parameter property="ret" jdbcType="INTEGER" javaType="java.lang.Integer" mode="OUT"/>
<parameter property="id" jdbcType="INTEGER" javaType="java.lang.Integer" mode="OUT"/>
<parameter property="name" jdbcType="VARCHAR" javaType="java.lang.String" mode="OUT"/>
<parameter property="phone" jdbcType="VARCHAR" javaType="java.lang.String" mode="OUT"/>
<parameter property="email" jdbcType="VARCHAR" javaType="java.lang.String" mode="OUT"/>
<parameter property="ref" jdbcType="OTHER" javaType="java.sql.ResultSet" mode="OUT"/> //返回结果集
</parameterMap>
<procedure id="test" parameterMap="ParameterMap" resultMap="ResultMap">
{call test(?,?,?,?,?,?,?,?)}
</procedure>
dao的实现跟方法1 相同
分享到:
相关推荐
7. **存储过程与函数**:编写PL/pgSQL等语言的存储过程和函数,提高代码复用和数据库内计算能力。 8. **性能监控与调优**:通过`pg_stat_activity`、`pg_stat_user_tables`等视图监控数据库性能,调整参数优化性能。...
4. **游标和PL/SQL兼容性**:尽管PostgreSQL原生支持PL/pgSQL,但orafce提供了更接近Oracle的PL/SQL语法,包括游标的使用和处理。 5. **序列对象**:Oracle中的序列在PostgreSQL中可以通过创建序列类型来模拟,...
- PostgreSQL有更强大的函数和过程支持,如PL/pgSQL,而MySQL有自己的存储过程和函数语法。 8. 分区表: - MySQL支持分区表,PostgreSQL也有类似的功能,但实现方式不同。 在实际转换过程中,可以使用自动化工具...
PL/pgSQL 是 PostgreSQL 内置的存储过程语言,这个教程详细介绍了如何编写和使用 PL/pgSQL 函数,包括流程控制语句、变量、异常处理等,以增强数据库的功能和性能。 6. **《SQL命令参考手册》** 这是一个全面的 ...
9. **函数和存储过程的移植示例**:文档可能详细介绍了将PL/SQL函数和存储过程移植到PL/pgSQL的过程,包括简单的函数、创建新函数的函数、涉及字符串操作和OUT参数的存储过程,以及更复杂的存储过程。 10. **总结**...
7. **PL/pgSQL编程语言**:内建的PL/pgSQL是一种类似SQL的编程语言,用于编写存储过程和触发器,提高了数据库的可编程性。 8. **ACID兼容**:保证了事务的原子性、一致性、隔离性和持久性,确保了数据的完整性。 ...
10. **扩展与集成**:PostgreSQL支持多种语言的存储过程(如PL/pgSQL、C),可以安装额外的模块和扩展,如地理空间支持、全文搜索、JSON处理等,以增强其功能并与其他应用系统无缝集成。 在学习和使用PostgreSQL时...
- PostgreSQL 8.3支持PL/pgSQL、Perl、Python、TCL等编程语言编写存储过程。 - 可以通过扩展插件接口添加更多功能,如PostGIS(地理信息系统)。 9. **升级与迁移**: - 考虑到PostgreSQL 8.3已经较旧,建议升级...
8. **扩展与社区**:PostgreSQL有一个活跃的开发者社区,提供了大量扩展,如PostGIS(地理空间数据)、PL/pgSQL(内建过程语言)、pg_stat_statements(性能监控)等。 9. **版本升级与兼容性**:随着时间推移,...
函数内部使用了PL/pgSQL,这是PostgreSQL的一种过程语言,它允许我们在存储过程中编写复杂的逻辑。在`seq_id`函数中,我们首先定义了一个时间戳的起始点`our_epoch`,并获取当前时间的毫秒值`now_millis`。通过取模...
在PostgreSQL的早期版本,可能需要使用PL/pgSQL这样的过程语言编写自定义函数来实现这个逻辑。然而,使用LATERAL联合,我们可以通过单个复杂的查询完成任务,无需额外的编程。 查询分为两部分。首先,我们找出每个...