惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

SecWiki News
SecWiki News
I
InfoQ
The Cloudflare Blog
人人都是产品经理
人人都是产品经理
博客园 - Franky
T
Tailwind CSS Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
量子位
博客园_首页
罗磊的独立博客
V
V2EX
李成银的技术随笔
大猫的无限游戏
大猫的无限游戏
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
T
True Tiger Recordings
Vercel News
Vercel News
Cyberwarzone
Cyberwarzone
Cisco Talos Blog
Cisco Talos Blog
F
Fox-IT International blog
D
Darknet – Hacking Tools, Hacker News & Cyber Security
M
Microsoft Research Blog - Microsoft Research
Know Your Adversary
Know Your Adversary
爱范儿
爱范儿
The Register - Security
The Register - Security
G
Google Developers Blog
The Hacker News
The Hacker News
Malwarebytes
Malwarebytes
S
Securelist
博客园 - 三生石上(FineUI控件)
Jina AI
Jina AI
T
Threat Research - Cisco Blogs
T
The Exploit Database - CXSecurity.com
S
SegmentFault 最新的问题
博客园 - 叶小钗
F
Fortinet All Blogs
Apple Machine Learning Research
Apple Machine Learning Research
宝玉的分享
宝玉的分享
博客园 - 聂微东
T
Threatpost
博客园 - 【当耐特】
D
Docker
P
Privacy & Cybersecurity Law Blog
www.infosecurity-magazine.com
www.infosecurity-magazine.com
G
GRAHAM CLULEY
V
Visual Studio Blog
C
Cisco Blogs
IT之家
IT之家
S
Security Archives - TechRepublic
Latest news
Latest news
阮一峰的网络日志
阮一峰的网络日志

ZDDHUB

PixelsMeasure 开发第二年总结 PixelsMeasure 开发一年总结 Swift/SwiftUI 踩坑记 为什么说 GPT 利好程序员 ChatGPT 编程实现 Web 数字水印 Web 数字水印探究 Micro Frontends for Mobile URL 加载系统(URL Loading System) Protocol Buffers GraphQL 从 0 到 1 开发一款 IOS 应用 - Swift 学习一个新技巧需要多久? 不停机数据库迁移 Rspec 如何 mock update 方法更新自己? Rails 使用 mysql2 出现的段错误 使用 Docker-compose 部署 Rails 应用到生产环境 Cocoa troubleshooting 独孤九剑 Dit (0x05) - 终端篇 Gem-based Jekyll theme 开发小记 Miscellaneous 前端手记 TodoMVC 之 Redux 篇 前端手记 TodoMVC 之 Server 篇 前端手记 TodoMVC 之 React 篇 前端手记 TodoMVC 之 CSS 篇 独孤九剑 Dit (0x04) - 测试篇 独孤九剑 Dit (0x03) - 缓存篇 英语小抄 LLDB debug Golang Make mistakes 大牛俱乐部上线啦 独孤九剑 Dit (0x02) - 数据结构篇 独孤九剑 Dit (0x01) - 总决 独孤九剑 Dit (0x00) - 我为什么要做 Dit 零值强制类型转换的使用 终端颜色输出重定向 Go语法简略 - 正则表达式 Makefile Go语法简略 - Duck框架探索 Go语法简略 - 依赖注入 Go语法简略 - web应用框架 Go语法简略 - 反射 Go语法简略 - 面向对象 Go语法简略 - 方法和接口 Go语法简略 - goroutine Go语法简略 - 基础篇 大牛 | 轻松科研 为 Android Studio 创建图标 未来这几年 Shell Git Vim 金庸答百问 论拖延症 Flex, A fast scanner generator 有理想的人 从虚拟到现实 常用视频转接口 Recognizer configuration on CentOS 整个世界清静了 《Python源码剖析》读书笔记
MV* 软件设计架构
2021-01-05 · via ZDDHUB

软件工程推荐设计高内聚,低耦合的程序,方便代码复用,测试和维护。除了微观上符合 SOLID 原则外,宏观上良好的分层/分类也能极大的降低实体之间的耦合。

客户端应用程序实体可以分成三大类:

  • 模型(Model): 负责数据的更新(CRUD)
  • 视图(View): 负责 UI 的绘制和交互
  • *: 其它实体。负责在 MV 之间传递数据和业务逻辑,常见的抽象有 Controller,Presenter 和 ViewModel 等

