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如何添加/删除输入法 Win11切换中英
- 如何在 ACF 中正确更新嵌套多层 Group 字
- Win11怎么关闭边缘滑动手势_Windows11
- Win11怎么查看显卡显存_查询Win11显卡详细
- Win11怎么开启远程桌面连接_Windows11
- Python列表推导式与字典推导式教程_简化代码高
- windows如何备份注册表_windows导出和
- Windows如何设置登录时的欢迎屏幕背景?(锁屏
- Win11怎么关闭任务栏小组件_Windows11
- php订单日志怎么按金额排序_php按订单金额排序
- Windows11怎么用“记事本”自动换行与编码
- 如何在Golang中实现WebSocket广播_使
- 如何使用 Selenium 正确获取篮球参考网站球
- 如何用::实现单例模式_php静态方法与作用域操作
- Win11怎么关闭自动调节亮度 Win11禁用内容
- C++如何获取CPU核心数?(std::threa
- c# 在ASP.NET Core中管理和取消后台任
- XSLT怎么生成动态的HTML属性名和标签名
- 如何使用Golang进行HTTP服务性能测试_测量
- 如何使用Golang匿名函数_快速定义临时函数逻辑
- Go语言中正确反序列化多个同级XML元素为结构体切
- 如何在Golang中写入JSON文件_保存结构体数
- Mac的“预览”如何合并多个PDF_Mac文件处理
- php文件怎么变mp4保存_php输出视频流保存为
- Win11怎么清理C盘系统日志_Win11清理系统
- Django密码修改后会话失效的解决方案
- Win11玩游戏全屏闪退怎么办_Win11全屏优化
- Win11截图快捷键是什么_Win11自带截图工具
- Win10怎么设置开机密码_Windows10账户
- php怎么操作Redis_Redis扩展连接与基本
- 如何在Golang中使用replace替换模块_指
- MAC如何设置网卡MAC地址克隆_MAC终端修改物
- Win11怎么修改DNS服务器 Win11设置DN
- c++获取当前时间戳_c++ time函数使用详解
- win11 OneDrive怎么彻底关闭 Win1
- Python函数缓存机制_lru_cache解析【
- Windows10如何更改盘符名称_Win10重命
- Windows蓝屏错误0x0000002C怎么解决
- c++ std::atomic如何保证原子性 c+
- C++ static_cast和dynamic_c
- Win11怎样彻底卸载自带应用_Win11彻底卸载
- Win11怎么清理C盘虚拟内存_Win11清理虚拟
- Windows10如何删除恢复分区_Win10 D
- mac怎么退出id_MAC退出iCloud账号与A
- Win10系统更新错误0x80240034怎么办
- Win11怎么自动隐藏任务栏_Win11全屏显示设
- Win11怎样安装企业微信_Win11安装企业微信
- php会话怎么开启_session_start函数
- VSC怎样用终端运行PHP_命令行执行脚本的步骤【
- Win11怎么更改电脑名称_Windows 11修


QQ客服