当前位置: 首页 > article >正文

PostgreSQL 完整案例:重构遗留代码,搭建现代化工程架构

一、前言PostgreSQL 完整案例重构遗留代码搭建现代化工程架构是后端工程师必须掌握的核心技能。本文从PostgreSQL出发覆盖开发中最实用的知识点配有完整可运行的 SQL/代码示例。二、索引设计与优化2.1 索引类型选择-- 基础索引 CREATE INDEX idx_user_id ON orders(user_id); -- 联合索引遵循最左前缀原则 CREATE INDEX idx_user_status ON orders(user_id, status, created_at); -- 唯一索引保证数据唯一性 CREATE UNIQUE INDEX uk_email ON users(email); -- 前缀索引大字符串列节省空间 CREATE INDEX idx_title ON articles(title(20));2.2 索引失效的典型场景-- ❌ 索引失效函数操作导致无法使用索引 SELECT * FROM users WHERE YEAR(created_at) 2026; -- ✅ 正确使用范围查询 SELECT * FROM users WHERE created_at 2026-01-01 AND created_at 2027-01-01; -- ❌ 索引失效类型转换 SELECT * FROM orders WHERE user_id 12345; -- user_id 是 int -- ✅ 正确类型匹配 SELECT * FROM orders WHERE user_id 12345;三、慢查询分析与优化3.1 开启慢查询日志-- 查看慢查询配置 SHOW VARIABLES LIKE slow_query%; SHOW VARIABLES LIKE long_query_time; -- 设置慢查询阈值为 1 秒 SET GLOBAL slow_query_log ON; SET GLOBAL long_query_time 1; -- 分析慢查询 EXPLAIN SELECT u.name, o.total FROM users u JOIN orders o ON u.id o.user_id WHERE u.status active ORDER BY o.created_at DESC LIMIT 20;3.2 分页优化-- ❌ 深度分页问题OFFSET 越大越慢 SELECT * FROM orders ORDER BY id LIMIT 100000, 20; -- ✅ 优化使用游标分页基于上一页最后一条 ID SELECT * FROM orders WHERE id #{last_id} ORDER BY id LIMIT 20; -- ✅ 优化子查询先定位主键 SELECT * FROM orders WHERE id IN ( SELECT id FROM orders WHERE user_id 123 ORDER BY id LIMIT 20 OFFSET 100000 );四、事务与锁4.1 事务隔离级别-- 查看当前隔离级别 SELECT tx_isolation; -- 设置隔离级别 SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED; -- 开启事务 START TRANSACTION; UPDATE accounts SET balance balance - 100 WHERE user_id 1; UPDATE accounts SET balance balance 100 WHERE user_id 2; -- 提交或回滚 COMMIT; -- ROLLBACK; -- 出错时回滚4.2 行锁与死锁处理-- 显式加锁用于数据一致性要求高的场景 SELECT * FROM inventory WHERE product_id 100 FOR UPDATE; -- 锁定行防止并发修改 -- 死锁查看 SHOW ENGINE INNODB STATUS;五、千万级数据实战5.1 分库分表策略-- 按用户 ID 哈希分表8 张表 -- table_0 ~ table_7 hash(user_id) % 8 -- 查询路由逻辑应用层实现 function getTableName(userId) { const tableIndex userId % 8; return orders_${tableIndex}; }5.2 数据归档-- 历史数据归档到单独表 INSERT INTO orders_archive SELECT * FROM orders WHERE created_at DATE_SUB(NOW(), INTERVAL 1 YEAR); -- 删除已归档数据 DELETE FROM orders WHERE created_at DATE_SUB(NOW(), INTERVAL 1 YEAR); -- 定期执行建议低峰期 -- 0 2 * * 0 /usr/bin/mysql -e CALL archive_old_orders()六、总结索引不是越多越好——每个索引都增加写入开销慢查询分析是优化第一步——不要猜要实测深度分页用游标替代 OFFSET定期归档历史数据保持表轻盈收藏本文关注我后续更新更多数据库实战系列。三、实战进阶PostgreSQL 最佳实践3.1 错误处理与异常设计在生产环境中完善的错误处理是系统稳定性的基石。以下是 PostgreSQL 的推荐错误处理模式-- 错误处理使用事务保证数据一致性 CREATE OR REPLACE FUNCTION safe_transfer( from_id BIGINT, to_id BIGINT, amount DECIMAL(10,2) ) RETURNS BOOLEAN AS $$ DECLARE from_balance DECIMAL(10,2); BEGIN -- 加锁查询余额防止并发问题 SELECT balance INTO from_balance FROM accounts WHERE id from_id FOR UPDATE; IF from_balance amount THEN RAISE EXCEPTION 余额不足: 当前余额 %, 需要 %, from_balance, amount; END IF; UPDATE accounts SET balance balance - amount WHERE id from_id; UPDATE accounts SET balance balance amount WHERE id to_id; INSERT INTO transfer_logs(from_id, to_id, amount, created_at) VALUES(from_id, to_id, amount, NOW()); RETURN TRUE; EXCEPTION WHEN OTHERS THEN ROLLBACK; RAISE NOTICE 转账失败: %, SQLERRM; RETURN FALSE; END; $$ LANGUAGE plpgsql;3.2 性能监控与可观测性现代系统必须具备三大可观测性Metrics指标、Logs日志、Traces链路追踪。-- 慢查询监控与分析 -- 开启慢查询日志 SET GLOBAL slow_query_log ON; SET GLOBAL long_query_time 0.5; -- 超过 500ms 的查询记录 -- 实时查看正在执行的慢查询MySQL SELECT id, user, host, db, command, time AS seconds, LEFT(info, 100) AS query_preview FROM information_schema.processlist WHERE command ! Sleep AND time 1 ORDER BY time DESC; -- 查询统计PostgreSQL SELECT query, calls, total_exec_time / 1000 AS total_seconds, mean_exec_time AS avg_ms, rows / calls AS avg_rows FROM pg_stat_statements WHERE calls 10 ORDER BY mean_exec_time DESC LIMIT 20; -- 表大小与索引使用情况 SELECT schemaname, tablename, pg_size_pretty(pg_total_relation_size(schemaname||.||tablename)) AS total_size, seq_scan AS full_scans, idx_scan AS index_scans FROM pg_stat_user_tables ORDER BY pg_total_relation_size(schemaname||.||tablename) DESC;3.3 测试策略单元测试 集成测试高质量代码离不开完善的测试覆盖。以下是 PostgreSQL 推荐的测试实践-- 数据库测试使用事务回滚保证测试幂等 -- PostgreSQL 测试模式每次测试在事务中执行结束后回滚 BEGIN; -- 测试数据插入 INSERT INTO users (username, email, password_hash) VALUES (test_user, testexample.com, hashed_password); -- 验证插入结果 DO $$ DECLARE user_count INT; BEGIN SELECT COUNT(*) INTO user_count FROM users WHERE email testexample.com; IF user_count ! 1 THEN RAISE EXCEPTION Test failed: expected 1 user, got %, user_count; END IF; RAISE NOTICE Test passed: user inserted correctly; END $$; -- 测试唯一约束 DO $$ BEGIN BEGIN INSERT INTO users (username, email, password_hash) VALUES (test_user2, testexample.com, hash2); RAISE EXCEPTION Test failed: unique constraint not triggered; EXCEPTION WHEN unique_violation THEN RAISE NOTICE Test passed: unique constraint works correctly; END; END $$; ROLLBACK; -- 回滚所有测试数据不影响正式数据库3.4 生产部署清单上线前必检检查项具体内容优先级配置安全密钥不在代码中用环境变量或 VaultP0错误处理所有 API 有 fallback不暴露内部错误P0日志规范结构化 JSON 日志含 traceIdP0健康检查/health 接口K8s readiness/liveness probeP0限流保护API 网关或应用层限流P1监控告警错误率/响应时间/CPU/内存 四大指标P1压测验证上线前跑 10 分钟压测确认 QPS/延迟P1回滚预案蓝绿部署或金丝雀发布问题 1 分钟回滚P1四、常见问题排查4.1 PostgreSQL 内存占用过高排查步骤确认泄漏存在观察内存是否持续增长而非偶发峰值生成内存快照使用对应工具Chrome DevTools / heapdump / memory_profiler比对两次快照找到两次快照间新增且未释放的对象溯源代码找到对象创建的调用栈确认是否被缓存/全局变量/闭包持有常见原因全局/模块级变量无限增长缓存无上限事件监听器添加但未移除定时器/interval 未清理闭包意外持有大对象引用4.2 性能瓶颈在哪里通用排查三板斧数据库explain 慢查询加索引缓存热点数据网络 IO接口耗时分布P50/P90/P99N1 查询问题CPU火焰图flamegraph找热点函数减少不必要计算五、总结与最佳实践学习 PostgreSQL 的正确姿势先跑通再优化先让代码工作再根据性能测试数据做针对性优化了解底层原理知道框架帮你做了什么才知道什么时候需要绕过它从错误中学习每次线上问题都是提升的机会认真做 RCA根因分析保持代码可测试依赖注入、单一职责让每个函数都能独立测试关注社区动态订阅官方博客/Release Notes及时了解新特性和 Breaking Changes觉得有帮助点赞收藏关注持续更新 PostgreSQL 实战系列。觉得有用的话点个赞收藏关注我持续更新优质技术内容标签PostgreSQL | 重构 | 工程化 | 实战 | 案例