根据对其它实体的抽象,出现了一系列 MV* 软件设计架构。本文整理并比较不同设计架构的优缺点和使用场景。

MV

Model-View (MV) 是最简单的一种实现。View 拥有 Model 所有操作权限,负责根据用户输入更新 Model,刷新自己。

MV diagram

优点

  • 代码简单,不需要中间层

缺点

  • Model 和 View 耦合度高,View 负责更新 Model
  • 混有业务逻辑的 View 很难复用

适用场景

  • 简单原型项目
  • View 只读 Model 数据,不更新

MVC

Model-View-Controller (MVC) 是最常用的架构,最原始的实现如下:

MVC diagram

优点

  • 不直接通过 View 更新 Model
  • 使用广泛,被多次工程实践所验证。有多个变种,针对领域问题有成熟的解决方案
  • 易用性良好

缺点

  • Model 和 View 有绑定,没有完全解耦
  • View 和 Controller 强耦合
  • Controller 可以负责 View 的创建,可避免
  • Controller 负责与 View 交互和业务逻辑,很容易写出臃肿的 Controller。在实际使用时,可将 Controller 进一步拆分。

适用场景

  • 多数项目,保底选择

MVC 有很多变种,比如 Cocoa 的实现:Model 和 View 完全解耦。

Apple MVC diagram

还有一种后端常用的架构,直接从 Controller 开始,到 Render 结束:

Backend MVC diagram

MVP

Model-View-Presenter (MVP) 是 MVC 的另一种演变。把构建 View 的代码从 Controller 中剥离出去(放在 View 中),命名为 Presenter。View 获得用户事件后,把 Event 发给 Presenter 处理。Presenter 通常使用接口(interface)与 View 通信,只负责处理 View 中的用户交互事件,不涉及视图逻辑(比如:状态管理和数据绑定等高级功能)。

MVP

优点

  • 不直接通过 View 更新 Model
  • Presenter 不负责 View 的创建
  • Presenter 只负责处理 View 的行为,不负责 View 的状态管理

缺点

  • Presenter 比较薄,导致 Model 和 View 比较重,一般使用 Supervising Controller 给 Model 减负
  • 允许 Model 和 View 有绑定,没有完全解耦
  • View 仍然包含状态管理、数据绑定等逻辑

适用场景

  • 一般来说,能使用 MVC 的场景就能使用 MVP

MVVM

Model-View-ViewModel (MVVM) 首次出现在微软的 WPF 和 Silverlight 项目中,是 MVP 的变种。

MVVM diagram

如图所示,MVVM 和 Cocoa MVC 的实现非常接近,不同之处在于:Cocoa MVC 版本中的 View Controller 除了负责业务逻辑外,还负责创建 UI,和 View 的耦合更紧密。而 MVVM 中的 ViewModel 不创建任何 UI,只封装与 UI 相关的状态。ViewModel 相比于 ViewController 更加轻量。

MVVM 会利用绑定技术,View 变化时,ViewModel 会自动变化,反之亦然。所以,MVVM 模式又被称作:model-view-binder 模式

优点

  • Model 和 View 彻底解耦
  • ViewModel 只负责 View 状态的改变,不创建 UI
  • 业务逻辑独立于 Model 和 View,可单独测试

缺点

  • 数据绑定使得 Bug 很难被调试
  • 对于过大的项目,数据绑定和数据转化需要花费更多的内存(成本)
  • 给每个 View 设置一个 ViewModel 会增加代码量

适用场景

  • 对数据绑定友好支持的平台/项目
  • 高模块化应用

比较和总结

据我理解,MV* 架构的不同点体现在 Controller, Presenter, ViewModel 的职责和与 MV 的协作方式上。比较如下:

  职责 与 MV 的协作方式
Controller 操作数据并控制 View 在 MV 中占主导地位,同时控制 MV
Presenter 负责响应 View 的行为 连接 Model 和 View
ViewModel 负责 View 的状态 并响应 View 的行为 和 View 强(弱)绑定

在实际使用时,并不单一使用某一种架构。也不只分成三层,可以构建 Service,Storage,NetworkRequest 等多层。

参考资料

如果你喜欢这篇文章,欢迎赞赏作者以示鼓励