楼主: 北方有热忱
45 0

[图行天下] 【软件架构设计方法论(14)】粗粒度功能模块划分:从功能树到架构设计 [推广有奖]

  • 0关注
  • 0粉丝

等待验证会员

学前班

40%

还不是VIP/贵宾

-

威望
0
论坛币
0 个
通用积分
0
学术水平
0 点
热心指数
0 点
信用等级
0 点
经验
20 点
帖子
1
精华
0
在线时间
0 小时
注册时间
2018-5-15
最后登录
2018-5-15

楼主
北方有热忱 发表于 2025-11-17 18:16:47 |AI写论文

+2 论坛币
k人 参与回答

经管之家送您一份

应届毕业生专属福利!

求职就业群
赵安豆老师微信:zhaoandou666

经管之家联合CDA

送您一个全额奖学金名额~ !

感谢您参与论坛问题回答

经管之家送您两个论坛币!

+2 论坛币

文章目录

  • 开头引入
  • 核心知识点铺垫
  • 知识点1:功能树——需求分析的"功能清单"
  • 知识点2:功能分解≠结构分解——区分问题域和解决方案
  • 知识点3:从功能组到功能模块——高内聚、松耦合的核心原理
  • 知识点4:功能树的获得与评审——确保需求完整准确
  • 实战场景串联
  • 场景背景
  • 第一步:获得功能树
  • 第二步:评审功能树
  • 第三步:从功能组到功能模块划分
  • 第四步:验证模块划分结果
  • 总结收尾
  • 通用应用逻辑
  • 可落地的交付物

开头引入

功能模块划分是架构师的“核心技能”和“基本职责”——通过将系统拆分为功能模块,实现模块内高内聚、模块间松耦合,使大型系统变得易于管理。

正如Hassan Gomaa所言:“要成功应对大型软件系统的固有复杂性,必须提供一种将系统拆分为子系统的方法…通过拆分并仔细定义子系统之间的接口,每个子系统可以独立设计。”(本质上是控制复杂度)

本文通过功能树方法和酒店管理系统实战案例,帮助你掌握“从需求到设计”的模块划分方法。

核心知识点铺垫

知识点1:功能树——需求分析的“功能清单”

通俗理解:功能树如同一张“功能清单树”,将系统要实现的功能按照“功能类别→功能组→功能项”的层次结构组织起来,让你一目了然地看到系统的所有功能。

核心作用:功能树是需求分析的框架工具,帮助分析师逐层选择和确定系统必须具备的功能(Function)和特性(Feature),是需求分析阶段的成果。

核心要点的三个要点:

  1. 功能树的本质
    • 本质:功能树是功能分解结构,描述的是“问题域”(系统要解决什么问题),属于需求分析层。它回答的是“系统要做什么”,而不是“系统怎么做”。
    • 为什么需要功能树?因为大型系统的功能通常很多,如果不组织成树状结构,需求分析会变得混乱。功能树通过层次化的方式,让分析师能够系统化地梳理功能,避免遗漏。
    • 深层原理:功能树体现了“分而治之”的思想——将复杂的系统功能分解为多个层次,每个层次关注不同粒度的功能,从而降低需求分析的复杂度。
  2. 功能树 vs 功能模块结构图
    • 功能树:是功能分解结构,描述问题域,属于需求分析层,是架构师从上游(需求分析师)获取的。
    • 功能模块结构图:是结构分解结果,描述解决方案,属于设计层,是架构师自己设计的。
    • 为什么必须区分?因为软件架构设计的难点在于“架起从现实世界(问题域)到计算机世界(解决方案)的桥梁”。需求分析阶段关注“问题是什么”(功能树),架构设计阶段关注“怎么解决”(功能模块结构图)。如果混淆两者,会导致用需求分析的思维做架构设计,无法正确划分模块。
  3. 功能树的结构
    • 层次结构:功能树按照“功能类别→功能组→功能项”的层次组织,形成树状结构。
    • 为什么是树状结构?因为功能之间存在“隶属关系”和“支撑关系”——上层功能是下层的概括,下层功能是上层的细化。树状结构能够清晰地表达这种层次关系。
    • 实际例子:呼叫中心系统的功能树,顶层是“自动服务”、“人工服务”、“其他功能”等功能类别,每个类别下又有功能组(如“自动语音服务”),功能组下又有功能项(如“最新公告”、“用户自助管理”)。

