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

MVC分层架构模式深入剖析

在这里插入图片描述

🔄 MVC 交互流程

User View Controller Model 点击按钮 调用Controller方法(如onButtonClick()) 修改数据(如model.setData()) 自动通知更新(通过观察者模式) 刷新UI User View Controller Model

💡 核心原则:View 的“被动性”

  1. View 只做两件事

    • 展示 UI 元素(按钮、文本框等)
    • 将用户事件转交给 Controller(自身不处理业务逻辑)
  2. Controller 是真正的决策者

    • 决定如何响应事件
    • 操作 Model 修改数据
    • 触发 View 更新(或由 Model 自动触发)

🧩 代码示例:PyQt 中的 MVC 实现

# Model(数据层)
class CounterModel:def __init__(self):self.value = 0self.observers = []  # 观察者列表(View)def add_observer(self, observer):self.observers.append(observer)def increment(self):self.value += 1# 数据变更后通知所有观察者(View)for observer in self.observers:observer.update(self.value)# View(界面层)
class CounterView(QWidget):def __init__(self, controller):super().__init__()self.controller = controllerself.label = QLabel("0")button = QPushButton("点击+1")# 关键点:View 捕获点击事件,但立刻转交给 Controllerbutton.clicked.connect(self.controller.handle_button_click)  layout = QVBoxLayout()layout.addWidget(self.label)layout.addWidget(button)self.setLayout(layout)def update(self, value):  # 被Model调用self.label.setText(str(value))# Controller(逻辑层)
class CounterController:def __init__(self):self.model = CounterModel()self.view = CounterView(self)self.model.add_observer(self.view)  # 注册View为观察者def handle_button_click(self):# View 点击事件最终由 Controller 处理self.model.increment()  # 修改Model# 主程序
app = QApplication([])
controller = CounterController()
controller.view.show()
app.exec_()

❓ 为什么不是 View 直接调用 Model?

若允许 View 直接操作 Model:

# 错误示例(违反 MVC)
class BadView(QWidget):def __init__(self, model):...button.clicked.connect(model.increment)  # View 直接调用 Model

会导致:

  1. 紧耦合:更换 Model 需修改 View 代码
  2. 无法复用:该 View 只能搭配特定 Model
  3. 测试困难:需启动完整 UI 才能测试逻辑

✅ MVC 的正确分工总结

组件职责
View1. 渲染界面
2. 转发用户事件 → Controller
Controller1. 接收 View 事件
2. 调用 Model 修改数据
3. 协调更新流程
Model1. 存储数据/逻辑
2. 数据变更后通知 View(观察者模式)

简单记忆:View 是秘书(只传递文件),Controller 是经理(做决策),Model 是仓库(保管数据)

http://www.lryc.cn/news/2403824.html

相关文章:

  • 【方案分享】蓝牙Beacon定位精度优化(包含KF、EKF与UKF卡尔曼滤波算法详解)
  • 新能源汽车热管理核心技术解析:冬季续航提升40%的行业方案
  • LeetCode 239. 滑动窗口最大值(单调队列)
  • 华为云Flexus+DeepSeek征文|DeepSeek-V3/R1开通指南及使用心得
  • 鸿蒙图片缓存(一)
  • 运行示例程序和一些基本操作
  • 学习数字孪生,为你的职业发展开辟新赛道
  • WebRTC源码线程-1
  • python学习打卡day47
  • MySQL中的内置函数
  • Ansible自动化运维全解析:从设计哲学到实战演进
  • YOLOv8n行人检测实战:从数据集准备到模型训练
  • 国标GB28181设备管理软件EasyGBS远程视频监控方案助力高效安全运营
  • 网络寻路--图论
  • LangChain4j 学习教程项目
  • 【Go语言基础【15】】数组:固定长度的连续存储结构
  • 【读论文】U-Net: Convolutional Networks for Biomedical Image Segmentation 卷积神经网络
  • Komiko 视频到视频功能炸裂上线!
  • Linux 文件系统与 I/O 编程核心原理及实践笔记
  • vite+tailwind封装组件库
  • Gin框架实战指南:从入门到进阶
  • 【Java学习笔记】包装类
  • 【高效开发工具系列】Blackmagic Disk Speed Test for Mac:专业硬盘测速工具
  • QtDBus模块功能及架构解析
  • 光学字符识别(OCR)理论概述与实践教程
  • 关键字--sizeof
  • Ubuntu20.04启动python的虚拟环境
  • 网页在线客服系统自动欢迎语实现方案(PHP+MySQL)
  • UniRig:如何在矩池云一站式解决 3D 模型绑定难题
  • 用函数实现模块化程序设计(适合考研、专升本)