相关文章:

PostgreSQL 完整案例:重构遗留代码,搭建现代化工程架构

一、前言PostgreSQL 完整案例:重构遗留代码,搭建现代化工程架构是后端工程师必须掌握的核心技能。本文从PostgreSQL出发,覆盖开发中最实用的知识点,配有完整可运行的 SQL/代码示例。二、索引设计与优化2.1 索引类型选择-- 基础索引…...

大一小白也能拿奖?我们如何用HTML+CSS+JS做了一个考研计划网站参加C4网络技术挑战赛

大一团队如何用基础Web技术斩获C4网络技术挑战赛奖项 去年秋天,当宿舍里三位计算机专业的大一新生决定组队参加"中国高校计算机大赛-网络技术挑战赛"时,他们手里只有半学期学到的HTML/CSS基础知识和几节JavaScript入门课。令人意外的是&#x…...

如何在Windows电脑上无缝安装Android应用:告别模拟器的5步终极指南

如何在Windows电脑上无缝安装Android应用:告别模拟器的5步终极指南 【免费下载链接】APK-Installer An Android Application Installer for Windows 项目地址: https://gitcode.com/GitHub_Trending/ap/APK-Installer 你是否曾经想在Windows电脑上使用Androi…...

嵌入式系统模型检查与执行时间分析技术

