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 相同????
?
- 數據表
- 模式Schema
- 表的繼承和分區
- 常用數據類型
- 函數和操作符-一
- 函數和操作符-二
- 函數和操作符-三
- 索引
- 事物隔離
- 性能提升技巧
- 服務器配置
- 角色和權限
- 數據庫管理
- 數據庫維護
- 系統表
- 系統視圖
- SQL語言函數
- PL-pgSQL過程語言
- PostgreSQL 序列(SEQUENCE)
- PostgreSQL的時間-日期函數使用
- PostgreSQL 查看數據庫,索引,表,表空間大小
- 用以查詢某表的詳細 包含表字段的注釋信息
- PostgreSQL 系統表查看系統信息
- postgre存儲過程簡單實用方法
- PostgreSQL實用日常維護SQL
- PostgreSQL的時間函數使用整理
- 命令
- pg_ctl控制服務器
- initdb 初始化數據庫簇
- createdb創建數據庫
- dropdb 刪除數據庫
- createuser創建用戶
- dropuser 刪除用戶
- psql交互式工具
- psql命令手冊
- pg_dump 數據庫轉儲
- pg_restore恢復數據庫
- vacuumdb 清理優化數據庫
- reindexdb 數據庫重創索引
- createlang 安裝過程語言
- droplang 刪除過程語言
- pg_upgrade 升級數據庫簇
- 調試存儲過程
- 客戶端命令-一
- 客戶端命令-二
- 使用技巧
- PostgreSQL刪除重復數據
- postgresql 小技巧
- PostgreSQL的10進制與16進制互轉
- PostgreSQL的漢字轉拼音
- Postgres重復數據的更新一例
- PostgreSQL使用with一例
- PostgreSQL在函數內返回returning
- PostgreSQL中的group_concat使用
- PostgreSQL數據庫切割和組合字段函數
- postgresql重復數據的刪除
- PostgreSQL的遞歸查詢(with recursive)
- PostgreSQL函數如何返回數據集
- PostgreSQL分區表(Table Partitioning)應用 - David_Tang - 博客園
- PostgreSQL: function 返回結果集多列和單列的例子
- 利用pgAgent創建定時任務
- 淺談 PostgreSQL 類型轉換類似Oracle
- postgresql在windows(包括win7)下的安裝配置
- PostgreSQL簡介、安裝、用戶管理、啟動關閉、創建刪除數據庫 (2010-11-08 12-52-51)轉載▼標簽: 雜談分類: PostgreSQL
- PostgreSQL的generate_series函數應用
- PostgreSQL 8.3.1 全文檢索(Full Text Search)
- postgresql record 使用
- 備份恢復
- PostgreSQL基于時間點恢復(PITR)
- Postgresql基于時間點恢復PITR案例(二)
- Postgres邏輯備份腳本
- Postgres invalid command \N數據恢復處理