
本文介绍如何将顺序依赖、条件跳过的传统过程式算法重构为可维护的面向对象结构,重点推荐策略模式与责任链模式组合方案,避免继承滥用和重复代码,提升扩展性与测试性。
本文介绍如何将顺序依赖、条件跳过的传统过程式算法重构为可维护的面向对象结构,重点推荐策略模式与责任链模式组合方案,避免继承滥用和重复代码,提升扩展性与测试性。
在将旧有过程式算法迁移到面向对象范式时,一个常见误区是简单套用“抽象基类 + 子类重写”的继承模型——如为每种执行路径创建子类,并在 execute() 中硬编码调用顺序。这不仅导致大量重复(如 initializeVariables() 和 saveTemporaryData() 在所有子类中重复调用),还使新增用例需修改多处、难以单元测试,且违反开闭原则。
更优雅的解法是分离“流程编排”与“行为实现”。我们推荐两种互补模式:
✅ 推荐方案一:策略模式 + 流程引擎(推荐首选)
将每种完整执行路径定义为一个策略(ProcessingStrategy),流程引擎负责统一调度:
// 行为接口:各步骤职责清晰、无状态
interface DataProcessor {
void initializeVariables();
void saveTemporaryData();
void addDataFromExternalSource();
void addDataFromDraft();
void addDataFromPreviousVersion();
void sortData();
void prepareOutput();
}
// 策略接口:定义该用例的执行序列
@FunctionalInterface
interface ProcessingStrategy {
void execute(DataProcessor processor);
}
// 具体策略:仅声明“做什么”,不耦合实现细节
class UseCaseA implements ProcessingStrategy {
@Override
public void execute(DataProcessor processor) {
processor.initializeVariables();
processor.saveTemporaryData();
processor.addDataFromExternalSource();
processor.sortData();
processor.prepareOutput();
}
}
class UseCaseB implements ProcessingStrategy {
@Override
public void execute(DataProcessor processor) {
processor.initializeVariables();
processor.saveTemporaryData();
processor.addDataFromDraft(); // 条件分支在此体现
processor.addDataFromExternalSource();
processor.sortData();
processor.prepareOutput();
}
}使用时解耦清晰:
DataProcessor impl = new DefaultDataProcessor(); // 实现具体逻辑 ProcessingStrategy strategy = config.getStrategy(); // 运行时决定 strategy.execute(impl);
✅ 推荐方案二:责任链 + 条件处理器(适合动态跳过)
当跳过逻辑高度依赖运行时状态(如 draft != null),可构建可配置的责任链:
abstract class ProcessingStep {
protected ProcessingStep next;
public ProcessingStep setNext(ProcessingStep next) {
this.next = next;
return next;
}
public void handle(Context ctx) {
if (shouldExecute(ctx)) {
doExecute(ctx);
}
if (next != null) next.handle(ctx);
}
protected abstract boolean shouldExecute(Context ctx);
protected abstract void doExecute(Context ctx);
}
// 示例:仅当 draft 存在时才执行
class AddDataFromDraftStep extends ProcessingStep {
@Override
protected boolean shouldExecute(Context ctx) {
return ctx.getDraft() != null;
}
@Override
protected void doExecute(Context ctx) {
// 调用业务逻辑
ctx.getProcessor().addDataFromDraft();
}
}初始化链:
ProcessingStep chain = new InitializeStep()
.setNext(new SaveTempStep())
.setNext(new AddDataFromDraftStep())
.setNext(new AddDataFromExternalStep())
.setNext(new SortStep())
.setNext(new PrepareOutputStep());
chain.handle(new Context(draft, processor));⚠️ 注意事项与最佳实践
- 避免在方法内部做条件判断调用其他方法(如原答案中的 if(draft != null) addDataFromDraft()):这会将流程逻辑与业务逻辑混杂,破坏单一职责,且难以复用或替换某一步骤。
- 优先组合优于继承:用策略/链式结构替代类继承,降低耦合,便于测试(可单独 mock 某个策略或链节点)。
- 状态传递用 Context 对象:将共享数据(如 draft, tempData, result)封装进 Context,避免全局变量或参数爆炸。
- 配置驱动流程:将执行顺序和条件规则外置为 JSON/YAML,支持热更新用例而无需重新编译。
通过以上设计,您不仅能消除重复代码,还能让新用例的添加变成“新增一个策略类”或“调整配置文件”,真正实现高内聚、低耦合的 OOP 实践。