功能树示例(呼叫中心系统):

呼叫中心系统
├── 自动服务
│   ├── 自动语音服务
│   │   ├── 最新公告
│   │   ├── 用户自助管理
│   │   └── 用户留言
│   ├── 自动传真服务
│   │   └── 发送传真服务
│   │       ├── 在线传真
│   │       └── 离线传真
│   └── 接受传真服务
├── 人工服务
│   ├── 被动服务
│   │   ├── 技术咨询
│   │   ├── 服务与产品购买
│   │   └── 投诉建议
│   └── 主动服务
│       ├── 电话营销
│       └── 满意度调查
└── 其他功能
    ├── 系统管理
    └── 知识管理

知识点2:功能分解≠结构分解——区分问题域和解决方案

通俗理解:功能分解像“列购物清单”(要买什么),结构分解像“规划购物路线”(怎么去买)。两者完全不同,但很多人容易混淆。

核心作用:理解功能分解和结构分解的区别,是正确进行模块划分的前提。功能分解关注“做什么”(问题域),结构分解关注“怎么做”(解决方案)。

核心要点的三个要点:

  1. 本质区别
    • 功能分解:将系统功能按照业务逻辑进行分解,形成功能树。它描述的是“问题域”——系统要解决什么问题,要提供什么功能。
    • 结构分解:将系统结构按照技术实现进行分解,形成功能模块结构图。它描述的是“解决方案”——系统怎么实现这些功能,有哪些模块。
    • 为什么必须区分?因为软件架构设计是“从问题域到解决方案的桥梁”。需求分析阶段(功能分解)关注“问题是什么”,架构设计阶段(结构分解)关注“怎么解决”。如果混淆两者,会导致用需求分析的思维做架构设计,无法正确划分模块。
  2. 为什么容易混淆?
    • 表面相似:功能树和功能模块结构图都是树状结构,都涉及“分解”,容易让人以为是一回事。
    • 深层原因:很多人没有理解“问题域”和“解决方案”的区别,认为“功能”就是“模块”,导致直接用功能树作为模块划分的依据。
    • 实际危害:如果直接用功能树划分模块,会导致模块划分不合理——模块之间耦合度高,模块内聚度低,无法实现“高内聚、松耦合”的目标。
  3. 如何正确理解?
    • 功能树(问题域):描述“系统要做什么”,是功能分解结构,架构师从上游获得。
    • 功能模块结构图(解决方案):描述“系统怎么做”,是结构分解结果。

问题域 vs 解决方案示意图:

需求
  ↓
问题领域(功能树:功能分解)
  ↓
软件架构(转换桥梁)
  ↓
解决方案(功能模块结构图:结构分解)
  ↓
软件系统

功能模块结构图(结构分解)

描述“系统如何运作”,是架构设计的成果,由架构师自行设计。

转换过程

从功能树到功能模块结构图,需要架构师进行“设计转换”——分析功能之间的实现关系,将业务上紧密关联的功能组映射到功能模块,实现“高内聚、低耦合”。

知识点3:从功能组到功能模块——高内聚、低耦合的核心原理

通俗理解

就像“把相关的工具放在同一个工具箱”——业务上紧密关联的功能,实现时通常涉及相同的类和数据,应放在同一个模块中。

核心作用

将“功能组”映射到“功能模块”,是实现模块内高内聚、模块间低耦合的核心方法。通过分析功能的实现关系,将业务上紧密关联的功能组划分到同一个功能模块。

核心要点的3个要点

要点1:核心原理

原理:业务上紧密关联的一组功能,实现时通常涉及相似的类、相似的数据结构。因此,将“功能组”映射到“功能模块”,可以实现模块内的高内聚(功能相关、实现相关)、模块间的低耦合(不同模块的实现相对独立)。

为什么这样设计?因为功能的实现不是由单个类完成的,而是由多个相互协作的软件元素共同完成的。如果业务上紧密关联的功能涉及相同的软件元素,将它们放在同一个模块中,可以让这些元素共享,减少重复代码,提高内聚度。

