news 2026/4/16 9:24:22

postgresql存贮过程编写

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
postgresql存贮过程编写

我来为您详细介绍 PostgreSQL 存储过程的编写方法。PostgreSQL 从 11 版本开始引入了完整的存储过程(`PROCEDURE`)支持,在此之前通常使用函数(`FUNCTION`)来实现类似功能。

一、存储过程 vs 函数

特性 函数 (FUNCTION) 存储过程 (PROCEDURE)
返回值 必须有返回值 无返回值
事务控制 不能控制事务 可以使用 `COMMIT`/`ROLLBACK`
调用方式 `SELECT func()` 或 SQL 表达式 `CALL proc()`
适用场景 计算、查询 复杂业务逻辑、批量操作

二、基础语法

1. 创建存储过程

```sql
-- 基本结构
CREATE OR REPLACE PROCEDURE 过程名(
参数名 数据类型 [IN | OUT | INOUT],
...
)
LANGUAGE plpgsql
AS $$
DECLARE
-- 声明变量
BEGIN
-- 业务逻辑
EXCEPTION
-- 异常处理
END;
$$;
```

2. 完整示例

```sql
-- 示例1:简单的用户注册过程
CREATE OR REPLACE PROCEDURE sp_user_register(
p_username VARCHAR(50),
p_email VARCHAR(100),
p_password VARCHAR(255)
)
LANGUAGE plpgsql
AS $$
BEGIN
-- 检查用户名是否已存在
IF EXISTS (SELECT 1 FROM users WHERE username = p_username) THEN
RAISE EXCEPTION '用户名已存在: %', p_username;
END IF;

-- 插入新用户
INSERT INTO users (username, email, password, created_at)
VALUES (p_username, p_email, p_password, NOW());

RAISE NOTICE '用户 % 注册成功', p_username;
END;
$$;

-- 调用方式
CALL sp_user_register('john_doe', 'john@example.com', 'hashed_password');
```

三、常用控制结构

1. 条件判断

```sql
-- IF-THEN-ELSE
IF 条件 THEN
-- 语句
ELSIF 条件 THEN
-- 语句
ELSE
-- 语句
END IF;

-- CASE 表达式
CASE
WHEN 条件1 THEN 结果1
WHEN 条件2 THEN 结果2
ELSE 默认结果
END;
```

2. 循环结构

```sql
-- LOOP 循环
LOOP
EXIT WHEN 条件; -- 退出条件
END LOOP;

-- WHILE 循环
WHILE 条件 LOOP
-- 语句
END LOOP;

-- FOR 循环(整数范围)
FOR i IN 1..10 LOOP
-- 语句
END LOOP;

-- FOR 循环(查询结果集)
FOR rec IN SELECT * FROM users LOOP
RAISE NOTICE '用户: %', rec.username;
END LOOP;

-- FOREACH(遍历数组)
FOREACH item IN ARRAY arr LOOP
-- 语句
END LOOP;
```

四、事务控制(存储过程特有)

```sql
CREATE OR REPLACE PROCEDURE sp_batch_transfer()
LANGUAGE plpgsql
AS $$
DECLARE
v_count INT := 0;
BEGIN
-- 第一批操作
INSERT INTO logs (msg) VALUES ('开始批量转账');
COMMIT; -- 提交第一批

-- 第二批操作
UPDATE accounts SET balance = balance - 100 WHERE id = 1;
UPDATE accounts SET balance = balance + 100 WHERE id = 2;

GET DIAGNOSTICS v_count = ROW_COUNT;

IF v_count = 0 THEN
ROLLBACK; -- 回滚
RAISE EXCEPTION '转账失败';
ELSE
COMMIT; -- 提交第二批
END IF;

EXCEPTION
WHEN OTHERS THEN
ROLLBACK;
RAISE;
END;
$$;
```

五、游标操作(处理大数据量)

```sql
CREATE OR REPLACE PROCEDURE sp_process_large_table()
LANGUAGE plpgsql
AS $$
DECLARE
cur CURSOR FOR SELECT id, name FROM large_table WHERE processed = false;
rec RECORD;
BEGIN
OPEN cur;
LOOP
FETCH cur INTO rec;
EXIT WHEN NOT FOUND;

-- 逐行处理
UPDATE large_table
SET processed = true, updated_at = NOW()
WHERE id = rec.id;

-- 每1000条提交一次,避免长事务
IF rec.id % 1000 = 0 THEN
COMMIT;
END IF;
END LOOP;
CLOSE cur;
END;
$$;
```

六、动态 SQL

```sql
CREATE OR REPLACE PROCEDURE sp_dynamic_query(
p_table_name VARCHAR(50),
p_limit INT DEFAULT 100
)
LANGUAGE plpgsql
AS $$
DECLARE
sql_query TEXT;
rec RECORD;
BEGIN
-- 构建动态SQL(注意SQL注入风险)
sql_query := format('SELECT * FROM %I LIMIT %L', p_table_name, p_limit);

-- 执行动态查询
FOR rec IN EXECUTE sql_query LOOP
RAISE NOTICE '记录: %', rec;
END LOOP;

-- 或使用 EXECUTE ... INTO
EXECUTE 'SELECT COUNT(*) FROM ' || quote_ident(p_table_name) INTO v_count;
END;
$$;
```

