Python 代码重构方法与案例
技术百科
舞姬之光
发布时间:2026-01-27
浏览: 次 重构应先确认坏味道:频繁出错、测试易挂、扩展困难的代码才需动;提取函数需重命名变量、控制副作用、封装参数;类型判断应改为协议或注册表;测试是重构刹车系统,须覆盖边界、冻结时间、全量验证。
重构前先确认「坏味道」是否真需要动
不是所有看起来不顺眼的代码都值得重构。比如 if-else 嵌套三层,如果逻辑稳定、测试覆盖全、没人改它,强行拆成策略模式反而增加维护成本。真正该动的是:频繁修改却总出错的函数、单元测试一跑就挂的模块、新需求加进去要改五六个地方的类。
常见信号包括:duplicate code(相同逻辑在三处以上)、long method(单个函数超 40 行且含多层条件)、feature envy(一个方法总在调用另一个类的多个属性)。遇到这些,再动手不迟。
提取函数时别只剪切粘贴
把一段 30 行的逻辑剪出来单独成函数,只是第一步。关键在参数设计和副作用控制:
- 优先用
return传递结果,避免通过list.append()或dict.update()修改传入的可变对象——这会让调用方难以预料状态变化 - 参数超过 3 个时,考虑封装成
dataclass或命名元组,比如把process_order(user_id, item_id, discount_rate, is_vip, timestamp)改成process_order(order: OrderRequest) - 原函数里临时变量名如
tmp、res搬过去后必须重命名,否则重构后更难读
用 isinstance() 判断类型往往是重构起点
当看到类似这样的代码,就是典型的「类型分支坏味道」:
if isinstance(obj, PDFDocument):
return obj.render_pdf()
elif isinstance(obj, Markdo
wnDocument):
return obj.render_html()
elif isinstance(obj, EPUBDocument):
return obj.export_epub()
这不是不能运行,而是每次加新文档类型都要改这个 if-elif 链,违反开闭原则。可行路径有两条:
- 引入协议(
Protocol)或抽象基类,让各类实现统一接口render(),调用方完全不关心具体类型 - 用注册表模式:定义
RENDERERS = {PDFDocument: pdf_render, ...},新增类型只需往字典里塞键值对,不碰原有逻辑
选哪条取决于扩展频率——如果新格式每月加一种,注册表更轻量;如果已稳定在 4 种以内,协议更利于 IDE 类型推导。
测试不是重构的负担,是唯一刹车系统
没有测试就重构 Python,等于蒙眼过独木桥。重点不在覆盖率数字,而在「边界是否被卡住」:
- 对输入做
assert校验的函数,必须测ValueError是否如期抛出 - 涉及时间的逻辑(比如
is_expired()),别用datetime.now()硬写,用freezegun冻结时间,否则测试会随系统时钟飘移 - 重构后若发现某个测试从「通过」变「跳过」,不是漏写了,很可能是断言里依赖了被删掉的私有属性名
最常被忽略的一点:重构完别只跑自己改的那几个测试,至少执行一次完整模块的 pytest tests/ -k "not slow"——有些耦合藏得深,表面无关的改动会意外触发老 bug。
# 的是
# 多个
# 而在
# python
# markdown
# 都要
# 只需
# 没人
# 难以预料
# app
# 重命名
# 注册表
# 对象
# if
# html
# 接口
# 重构
# red
# bug
# 键值对
# 封装
# pdf
# ide
# elif
# append
# timestamp
# pytest
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- Win11怎么开启游戏模式_Windows11优化
- windows如何禁用驱动程序强制签名_windo
- php串口通信波特率怎么选_根据硬件手册设置正确波
- Win10如何卸载自带Edge_Win10彻底卸载
- Windows驱动无法加载错误解决方法_驱动签名验
- Win11怎么关闭右下角弹窗_Win11拦截系统通
- 微信短链接怎么还原php_用浏览器开发者工具抓包获
- 如何在JavaScript中动态拼接PHP的bas
- php订单日志怎么记录评价_php记录订单评价日志
- Win10怎样安装PPT模板_Win10安装PPT
- Mac自带的词典App怎么用_Mac添加和使用多语
- Windows蓝屏错误0x0000002C怎么解决
- Go语言中CookieJar的持久化机制解析:内存
- Go 中实现 Python urllib.quot
- Win10如何更改网络连接_Windows10以太
- Windows任务计划服务异常原因_任务调度失败的
- PHP怎么接收前端传的时间戳_处理时间戳参数转换技
- 如何使用Golang反射创建map对象_动态生成键
- 如何在Golang中处理数据库事务错误_回滚和日志
- 如何使用正则表达式批量替换重复的 *- 模式为固定
- Win11怎么打开注册表_Windows 11注册
- 如何在Golang中捕获JSON序列化错误_Gol
- Win11触摸板没反应怎么办_开启Win11笔记本
- Win11怎么设置桌面图标间距_Windows11
- Win11局域网共享怎么设置 Win11文件夹网络
- Windows10如何更改开机密码_Win10登录
- LINUX如何开放防火墙端口_Linux fire
- php8.4新语法match怎么用_php8.4m
- 如何用正则表达式精确匹配“start”到“end”
- Windows笔记本无法进入睡眠模式怎么办?(电源
- MAC如何快速搜索大文件_MAC磁盘空间分析与冗余
- VSC怎样用终端运行PHP_命令行执行脚本的步骤【
- Linux怎么设置磁盘配额_Linux系统Quot
- php内存溢出怎么排查_php内存限制调试与优化方
- Win11怎么开启上帝模式_创建Windows 1
- Django密码修改后会话失效的解决方案
- Windows11如何设置专注助手_Windows
- 如何在Golang中使用闭包_封装变量与函数作用域
- 如何用正则与预处理高效拦截带干扰符的恶意域名
- Win11声音太小怎么办_Windows 11开启
- Win10如何卸载WindowsDefender_
- Windows 11怎么更改锁屏超时时间_Wind
- php错误怎么开启_display_errors与
- Mac如何彻底清理浏览器缓存?(Safari与Ch
- Win10电脑怎么设置休眠快捷键_Windows1
- Win11用户账户控制怎么关_Win11关闭UAC
- 如何在Golang中实现服务熔断与限流_Golan
- Win11怎么设置鼠标宏_Win11鼠标按键自定义
- php报错怎么查看_定位PHP致命错误与警告的方法
- c++中如何对数组进行排序_c++数组排序算法汇总


QQ客服