深层原因:软件设计的核心目标是“高内聚、低耦合”。高内聚意味着模块内的元素紧密相关,低耦合意味着模块间的依赖关系简单。通过将功能组映射到功能模块,可以实现这个目标。

要点2:映射关系

功能组:功能树中业务上紧密关联的功能集合(如“办理预定”、“办理入住”、“办理退房”都是酒店前台服务相关的功能)。

功能模块:架构设计中实现这些功能的模块(如“宾客服务”模块)。

映射逻辑:分析功能的实现关系,如果一组功能涉及相同的类、相似的数据结构,就将它们映射到同一个功能模块。

为什么这样映射?因为这样可以实现代码复用(相同的类可以被多个功能共享)、降低耦合(不同模块的实现相对独立)、便于分工(一个模块可以分配给一个开发小组)。

要点3:通用模块的分离

通用模块:一些公共服务(如错误报告、日志记录、安全验证)同时支持多组功能的实现,不属于任何单一“功能模块”。

为什么需要分离?因为这些公共服务是多个功能模块都需要的基础设施,如果放在某个功能模块中,会导致其他模块依赖这个模块,增加耦合度。

如何分离?将这些公共服务独立模块化,放入对应的“通用模块”或“通用机制”中,让所有功能模块都可以使用,但不依赖特定的功能模块。

功能模块划分示意图

酒店管理系统
├── 功能模块(4个)
│   ├── 资产管理模块
│   ├── 宾客服务模块
│   ├── 人员考核模块
│   └── 系统维护模块
└── 通用模块(1个)
    └── 角色与权限管理框架

从功能组到功能模块的映射

功能树(问题域)             功能模块结构图(解决方案)
─────────────────          ──────────────────────
宾客服务(功能组)    →      宾客服务模块
├── 办理预定                ├── Reservation UI
├── 办理入住                ├── Checkin UI
└── 办理退房                ├── Checkout UI
                            ├── ServiceManager
                            ├── PayManager
                            ├── Reservation
                            └── Room

知识点4:功能树的获得与评审——确保需求完整准确

通俗理解

功能树像“需求清单”,需要从各种渠道收集,然后检查是否完整、是否准确。

核心作用

获得功能树是模块划分的第一步,评审功能树是确保需求完整准确的关键。只有获得准确完整的功能树,才能正确划分功能模块。

核心要点的3个要点

要点1:如何获得功能树?

三种方式:获取文档、沟通交流、分析产品。

获取文档:《软件需求规格说明书(SRS)》是最可能出现功能树的地方,因为SRS会系统化地描述功能,其“章-节-小节”结构往往就体现了功能树。更上游的文档如《愿景文档》《方案建议书》也可能包含功能树。

沟通交流:与业务人员、需求分析师沟通,收集功能信息,然后自己组织成功能树。

分析产品:分析本公司或竞争对手的产品(包括文档、市场材料、UI界面),提炼功能树。

为什么需要多种方式?因为功能树可能出现在不同的地方,单一渠道可能不完整。通过多种方式收集,可以确保功能树的完整性。

要点2:如何识别真正的功能树?

功能树 vs 页面流转图:功能树是功能分解结构,描述“系统要做什么”;页面流转图是页面跳转关系,描述“用户怎么操作”。页面流转图不能作为功能树,因为它描述的是交互流程,不是功能分解。

判断标准:功能树的上层节点是更高层次的“功能组”,下层节点是“子功能”,上下层节点之间是“隶属关系”和“支撑关系”。页面流转图描述的是“页面跳转关系”,不是功能分解关系。

为什么必须区分?因为如果用页面流转图作为功能树,会导致模块划分错误——按照页面跳转关系划分模块,无法实现“高内聚、低耦合”。

功能树 vs 页面流转图对比

? 页面流转图(不是功能树):

系统登录
├── 数据编辑
│   ├── 第一个
│   ├── 前一个
│   ├── 下一个
│   └── 添加/编辑/删除
├── 数据查询
│   ├── 搜索
│   └── 全部
└── 系统维护
    ├── 系统数据转换
    └── 修改系统密码