> 安全提示:使用 `format()` 的 `%I`(标识符)和 `%L`(字面值)或 `quote_ident()`/`quote_literal()` 防止 SQL 注入。

七、异常处理

```sql
CREATE OR REPLACE PROCEDURE sp_safe_operation()
LANGUAGE plpgsql
AS $$
BEGIN
-- 主逻辑
INSERT INTO critical_table VALUES (...);

EXCEPTION
-- 唯一约束冲突
WHEN unique_violation THEN
RAISE NOTICE '记录已存在,跳过';

-- 外键约束冲突
WHEN foreign_key_violation THEN
RAISE EXCEPTION '关联数据不存在';

-- 检查约束冲突
WHEN check_violation THEN
RAISE EXCEPTION '数据不符合约束条件';

-- 捕获所有其他异常
WHEN OTHERS THEN
RAISE EXCEPTION '未知错误: % - %', SQLSTATE, SQLERRM;
-- 或使用 RAISE NOTICE 记录后忽略
END;
$$;
```

八、输出参数

```sql
CREATE OR REPLACE PROCEDURE sp_get_user_stats(
IN p_user_id INT,
OUT total_orders INT,
OUT total_amount NUMERIC
)
LANGUAGE plpgsql
AS $$
BEGIN
SELECT COUNT(*), COALESCE(SUM(amount), 0)
INTO total_orders, total_amount
FROM orders
WHERE user_id = p_user_id;
END;
$$;

-- 调用
CALL sp_get_user_stats(123); -- 返回结果集
```

九、最佳实践

1. 命名规范:使用 `sp_` 前缀区分存储过程
2. 参数命名:输入参数用 `p_` 前缀,输出用 `o_`,内部变量用 `v_`
3. 错误处理:始终包含 EXCEPTION 块
4. 事务粒度:合理控制事务范围,避免长事务
5. 权限控制:使用 `SECURITY DEFINER` 或 `SECURITY INVOKER`
6. 文档注释:添加详细的注释说明

```sql
COMMENT ON PROCEDURE sp_user_register IS
'用户注册存储过程
参数:
p_username - 用户名
p_email - 邮箱
p_password - 加密后的密码
异常:
用户名已存在时抛出异常';
```

十、管理命令

```sql
-- 查看所有存储过程
SELECT * FROM pg_proc WHERE prokind = 'p';

-- 修改存储过程
ALTER PROCEDURE sp_name RENAME TO new_name;

-- 删除存储过程
DROP PROCEDURE IF EXISTS sp_name;

-- 查看存储过程源码
SELECT pg_get_functiondef('sp_name'::regprocedure);
```

需要我针对某个具体业务场景(如订单处理、数据同步、报表生成等)编写更详细的存储过程示例吗?

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 9:24:13

python大学生志愿填报招生网站系统vue3

目录 志愿填报系统技术架构核心功能模块关键技术实现数据安全措施扩展功能建议 开发技术路线相关技术介绍核心代码参考示例结论源码lw获取/同行可拿货,招校园代理 :文章底部获取博主联系方式! 志愿填报系统技术架构 采用前后端分离设计,后端…

作者头像 李华
网站建设 2026/4/15 12:24:06

Edge-TTS 403错误的技术解析与解决方案探索

Edge-TTS 403错误的技术解析与解决方案探索 【免费下载链接】edge-tts Use Microsoft Edges online text-to-speech service from Python WITHOUT needing Microsoft Edge or Windows or an API key 项目地址: https://gitcode.com/GitHub_Trending/ed/edge-tts 在使用E…

作者头像 李华
网站建设 2026/4/16 9:23:35

2025大模型趋势入门必看:Qwen3开源系列+GPU按需计费方案

2025大模型趋势入门必看:Qwen3开源系列GPU按需计费方案 1. Qwen3-1.7B:轻量高效的新选择 如果你正在找一个既省资源又够聪明的小模型,Qwen3-1.7B可能就是那个“刚刚好”的答案。它不是动辄几十亿参数的庞然大物,而是一款专为实际…

作者头像 李华
网站建设 2026/4/14 10:24:52

Llama3-8B跨境电商客服:多语言支持部署优化指南

Llama3-8B跨境电商客服:多语言支持部署优化指南 1. 为什么选Llama3-8B做跨境客服? 做跨境电商的老板们常遇到这些头疼事: 客服要同时应付英语、法语、德语、西班牙语买家,招多语种人工成本高、培训难;用现成的SaaS客…

作者头像 李华
网站建设 2026/4/15 7:13:54

CANFD和CAN的区别:汽车电子通信协议全面讲解

以下是对您提供的博文内容进行 深度润色与专业重构后的技术文章 。全文已彻底去除AI痕迹、模板化表达与生硬结构,转而采用 资深嵌入式系统工程师口吻 ,以真实项目经验为底色,融合教学逻辑与工程直觉,语言自然流畅、节奏张弛有度,兼具技术深度与可读性。文中所有关键概…

作者头像 李华
网站建设 2026/4/15 5:48:44

MacOS下如何配置SGLang?详细步骤来了

MacOS下如何配置SGLang?详细步骤来了 SGLang(Structured Generation Language)不是另一个大模型,而是一个专为高效推理设计的结构化生成框架。它不替代LLM,而是让LLM跑得更快、更稳、更聪明——尤其适合在MacOS这类资…

作者头像 李华