代码异味目录
这是基于 Martin Fowler《Refactoring》(第 2 版)的代码异味参考目录。代码异味是更深层问题的表面症状,它们说明代码设计可能出了问题。
“代码异味通常是系统中更深层问题的表面信号。” — Martin Fowler
过度膨胀类异味
表示对象或函数已经大到不再好用。
长函数
迹象: - 函数超过 30-50 行 - 需要滚动才能看完整个函数 - 嵌套层级很多 - 需要注释来解释局部逻辑
为什么不好: - 难理解 - 难单独测试 - 改动容易波及其他逻辑 - 重复逻辑容易藏在里面
可用重构: - Extract Method - Replace Temp with Query - Introduce Parameter Object - Replace Method with Method Object - Decompose Conditional
示例(重构前):
function processOrder(order) {
// Validate order (20 lines)
if (!order.items) throw new Error('No items');
if (order.items.length === 0) throw new Error('Empty order');
// ... more validation
// Calculate totals (30 lines)
let subtotal = 0;
for (const item of order.items) {
subtotal += item.price * item.quantity;
}
// ... tax, shipping, discounts
// Send notifications (20 lines)
// ... email logic
}
示例(重构后):
function processOrder(order) {
validateOrder(order);
const totals = calculateOrderTotals(order);
sendOrderNotifications(order, totals);
return { order, totals };
}
大类
迹象: - 实例变量很多(>7-10) - 方法很多(>15-20) - 类名含糊(Manager、Handler、Processor) - 方法没有用到全部字段
为什么不好: - 违反单一职责原则 - 难测试 - 改动会波及无关功能 - 难以复用局部能力
可用重构: - Extract Class - Extract Subclass - Extract Interface
检测参考:
代码行数 > 300
方法数量 > 15
字段数量 > 10
基础类型沉迷
迹象: - 用基础类型表达领域概念(用 string 表示邮箱、用 int 表示金额) - 使用基础类型数组,而不是对象 - 用字符串常量表示类型码 - 大量 magic number / magic string
为什么不好: - 类型层面没有验证 - 逻辑散落在整个代码库里 - 容易传错值 - 缺少领域对象
可用重构: - Replace Primitive with Object - Replace Type Code with Class - Replace Type Code with Subclasses - Replace Type Code with State/Strategy
示例(重构前):
const user = {
email: 'john@example.com', // 只是字符串
phone: '1234567890', // 只是字符串
status: 'active', // 魔法字符串
balance: 10050 // 以分为单位的整数
};
示例(重构后):
const user = {
email: new Email('john@example.com'),
phone: new PhoneNumber('1234567890'),
status: UserStatus.ACTIVE,
balance: Money.cents(10050)
};
长参数列表
迹象: - 方法参数超过 4 个 - 一些参数总是一起出现 - 布尔参数改变方法行为 - 经常传 null / undefined
为什么不好: - 难正确调用 - 参数顺序容易搞混 - 暗示方法做了太多事 - 难以增加新参数
可用重构: - Introduce Parameter Object - Preserve Whole Object - Replace Parameter with Method Call - Remove Flag Argument
示例(重构前):
function createUser(firstName, lastName, email, phone,
street, city, state, zip,
isAdmin, isActive, createdBy) {
// ...
}
示例(重构后):
function createUser(personalInfo, address, options) {
// personalInfo: { firstName, lastName, email, phone }
// address: { street, city, state, zip }
// options: { isAdmin, isActive, createdBy }
}
数据泥团
迹象: - 同样的 3 个以上字段反复一起出现 - 参数总是成对或成组传递 - 类中的字段子集总是一起使用
为什么不好: - 逻辑重复 - 缺少抽象 - 难扩展 - 暗示隐藏的类应该存在
可用重构: - Extract Class - Introduce Parameter Object - Preserve Whole Object
示例:
// 数据泥团:坐标 (x, y, z)
function movePoint(x, y, z, dx, dy, dz) { }
function scalePoint(x, y, z, factor) { }
function distanceBetween(x1, y1, z1, x2, y2, z2) { }
// 提取 Point3D 类
class Point3D {
constructor(x, y, z) { }
move(delta) { }
scale(factor) { }
distanceTo(other) { }
}
面向对象滥用类异味
表示 OOP 原则没有被正确使用。
switch 语句
迹象: - 长 switch/case 或 if/else 链 - 多处出现相同 switch - 按类型码分支 - 新增 case 时需要改很多地方
为什么不好: - 违反开放 / 封闭原则 - 改动会扩散到所有 switch 位置 - 难扩展 - 往往说明缺少多态
可用重构: - Replace Conditional with Polymorphism - Replace Type Code with Subclasses - Replace Type Code with State/Strategy
示例(重构前):
function calculatePay(employee) {
switch (employee.type) {
case 'hourly':
return employee.hours * employee.rate;
case 'salaried':
return employee.salary / 12;
case 'commissioned':
return employee.sales * employee.commission;
}
}
示例(重构后):
class HourlyEmployee {
calculatePay() {
return this.hours * this.rate;
}
}
class SalariedEmployee {
calculatePay() {
return this.salary / 12;
}
}
临时字段
迹象: - 实例变量只在部分方法中使用 - 字段只在某些条件下设置 - 某些情况初始化过程很复杂
为什么不好: - 容易混淆:字段存在但可能为 null - 对象状态难理解 - 说明条件逻辑被隐藏了
可用重构: - Extract Class - Introduce Null Object - Replace Temp Field with Local
拒绝遗赠
迹象: - 子类没有用到继承来的方法 / 数据 - 子类重写方法只是为了不做事 - 继承被当成代码复用,而不是真正的 IS-A
为什么不好: - 抽象不对 - 违反里氏替换原则 - 继承层次误导人
可用重构: - Push Down Method / Field - Replace Subclass with Delegate - Replace Inheritance with Delegation
不同接口的相似类
迹象: - 两个类做的事情差不多 - 同一个概念却用了不同方法名 - 理论上可以互换使用
为什么不好: - 重复实现 - 没有公共接口 - 难切换
可用重构: - Rename Method - Move Method - Extract Superclass - Extract Interface
变更阻碍类异味
这类异味会让改动变得困难,一次改动需要波及很多地方。
发散式变化
迹象: - 一个类因为很多不同原因而被修改 - 不同区域的变更都会触发同一个类的编辑 - 类变成了“上帝类”
为什么不好: - 违反单一职责原则 - 变更频率高 - 容易产生合并冲突
可用重构: - Extract Class - Extract Superclass - Extract Subclass
示例:
一个 User 类同时因为这些原因变化:
- 身份验证变化
- 个人资料变化
- 计费变化
- 通知变化
→ 可以拆成:AuthService、ProfileService、BillingService、NotificationService
散弹式修改
迹象: - 一个改动需要编辑很多类 - 一个小功能要触碰 10+ 个文件 - 改动分散,难以一次找全
为什么不好: - 很容易漏改 - 耦合高 - 容易出错
可用重构: - Move Method - Move Field - Inline Class
检测参考: 如果新增一个字段需要改 5 个以上文件,就要警惕。
平行继承体系
迹象:
- 在一个继承体系中创建子类,另一个体系也得跟着创建
- 类前缀相似(如 DatabaseOrder、DatabaseProduct)
为什么不好: - 维护成本翻倍 - 两个层次之间耦合 - 容易漏掉一边
可用重构: - Move Method - Move Field - 消除其中一个层次
可舍弃类异味
表示有些东西已经不必要了,应该移除。
注释过多
迹象: - 注释在解释代码“做了什么” - 注释掉的代码 - 长期存在的 TODO / FIXME - 注释里带道歉语气
为什么不好: - 注释会过时 - 代码本身应该能自解释 - 死代码会造成混乱
可用重构: - Extract Method(让方法名解释意图) - Rename(通过命名提升清晰度) - 删除注释掉的代码 - Introduce Assertion
好与坏的注释:
// BAD: 解释做了什么
// 遍历用户并检查是否活跃
for (const user of users) {
if (user.status === 'active') { }
}
// GOOD: 解释为什么
// 只保留活跃用户,未活跃用户由清理任务处理
const activeUsers = users.filter(u => u.isActive);
重复代码
迹象: - 多处出现相同代码 - 只有少量差异的相似代码 - 复制粘贴模式
为什么不好: - 修复要改多处 - 容易不一致 - 代码库臃肿
可用重构: - Extract Method - Extract Class - Pull Up Method(在继承层次中) - Form Template Method
检测规则: 任何重复 3 次以上的代码都应该考虑提取。
惰性类
迹象: - 类做的事情不足以支撑它的存在 - 只是一个没有增值的包装器 - 过度设计的产物
为什么不好: - 增加维护开销 - 多了一层没必要的间接 - 有复杂度但没收益
可用重构: - Inline Class - Collapse Hierarchy
死代码
迹象: - 无法到达的代码 - 未使用的变量 / 方法 / 类 - 注释掉的代码 - 位于不可能条件分支后的代码
为什么不好: - 造成困惑 - 增加维护负担 - 降低理解速度
可用重构: - Remove Dead Code - Safe Delete
检测:
# 查找未使用的导出
# 查找未引用的函数
# IDE 的“unused”警告
臆想泛化
迹象: - 只有一个子类的抽象类 - 为“未来可能需要”而加的未使用参数 - 只是转发的函数 - 为一个用例设计的“框架”
为什么不好: - 复杂度没有收益 - 违背 YAGNI - 更难理解
可用重构: - Collapse Hierarchy - Inline Class - Remove Parameter - Rename Method
耦合类异味
表示类之间耦合过强。
Feature Envy
迹象: - 一个方法使用别的类的数据比自己更多 - 对另一个对象调用很多 getter - 数据和行为分离
为什么不好: - 行为放错地方了 - 封装性差 - 难维护
可用重构: - Move Method - Move Field - Extract Method(然后移动)
示例(重构前):
class Order {
getDiscountedPrice(customer) {
// 这里大量使用 customer 数据
if (customer.loyaltyYears > 5) {
return this.price * customer.discountRate;
}
return this.price;
}
}
示例(重构后):
class Customer {
getDiscountedPriceFor(price) {
if (this.loyaltyYears > 5) {
return price * this.discountRate;
}
return price;
}
}
不恰当的亲密
迹象: - 类之间互相访问私有细节 - 双向引用很多 - 子类知道父类太多细节
为什么不好: - 耦合高 - 改一边会连带另一边 - 难单独修改
可用重构: - Move Method - Move Field - Change Bidirectional to Unidirectional - Extract Class - Hide Delegate
消息链
迹象:
- 方法调用链很长:a.getB().getC().getD().getValue()
- 客户端依赖导航结构
- “火车残骸”式代码
为什么不好: - 很脆弱,任何一层变动都会坏 - 违反迪米特法则 - 绑定到对象结构
可用重构: - Hide Delegate - Extract Method - Move Method
示例:
// 差:消息链
const managerName = employee.getDepartment().getManager().getName();
// 更好:隐藏委派
const managerName = employee.getManagerName();
中间人
迹象: - 类只是把调用转发给另一个类 - 一半以上的方法都是委派 - 没有额外价值
为什么不好: - 多了一层没必要的间接 - 维护成本更高 - 架构令人困惑
可用重构: - Remove Middle Man - Inline Method
严重性指南
| 严重性 | 说明 | 处理方式 |
|---|---|---|
| Critical | 阻塞开发,导致 bug | 立刻修复 |
| High | 明显增加维护负担 | 本迭代修复 |
| Medium | 有问题但还能接受 | 近期计划修复 |
| Low | 小问题 | 视情况顺手修 |
快速检测清单
扫描代码时使用这个清单:
- [ ] 有没有超过 30 行的函数?
- [ ] 有没有超过 300 行的类?
- [ ] 有没有参数超过 4 个的函数?
- [ ] 有没有重复代码块?
- [ ] 有没有按类型码分支的 switch/case?
- [ ] 有没有未使用的代码?
- [ ] 有没有大量使用其他类数据的方法?
- [ ] 有没有很长的方法调用链?
- [ ] 有没有解释“是什么”而不是“为什么”的注释?
- [ ] 有没有应该封装成对象的基础类型?
延伸阅读
- Fowler, M. (2018). Refactoring: Improving the Design of Existing Code (2nd ed.)
- Kerievsky, J. (2004). Refactoring to Patterns
- Feathers, M. (2004). Working Effectively with Legacy Code