说明:这是页面跳转关系,不是功能分解

? 功能树(真正的功能树):

设备管理系统
├── 设备管理(功能组)
│   ├── 录入设备记录(功能项)
│   ├── 维护设备记录(功能项)
│   └── 维护维修记录(功能项)
├── 系统管理(功能组)
│   ├── 组织结构管理(功能项)
│   └── 设备类别管理(功能项)
└── 设备统计查询(功能组)
    ├── 维修统计查询(功能项)
    └── 调拨统计查询(功能项)

说明:这是功能分解结构,描述“系统要做什么”

要点3:如何评审功能树?

两个标准:用户导向、全面覆盖。

用户导向:功能树应该反映用户价值,从用户视角组织功能,而不是从技术视角。如果功能树从技术视角组织(如“金额计算”、“按键处理”),说明没有理解用户需求。

全面覆盖:功能树应该全面覆盖系统功能,不能遗漏重要功能。如果功能树遗漏了关键功能(如“故障报警”、“现金箱控制”),说明需求分析不完整。

为何需要评审?

因为功能树是模块划分的基础,如果功能树不精确、不全面,后续的模块划分就会出错。评审功能树可以及时发现需求问题,避免设计阶段的重复工作。

差的功能树示例(自动售货机管理系统)

自动售货机管理系统
├── 金额计算
│   ├── 金额累加
│   ├── 清除现有金额
│   └── 计算退币金额
└── 按键处理
    ├── 按键按下处理
    └── 点亮可售货等待

问题:

  • ? 从技术角度组织(“金额计算”、“按键处理”),而非用户角度
  • ? 忽略重要功能:故障报警、现金箱控制、加载人员功能等
  • ? 无法指导模块划分:哪些功能应放在哪个模块?

优质的功能树应具备:

  • ? 从用户角度组织(如"售货服务"、“设备管理”、“故障处理”)
  • ? 全面覆盖系统功能,不忽略关键功能
  • ? 能够指导模块划分

实战场景串联

场景背景

项目:某酒店开发酒店管理系统,需支持前台办理预订、办理入住、办理退房,经理进行人员考核,管理员进行系统维护等功能。

需求文档:提供了用例图,展示了不同角色的功能需求:

  • 前台:办理预订、办理入住、收取押金、办理退房
  • 经理:人员考核相关功能
  • 管理员:系统维护相关功能

问题:如何将这些功能需求转化为功能模块划分?如何实现"高内聚、低耦合"?

第一步:获取功能树

目标:从需求文档中获取功能树,明确系统要提供哪些功能。

核心操作要点:

要点1:从用例图提取功能

分析用例图,识别所有功能项:前台功能(办理预订、办理入住、收取押金、办理退房)、经理功能(人员考核)、管理员功能(系统维护)。

为何从用例图提取?因为用例图描述了系统要提供的功能,是功能树的重要来源。

要点2:组织成功能树结构

按照"功能类别→功能组→功能项"的层次组织功能。

对于酒店管理系统,可以组织为:

  • 宾客服务(功能类别)
  • 预订服务(功能组):办理预订(功能项)
  • 入住服务(功能组):办理入住、收取押金(功能项)
  • 退房服务(功能组):办理退房(功能项)
  • 人员考核(功能类别):人员考核相关功能(功能组)
  • 系统维护(功能类别):系统维护相关功能(功能组)
  • 资产管理(功能类别):资产管理相关功能(功能组)

要点3:补充通用功能

识别所有角色都需要的基础功能:登录、退出、修改密码。

这些功能不属于任何单一功能类别,应作为通用功能处理。

为何如此设计?

从用例图提取:因为用例图是需求分析的标准产出物,描述了系统要提供的功能,是功能树的重要来源。

组织成树状结构:因为功能之间存在层级关系,树状结构能够清晰地表达这种关系,便于后续的模块划分。

补充通用功能:因为有些功能是多个角色都需要的基础功能,如果不单独识别,会导致后续模块划分时不知如何处理。

第二步:评审功能树

目标:确保功能树用户导向、全面覆盖,为模块划分提供准确的基础。

核心操作要点:

要点1:检查用户导向

检查功能树是否从用户角度组织功能,而不是从技术角度。