1. 模型检查与可达性分析基础在嵌入式系统开发中,形式化验证技术正变得越来越重要。模型检查作为其中的关键技术之一,能够系统地验证系统是否满足特定的时态逻辑规范。让我们从一个实际案例开始理解这个概念。案例:汽车刹车系统验证想象一个电…...

别再只盯着Datasheet了!NS4225 D类音频功放外围电路设计避坑指南(附完整原理图与PCB文件)

NS4225 D类功放实战设计:从数据手册到稳定输出的全流程解析 在硬件设计领域,D类音频功放以其高效率、小体积的优势逐渐成为音频系统的首选方案。NS4225作为一款集成式D类功放芯片,数据手册上简洁的典型应用电路往往让工程师产生"照搬就能…...

Spring Boot项目里MySQL连接突然断开的排查与修复(附HikariCP配置)

Spring Boot项目中MySQL连接断开的深度排查与HikariCP优化实战 凌晨三点,监控系统突然发出刺耳的警报声——你的Spring Boot应用在夜间低峰期出现了大量"Communications link failure"错误。这不是第一次了,每次都是夜深人静时发生&#xff0c…...

别再死记硬背了!用‘头歌’实战项目,5分钟搞懂Java数组的声明、赋值与遍历

用实战项目解锁Java数组:从零构建学生成绩分析系统 很多Java初学者在第一次接触数组时,常常陷入"学完就忘"的困境。传统的语法点逐个讲解方式,虽然逻辑清晰,却难以让学习者真正理解数组在实际开发中的应用场景。本文将带…...

MySQL常见八股:索引

