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

044-代码味道-数据类

代码味道-数据类

代码坏味道:Data Class 的深度解析与重构实践

一、Data Class 的定义与特征

Data Class(数据类)是指仅包含数据字段和简单访问方法(如getter/setter),缺乏业务逻辑和职责的类。它本质上是"哑数据容器",常见于面向对象编程中,表现为以下特征:

  • 仅有数据存储:仅包含字段定义和基础读写方法
  • 无行为逻辑:不包含数据验证、计算等业务方法
  • 低内聚高耦合:其他类需直接操作其内部数据
  • 集合管理缺陷:若包含容器类字段(如列表/字典),常缺乏封装

二、C++ 中的 Data Class 实例演示

重构前的问题代码

// 纯数据类示例 
class StudentRecord {
public:std::string name;std::vector<int> scores;StudentRecord(std::string n, std::vector<int> s) : name(n), scores(s) {}
};// 外部业务逻辑 
class GradeCalculator {
public:float calculateAverage(const StudentRecord& record) {if(record.scores.empty())  return 0.0f;int sum = 0;for(int score : record.scores)  {sum += score;}return static_cast<float>(sum) / record.scores.size(); }
};

主要问题分析
StudentRecord
仅作为数据容器存在
核心业务逻辑(平均分计算)外置于其他类
scores字段直接暴露,违反封装原则

三、重构解决方案与过程

重构流程图

计算逻辑
集合操作
字段访问
识别Data Class
分析数据操作点
操作类型
Move Method到数据类
Encapsulate Collection
Hide Field
验证内聚性提升

重构后优化代码

class Student {
private:std::string name;std::vector<int> scores;public:Student(std::string n, std::vector<int> s) : name(std::move(n)), scores(std::move(s)) {}// 封装集合操作 void addScore(int score) {scores.push_back(score); }// 内聚业务逻辑 float calculateAverage() const {if(scores.empty())  return 0.0f;return std::accumulate(scores.begin(),  scores.end(),  0.0f) / scores.size(); }// 受控访问方法 const std::string& getName() const { return name; }
};

四、关键重构步骤解析

  1. 封装集合字段
+ void addScore(int score) {
+     scores.push_back(score); 
+ }

解决思路:通过方法控制集合修改,避免直接操作容器

  1. 搬移业务逻辑
float calculateAverage() const {// 计算逻辑内聚到数据类
}

重构技术:应用Move Method模式

  1. 访问控制优化
- std::string name;
+ private:
+    std::string name;

设计原则:遵循最小公开原则

五、重构效果对比

指标重构前重构后
类职责单纯数据存储数据+业务聚合
方法调用次数外部类多次调用单点调用
修改影响范围需修改多处调用点仅修改类内部实现
单元测试难度需mock外部依赖自包含可测试

六、设计模式应用建议

  1. 策略模式(计算策略)
class ScoreStrategy {
public:virtual float calculate(const std::vector<int>&) = 0;
};class Student {// 注入不同的计算策略...
};
  1. 工厂模式(对象构建)
class StudentFactory {
public:static Student createFromDB(int id) {// 数据库查询逻辑...}
};

七、最佳实践原则

  • 单一职责原则:每个类应专注于单一功能域
  • 迪米特法则:对象应最小化对其他对象的了解
  • 开闭原则:通过扩展而非修改实现功能变更
  • 组合优于继承:使用组合封装数据行为

通过系统化的重构过程,Data Class可转变为具备业务能力的成熟对象。建议在代码审查阶段使用SonarQube等静态分析工具检测此类坏味道,结合持续重构保持代码健康度。

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

相关文章:

  • 《Deepseek从入门到精通》清华大学中文pdf完整版
  • 基于智能推荐的就业平台的设计与实现(招聘系统)(SpringBoot Thymeleaf)+文档
  • VTEP是什么
  • [MySQL数据库] SQL优化
  • MyBatis-Plus 常用用法总结
  • 电脑自带画图工具,提取颜色
  • 软考软件评测师——计算机组成与体系结构(CPU指令系统)
  • 脚本语言Lua
  • std::deque 底层实现结构
  • 组合问题(去重)
  • 【Java】Spring的声明事务在多线程场景中失效问题。
  • 欧拉计划 Project Euler 73(分数有范围计数)题解
  • 第七节第三部分:从JDK8开始接口新增的方法、接口的多继承、注意事项
  • 产品方法论与 AI Agent 技术的深度融合:从决策智能到价值创造
  • 串口共用错误排查指南(2025年5月15日 09:25)
  • AI大模型应用:17个实用场景解锁未来
  • 全国青少年信息素养大赛 Python编程挑战赛初赛 内部集训模拟试卷九及详细答案解析
  • 特种设备事故背后,叉车智能监控系统如何筑牢安全防线
  • 深度学习中--模型调试与可视化
  • 【电子通识】热敏纸的静态发色性能和动态发色性能测试方法
  • 济南网站哪家做的好/2022年关键词排名
  • 郑州优秀网站建设公司/万网域名查询
  • 分类信息网站建设方案/常州seo关键词排名
  • 东源县住房和城乡建设部网站/社群营销成功案例
  • css代码大全/seo排名赚app最新版本
  • 做网站公司圣辉友联/广州网站优化服务