好的功能树:从业务角度组织(如"宾客服务"、“人员考核”),用户能够理解。

差的功能树:从技术角度组织(如"金额计算"、“按键处理”),用户无法理解。

为何必须用户导向?因为功能树是需求分析的产出物,应反映用户价值。如果从技术角度组织,说明未理解用户需求,后续的模块划分也会出错。

要点2:检查全面覆盖

检查功能树是否涵盖了所有重要功能,无遗漏。

对于酒店管理系统,需检查:是否遗漏了"故障报警"、"现金箱控制"等关键功能?是否遗漏了"加载人员"的功能?是否遗漏了"资产管理"相关功能?

为何必须全面覆盖?因为功能树是模块划分的基础,如果遗漏了重要功能,后续的模块划分会不完整,导致系统功能缺失。

要点3:区分功能树和页面流转图

确保获得的是功能树(功能分解结构),而非页面流转图(页面跳转关系)。

功能树:描述"系统要做什么",按功能类别组织。

页面流转图:描述"用户如何操作",按页面跳转关系组织。

为何必须区分?因为页面流转图不能作为功能树,用它划分模块会导致模块划分错误。

为何如此设计?

检查用户导向:因为功能树应反映用户价值,从用户角度组织功能。如果从技术角度组织,说明需求分析有问题。

检查全面覆盖:因为功能树是模块划分的基础,如果遗漏了重要功能,后续的模块划分会不完整。

区分功能树和页面流转图:因为两者描述的内容不同,混淆会导致模块划分错误。

第三步:从功能组到功能模块划分

目标:将功能树中的功能组映射到功能模块,实现"高内聚、低耦合"。

核心操作要点:

要点1:分析功能的实现关系

分析每个功能的实现涉及哪些类、哪些数据结构。

对于酒店管理系统:

  • “办理预订”、“办理入住”、“办理退房”这三个功能,都涉及Room(房间)和Reservation(预订)等类。
  • PayManager(支付管理)和PrepayManager(预付费管理)职责相近、实现相似(都调用Finance Proxy)。
  • CheckinManager(入住管理)和CheckoutManager(退房管理)也以类似方式处理房间(Room)状态。

为什么分析实现关系?因为功能的实现不是由单一类完成的,而是由多个相互协作的软件元素共同完成的。如果业务上紧密关联的功能涉及相同的软件元素,应该将它们放在同一个模块中。

要点2:将功能组映射到功能模块

将业务上紧密关联的功能组映射到同一个功能模块。

对于酒店管理系统:

  • “办理预定”、“办理入住”、“办理退房”这些功能都涉及房间管理和预定管理,可以共享ServiceManager、PayManager、Reservation等程序实现,因此映射到“宾客服务”功能模块。
  • “人员考核”相关功能映射到“人员考核”功能模块。
  • “系统维护”相关功能映射到“系统维护”功能模块。
  • “资产管理”相关功能映射到“资产管理”功能模块。

为什么这样映射?因为这样可以实现模块内的高内聚(功能相关、实现相关)、模块间的松耦合(不同模块的实现相对独立),还便于将这组功能分配给一个程序小组负责开发。

要点3:分离通用模块

识别所有功能模块都需要的基础服务,将它们独立为通用模块。

对于酒店管理系统:

  • “登录”、“退出”、“修改密码”这些功能是所有角色都需要的基础功能,应该独立为“角色与权限管理”通用模块。
  • 错误报告、日志记录、安全验证等公共服务,也应该独立为通用模块或通用机制。

为什么需要分离?因为这些公共服务是多个功能模块都需要的基础设施,如果放在某个功能模块中,会导致其他模块依赖这个模块,增加耦合度。独立为通用模块,让所有功能模块都可以使用,但不依赖特定的功能模块。

为什么这么设计?

  • 分析实现关系:因为功能的实现涉及多个软件元素,只有分析实现关系,才能知道哪些功能应该放在同一个模块中。
  • 将功能组映射到功能模块:因为业务上紧密关联的功能,实现时往往涉及相同的软件元素,将它们放在同一个模块中,可以实现“高内聚、松耦合”。
  • 分离通用模块:因为通用服务是多个功能模块都需要的基础设施,独立为通用模块可以降低耦合度,提高复用性。