MySQL索引的最左前缀匹配原则是什么? 一句话概括:当MySQL在使用联合索引时,查询条件必须从索引的最左列开始匹配。这是因为联合索引在B树中的排列方式是"从左到右"的顺序。比如联合索引(first_name,last_name&#xff0…...

如何告别焦虑等待:Elsevier Tracker让学术投稿进度一目了然

如何告别焦虑等待:Elsevier Tracker让学术投稿进度一目了然 【免费下载链接】Elsevier-Tracker 项目地址: https://gitcode.com/gh_mirrors/el/Elsevier-Tracker 还在每天刷新Elsevier投稿页面,焦虑地等待审稿状态更新吗?Elsevier Tr…...

2025届学术党必备的五大AI辅助论文神器实测分析

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 在当下,人工智能生成内容越来越普及,好多用户面临着内容会被AI检测工…...

2025届毕业生推荐的五大AI学术神器推荐榜单

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 对于那些想要降低文本AI检测率的用户来讲,专业的降AI率网站能给出高效的解决办法…...

干货|GitHub 仓库推送避坑指南(附踩坑实录)

对了,分享一个我最近常看的AI人工智能学习渠道,讲得挺有章法的,不端着也不故弄玄虚。不感兴趣划走就行,感兴趣的可以自己去验证一下。 →传送门 干货|GitHub 仓库推送避坑指南(附踩坑实录) 文…...

从MobileNet到U-Net:聊聊那些‘非标准’卷积(空洞、深度可分离)在实战中的选择与调参

从MobileNet到U-Net:非标准卷积的工程实践指南 在计算机视觉领域,卷积神经网络(CNN)早已成为基础架构。但当我们从理论研究转向实际部署时,标准卷积操作往往难以满足多样化的工程需求——移动端需要极致的计算效率,医学图像分割要…...

告别鬼影!用PyTorch复现动态场景HDR融合论文,手把手教你搞定多曝光图像对齐与融合

动态场景HDR融合实战:PyTorch实现多曝光图像对齐与去鬼影技术 在数字摄影领域,高动态范围(HDR)成像技术一直是突破相机硬件限制的重要手段。当面对阳光直射的窗户与昏暗室内共存的场景时,单张照片往往难以同时保留亮部和暗部细节。传统解决方…...

别再死记硬背公式了!用Python+NumPy手把手带你理解矩阵白化(附完整代码)

用Python实战理解矩阵白化:从数学恐惧到代码掌控 很多数据科学初学者在面对矩阵白化这类数学概念时,常常陷入公式推导的泥潭而难以自拔。我们不妨换个思路——用Python代码和可视化手段,让抽象的数学原理变得触手可及。本文将带你用NumPy一步…...

从阶乘逆元到组合数计算:一个公式打通LeetCode刷题效率瓶颈

从阶乘逆元到组合数计算:一个公式打通LeetCode刷题效率瓶颈 在算法竞赛和LeetCode刷题中,组合数计算是许多动态规划和数论问题的核心操作。想象一下这样的场景:你正在解决一个需要频繁计算C(n, m) mod p的问题,每次调用都要重新计…...

用Python和NumPy动手实现8种DST变换:从公式到可视化基图像

用Python和NumPy动手实现8种DST变换:从公式到可视化基图像 在信号处理领域,离散正弦变换(DST)是一组与离散余弦变换(DCT)齐名的重要工具。不同于DCT的对称延拓特性,DST通过反对称延拓方式处理信…...

为什么90%的团队虚拟线程改造失败?揭秘3大反模式:阻塞IO、同步锁滥用、监控盲区(附诊断脚本)

第一章:虚拟线程的本质与高并发架构适配性再认知虚拟线程并非操作系统内核线程的简单封装,而是 JVM 在用户态实现的轻量级执行单元,其核心价值在于将“线程生命周期管理”从 OS 转移至运行时,从而解耦调度成本与并发规模。每个虚拟…...

【2024最硬核AI数据层教程】:用EF Core 10原生向量API构建低延迟RAG系统,实测P99<87ms

第一章:EF Core 10向量搜索扩展的演进与核心价值EF Core 10正式将向量搜索能力纳入官方生态,标志着ORM框架首次原生支持语义检索场景。这一演进并非简单叠加功能,而是深度整合了数据库向量索引、相似度计算与LINQ查询管道,使开发者…...

如何快速解锁NVIDIA消费级GPU虚拟化功能:完整操作指南

如何快速解锁NVIDIA消费级GPU虚拟化功能:完整操作指南 【免费下载链接】vgpu_unlock Unlock vGPU functionality for consumer grade GPUs. 项目地址: https://gitcode.com/gh_mirrors/vg/vgpu_unlock 在虚拟化环境中使用NVIDIA GPU加速一直是专业领域的特权…...

3分钟解锁B站缓存视频:免费开源m4s转MP4完整解决方案指南

3分钟解锁B站缓存视频:免费开源m4s转MP4完整解决方案指南 【免费下载链接】m4s-converter 一个跨平台小工具,将bilibili缓存的m4s格式音视频文件合并成mp4 项目地址: https://gitcode.com/gh_mirrors/m4/m4s-converter 你是否曾经在B站缓存了珍贵…...

告别繁琐操作!在Windows上轻松安装APK文件的终极指南

告别繁琐操作!在Windows上轻松安装APK文件的终极指南 【免费下载链接】APK-Installer An Android Application Installer for Windows 项目地址: https://gitcode.com/GitHub_Trending/ap/APK-Installer 你是否曾经遇到过这样的情况:在Windows电脑…...

用STM32和AD637搞定电路幅频特性测试:手把手教你复刻电赛D题核心模块

STM32与AD637构建的电路特性测试仪实战指南 在电子设计竞赛和实际工程中,快速准确地测量电路特性是每个硬件工程师的必备技能。本文将带你从零开始,用STM32微控制器和AD637真有效值检测芯片搭建一个功能完整的电路特性测试平台。不同于传统的赛题报告&am…...

Anaconda数据科学环境搭建:为千问3.5-9B模型服务准备Python生态

Anaconda数据科学环境搭建:为千问3.5-9B模型服务准备Python生态 1. 为什么需要Anaconda 在开始部署千问3.5-9B这类大模型之前,一个稳定、隔离的Python环境是必不可少的。Anaconda作为数据科学领域的瑞士军刀,能帮你轻松管理不同项目所需的P…...

从ProcessBuilder源码看Java进程创建:如何优雅地处理I/O流与子进程?

Java进程交互的深度实践:从ProcessBuilder源码到高效流处理 在分布式系统与自动化工具链开发中,Java进程管理能力直接影响着系统稳定性和资源利用率。当我们使用Runtime.getRuntime().exec()执行一个简单的ls命令时,背后究竟发生了多少层级的…...

Qwen3.5-2B模型处理网络协议分析:智能解析与异常流量识别

Qwen3.5-2B模型处理网络协议分析:智能解析与异常流量识别 1. 网络运维的痛点与AI解决方案 网络运维工程师每天都要面对海量的协议数据包和系统日志。传统分析方法需要人工逐条查看十六进制报文,或者编写复杂的过滤规则,效率低下且容易遗漏关…...

ComfyUI+Stable Audio Open:游戏开发者如何5分钟生成逼真环境音效(附实战案例)

ComfyUIStable Audio Open:游戏开发者如何5分钟生成逼真环境音效(附实战案例) 当你在深夜调试游戏场景时,突然发现缺少关键的环境音效——雨林中的虫鸣、古堡走廊的木质地板吱呀声、未来都市的悬浮车引擎嗡鸣。传统音效制作流程可…...

SAP ABAP开发避坑指南:BP业务伙伴的地址、银行、角色BAPI到底该怎么选?

SAP ABAP开发实战:BP业务伙伴BAPI选择策略与避坑技巧 每次打开SE37准备调用BP相关BAPI时,那些以BAPI_BUPA_开头的函数列表总让人眼花缭乱。上周刚踩过一个坑——用BAPI_BUPA_ADDRESS_CHANGE更新地址时,系统莫名其妙清空了邮政编码后三位。后来…...

别急着扔!华硕A555L老本升级实战:加内存、换系统,让它再战三年

华硕A555L老本重生指南:低成本升级方案与实战技巧 当手头的笔记本电脑开始力不从心,大多数人第一反应可能是"该换新机了"。但别急着把旧笔记本送进回收站——特别是像华硕A555L这样的机型,通过精准的硬件升级和系统优化&#xff0c…...

FrontPage练习题(3)

1、设置表单名称为“论坛个人信息设定表”。2、对照效果图fp:jp页面中尚有空缺的表单对象未完成插入。请插入空缺的表单对象,各对象的初始值见效果图。3、设置表单对象属性1:(1)设置表格第1行文本“论坛个人信息设定表…...