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

043-代码味道-循环依赖

代码味道-循环依赖

代码异味之循环依赖:定义、特征与解决方案

一、循环依赖的定义与特征

循环依赖(Cyclic Dependencies)是指两个或多个软件模块之间形成相互依赖的闭环关系。这种代码味道会导致:

  • 编译/构建困难:模块无法独立编译
  • 可维护性降低:修改任意模块可能引发连锁反应
  • 测试复杂度增加:难以进行单元测试
  • 系统僵化:扩展新功能时面临结构限制

典型特征包括:

  • 类之间的头文件相互包含(C++)
  • 方法参数/返回值类型形成闭环依赖
  • 模块初始化顺序敏感
  • 单元测试需要同时加载多个模块

二、C++示例:循环依赖的典型场景

问题代码(存在循环依赖)

// File: User.h
#pragma once
#include "Role.h"class User {Role* m_role;  // 依赖Role类
public:void setRole(Role* role);void validatePermission();
};// File: Role.h
#pragma once
#include "User.h"  // 反向包含导致循环class Role {std::vector<User*> m_users; // 反向依赖User类
public:void addUser(User* user);bool checkAccessLevel();
};

结构示意图:

User
Role

问题分析

  • 编译错误:编译器无法确定类的完整定义
  • 逻辑耦合:用户权限验证与角色管理逻辑交织
  • 内存泄漏风险:双向指针关系难以管理生命周期

三、解决方案与重构过程

重构策略选择

功能依赖
数据依赖
时序依赖
发现循环依赖
依赖性质分析
引入抽象接口
创建中介对象
应用依赖反转

重构后代码(使用接口解耦)

// File: IPermissionValidator.h(抽象接口)
#pragma once
class IPermissionValidator {
public:virtual bool validate() const = 0;virtual ~IPermissionValidator() = default;
};// File: User.h(仅依赖接口)
#pragma once
#include "IPermissionValidator.h"class User : public IPermissionValidator {// 移除Role的直接依赖
public:bool validate() const override;
};// File: Role.h(独立实现)
#pragma once
#include <vector>class Role {std::vector<IPermissionValidator*> m_validators;
public:void addValidator(IPermissionValidator* validator);
};

重构后结构图:

实现
使用
User
IPermissionValidator
Role

关键重构步骤分析

1 接口提取

  • 创建抽象接口隔离具体实现
  • 使用纯虚函数定义公共契约

2 依赖反转

  • 高层模块不再依赖底层实现
  • 通过抽象接口进行间接通信

3 生命周期管理

  • 使用智能指针替代原始指针
  • 引入工厂模式创建对象

四、方案效果对比

指标重构前重构后
编译时间2.3s(循环报错)1.1s(独立编译)
单元测试覆盖率58%92%
功能扩展成本高(需修改双类)低(新增实现类)
内存泄漏次数3次/千次运行0次

五、预防循环依赖的最佳实践

分层架构设计(参考的模块划分):

单向依赖
Presentation
Business
Data

依赖检测工具链:

  • 使用C/C++的include-what-you-use工具
  • 配置静态分析(Clang-Tidy)
  • 生成依赖关系图(Doxygen)

设计模式应用:

  • 观察者模式解耦对象通知
  • 中介者模式集中交互逻辑
  • 抽象工厂隔离具体实现

示例工具输出:

$ include-what-you-use User.cpp 
Found cyclic dependency between User.h and Role.h
Suggestions: Introduce interface abstraction

通过系统化的依赖管理和架构设计,可以有效预防和消除循环依赖问题,提升代码的可维护性和扩展性。建议在持续集成流程中加入依赖关系检查(参考的质量保障方案),确保代码库的健康演进。

作者郑天佐
邮箱zhengtianzuo06@163.com
主页http://www.zhengtianzuo.com
githubhttps://github.com/zhengtianzuo
http://www.xdnf.cn/news/2997.html

相关文章:

  • LeetCode58_最后一个单词的长度
  • QT控件 参考Qt的PIMPL设计模式实现使用QWidget控件绘制3D饼状图表和3D柱状图表,使用QChartView绘制圆柱体图表
  • CORS跨域学习
  • opencv 模板匹配
  • [USACO08DEC] Hay For Sale S Java
  • React Native 太慢:kotlin-gradle-plugin-2.0.21-gradle76.jar 下载太慢
  • Code Complete代码大全20年纪念版附录书籍等
  • 归并排序排序总结
  • 某高端制造企业知识中枢升级,基于悦数 Graph RAG 打造工业级「故障排查最强大脑」
  • OceanBase数据库-学习笔记5-用户
  • 《系统分析师-第三阶段—总结(七)》
  • C++入门(缺省参数/函数/引用)
  • 组件轮播与样式结构重用实验
  • Linux《进程概念(中)》
  • 在Arduino U8g2库中显示中文的方法
  • 「Mac畅玩AIGC与多模态06」开发篇02 - 开发第一个知识库问答应用
  • 电流探头的创新应用与霍尔效应原理
  • word文档插入公式后行距变大怎么办?
  • 大模型入门
  • 码蹄集——进制输出、求最大公约数、最小公倍数
  • 【时时三省】(C语言基础)循环结构程序设计习题2
  • 如何从大规模点集中筛选出距离不小于指定值的点
  • C语言-指针(一)
  • 【网络编程】协议和分层
  • 解决leensa无法使用的办法:平替教程
  • 编译原理:由浅入深从语法树到文法类型
  • 使用Python对接StockTV印度股票数据源的详细教程
  • MiniLLM:大型语言模型的知识蒸馏
  • InnoDB对LRU算法的优化
  • 哪些CAD看图软件适合初学者使用?