第四步:验证模块划分结果

目标:验证模块划分是否实现了“高内聚、松耦合”的目标。

核心操作要点:

要点1:检查模块内聚度

检查每个功能模块内的功能是否业务相关、实现相关。

对于“宾客服务”模块,检查“办理预定”、“办理入住”、“办理退房”这些功能是否涉及相同的类(如Room、Reservation),是否可以共享实现(如ServiceManager、PayManager)。

为什么检查内聚度?因为高内聚意味着模块内的元素紧密相关,如果模块内的功能不相关,说明模块划分不合理。

要点2:检查模块耦合度

检查不同功能模块之间的依赖关系是否简单。

对于酒店管理系统,检查“宾客服务”模块是否依赖“人员考核”模块?如果依赖,说明耦合度高,需要调整。

为什么检查耦合度?因为松耦合意味着模块间的依赖关系简单,如果模块之间相互依赖,说明模块划分不合理。

要点3:检查通用模块

检查通用模块是否被所有功能模块使用,但不依赖特定的功能模块。

对于“角色与权限管理”通用模块,检查是否所有功能模块都可以使用,但不依赖特定的功能模块。

为什么检查通用模块?因为通用模块应该被所有功能模块使用,但不应该依赖特定的功能模块,否则会增加耦合度。

为什么这么设计?

  • 检查内聚度:因为高内聚是模块划分的目标,如果模块内的功能不相关,说明模块划分不合理。
  • 检查耦合度:因为松耦合是模块划分的目标,如果模块之间相互依赖,说明模块划分不合理。
  • 检查通用模块:因为通用模块应该降低耦合度,如果通用模块依赖特定的功能模块,说明设计不合理。

总结收尾

通用应用逻辑功能模块划分的通用逻辑可以总结为:“获得功能树→评审功能树→分析实现关系→映射功能模块→分离通用模块→验证划分结果”。

通俗比喻:就像“整理工具箱”——先列出所有工具(获得功能树),检查工具是否齐全(评审功能树),分析哪些工具经常一起用(分析实现关系),把相关的工具放在同一个工具箱(映射功能模块),把通用工具单独放(分离通用模块),最后检查工具箱是否整理好了(验证划分结果)。

可落地的交付物

功能模块划分检查清单:

  • 功能树获得:
    • 从需求文档(SRS、愿景文档、方案建议书)中提取功能树
    • 与业务人员、需求分析师沟通,补充功能树
    • 分析产品(本公司或竞争对手),提炼功能树
    • 确保功能树是功能分解结构,不是页面流转图
  • 功能树评审:
    • 检查功能树是否用户导向(从用户视角组织功能)
    • 检查功能树是否全面覆盖(没有遗漏重要功能)
    • 区分功能树和页面流转图
  • 模块划分:
    • 分析功能的实现关系(涉及哪些类、哪些数据结构)
    • 将业务上紧密关联的功能组映射到功能模块
    • 分离通用模块(所有功能模块都需要的基础服务)
  • 验证结果:
    • 检查模块内聚度(模块内的功能是否业务相关、实现相关)
    • 检查模块耦合度(模块间的依赖关系是否简单)
    • 检查通用模块(是否被所有功能模块使用,但不依赖特定的功能模块)

最简操作模板:

功能模块划分三步走:
1. 获得功能树:从需求文档、沟通交流、分析产品中获得功能树
2. 评审功能树:检查用户导向、全面覆盖,区分功能树和页面流转图
3. 划分功能模块:分析实现关系→映射功能模块→分离通用模块→验证划分结果

记住

模块划分是架构师的“关键技能”和“基础职责”。借助功能树方法,将需求分析的功能拆分,转化为架构设计的结构拆分,达到“高度内聚、低度耦合”的目标,使大型系统更易于管理。

二维码

扫码加我 拉你入群

请注明:姓名-公司-职位

以便审核进群资格,未注明则拒绝

关键词:软件架构 方法论 reservation checkout function

您需要登录后才可以回帖 登录 | 我要注册

本版微信群
jg-xs1
拉您进交流群
GMT+8, 2025-12-21 13:04