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

MVVM、MVC、MVP 的区别

MVVM(Model-View-ViewModel)、MVC(Model-View-Controller)和MVP(Model-View-Presenter)是三种常见的软件架构模式,它们在客户端应用开发中被广泛使用。每种模式都有其特定的设计理念和应用场景,下面将详细介绍这三种模式的区别,并通过示例来说明。

1. MVC (Model-View-Controller)

核心思想

  • Model:负责数据的存储和业务逻辑。
  • View:负责展示数据,用户界面。
  • Controller:负责处理用户输入,更新模型和视图。

工作流程

  1. 用户与视图进行交互。
  2. 视图将用户的输入传递给控制器。
  3. 控制器根据输入调用模型的方法,更新模型的状态。
  4. 控制器通知视图更新以反映模型的变化。

优点

  • 分离关注点,使得代码结构清晰。
  • 便于维护和扩展。

缺点

  • 对于复杂的UI,控制器可能会变得臃肿。
  • 视图和控制器之间的耦合较高。

示例: 假设我们有一个简单的待办事项应用:

// Model
class TodoModel {constructor() {this.todos = [];}addTodo(todo) {this.todos.push(todo);}removeTodo(index) {this.todos.splice(index, 1);}
}// View
class TodoView {constructor(model, controller) {this.model = model;this.controller = controller;this.init();}init() {const addButton = document.getElementById('addButton');addButton.addEventListener('click', () => {const input = document.getElementById('todoInput');this.controller.addTodo(input.value);input.value = '';});const list = document.getElementById('todoList');this.model.todos.forEach((todo, index) => {const li = document.createElement('li');li.textContent = todo;const removeButton = document.createElement('button');removeButton.textContent = '删除';removeButton.addEventListener('click', () => {this.controller.removeTodo(index);});li.appendChild(removeButton);list.appendChild(li);});}update() {const list = document.getElementById('todoList');list.innerHTML = '';this.model.todos.forEach((todo, index) => {const li = document.createElement('li');li.textContent = todo;const removeButton = document.createElement('button');removeButton.textContent = '删除';removeButton.addEventListener('click', () => {this.controller.removeTodo(index);});li.appendChild(removeButton);list.appendChild(li);});}
}// Controller
class TodoController {constructor(model, view) {this.model = model;this.view = view;}addTodo(todo) {this.model.addTodo(todo);this.view.update();}removeTodo(index) {this.model.removeTodo(index);this.view.update();}
}const model = new TodoModel();
const view = new TodoView(model);
const controller = new TodoController(model, view);

2. MVP (Model-View-Presenter)

核心思想

  • Model:负责数据的存储和业务逻辑。
  • View:负责展示数据,用户界面。
  • Presenter:作为中间层,处理用户输入,更新模型,并控制视图的显示。

工作流程

  1. 用户与视图进行交互。
  2. 视图将用户的输入传递给Presenter。
  3. Presenter根据输入调用模型的方法,更新模型的状态。
  4. Presenter更新视图以反映模型的变化。

优点

  • 分离了视图和业务逻辑。
  • 测试更加容易。

缺点

  • Presenter可能变得复杂。
  • 视图和Presenter之间的接口需要仔细设计。

示例: 继续上面的待办事项应用,但这次使用MVP模式:

// Model
class TodoModel {constructor() {this.todos = [];}addTodo(todo) {this.todos.push(todo);}removeTodo(index) {this.todos.splice(index, 1);}getTodos() {return this.todos;}
}// View
class TodoView {constructor(presenter) {this.presenter = presenter;this.init();}init() {const addButton = document.getElementById('addButton');addButton.addEventListener('click', () => {const input = document.getElementById('todoInput');this.presenter.addTodo(input.value);input.value = '';});const list = document.getElementById('todoList');this.renderTodos(this.presenter.getTodos());}renderTodos(todos) {const list = document.getElementById('todoList');list.innerHTML = '';todos.forEach((todo, index) => {const li = document.createElement('li');li.textContent = todo;const removeButton = document.createElement('button');removeButton.textContent = '删除';removeButton.addEventListener('click', () => {this.presenter.removeTodo(index);});li.appendChild(removeButton);list.appendChild(li);});}
}// Presenter
class TodoPresenter {constructor(view, model) {this.view = view;this.model = model;}addTodo(todo) {this.model.addTodo(todo);this.view.renderTodos(this.model.getTodos());}removeTodo(index) {this.model.removeTodo(index);this.view.renderTodos(this.model.getTodos());}getTodos() {return this.model.getTodos();}
}const model = new TodoModel();
const view = new TodoView();
const presenter = new TodoPresenter(view, model);
view.presenter = presenter;

3. MVVM (Model-View-ViewModel)

核心思想

  • Model:负责数据的存储和业务逻辑。
  • View:负责展示数据,用户界面。
  • ViewModel:作为中介者,负责数据绑定、命令处理和状态管理。

工作流程

  1. 用户与视图进行交互。
  2. ViewModel监听视图的变化,并更新模型。
  3. ViewModel监听模型的变化,并更新视图。

优点

  • 数据绑定机制简化了视图和模型之间的同步。
  • 降低了视图和模型之间的耦合。

缺点

  • 可能会增加内存开销。
  • 复杂的数据绑定逻辑可能难以调试。

示例: 继续上面的待办事项应用,但这次使用MVVM模式(使用Vue.js实现):

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Vue MVVM示例</title><script src="https://cdn.jsdelivr.net/npm/vue@next"></script>
</head>
<body><div id="app"><h1>待办事项</h1><input v-model="newTodo" placeholder="添加新事项" @keyup.enter="addTodo"><ul><li v-for="(todo, index) in todos" :key="index">{{ todo }}<button @click="removeTodo(index)">删除</button></li></ul></div><script>const app = Vue.createApp({data() {return {newTodo: '',todos: []};},methods: {addTodo() {if (this.newTodo.trim()) {this.todos.push(this.newTodo);this.newTodo = '';}},removeTodo(index) {this.todos.splice(index, 1);}}});app.mount('#app');</script>
</body>
</html>

总结

  • MVC:适用于传统的Web应用,尤其是服务器端渲染的应用。它通过控制器协调模型和视图之间的交互。
  • MVP:适用于需要高度测试性的应用,特别是移动应用。它通过Presenter来分离视图和业务逻辑。
  • MVVM:适用于现代的单页应用(SPA),特别是使用JavaScript框架如Vue、Angular或React构建的应用。它通过数据绑定简化了视图和模型之间的同步。

选择哪种模式取决于具体的应用场景和需求。MVVM由于其简洁的数据绑定机制,在现代前端开发中非常流行。

相关文章:

MVVM、MVC、MVP 的区别

MVVM&#xff08;Model-View-ViewModel&#xff09;、MVC&#xff08;Model-View-Controller&#xff09;和MVP&#xff08;Model-View-Presenter&#xff09;是三种常见的软件架构模式&#xff0c;它们在客户端应用开发中被广泛使用。每种模式都有其特定的设计理念和应用场景&…...

【Verilog】期末复习

数字逻辑电路分为哪两类&#xff1f;它们各自的特点是什么&#xff1f; 组合逻辑电路&#xff1a;任意时刻的输出仅仅取决于该时刻的输入&#xff0c;而与电路原来的状态无关 没有记忆功能&#xff0c;只有从输入到输出的通路&#xff0c;没有从输出到输入的回路 时序逻辑电路&…...

C#都可以找哪些工作?

在国内学习C#&#xff0c;可以找的工作主要是以下4个&#xff1a; 1、游戏开发 需要学习C#编程、Unity引擎操作、游戏设计和3D图形处理等。 2、PC桌面应用开发 需要学习C#编程、WinForm框架/WPF框架、MVVM设计模式和UI/UX设计等。 3、Web开发 需要学习C#编程、ASP.NET框架…...

机器学习Python使用scikit-learn工具包详细介绍

一、简介 Scikit-learn是一个开源的机器学习库&#xff0c;用于Python编程语言。它建立在NumPy、SciPy和matplotlib这些科学计算库之上&#xff0c;提供了简单有效的数据挖掘和数据分析工具。Scikit-learn库包含了许多用于分类、回归、聚类和降维的算法&#xff0c;包括支持向量…...

蓝桥杯真题 - 扫雷 - 题解

题目链接&#xff1a;https://www.lanqiao.cn/problems/549/learning/ 个人评价&#xff1a;难度 1 星&#xff08;满星&#xff1a;5&#xff09; 前置知识&#xff1a;无 整体思路 按题意模拟&#xff1b;为了减少不必要的“数组越界”判断&#xff0c;让数组下标从 1 1 1…...

vue3项目结合Echarts实现甘特图(可拖拽、选中等操作)

效果图&#xff1a; 图一&#xff1a;选中操作 图二&#xff1a;上下左右拖拽操作 本案例在echarts​​​​​​​示例机场航班甘特图的基础上修改​​​​​​​ 封装ganttEcharts组件&#xff0c;测试数据 airport-schedule.jsonganttEcharts代码: 直接复制粘贴可测​​​​…...

Log4j2 插件的简单使用

代码: TestPlugin.java package com.chenjiacheng.webapp.plugins;import org.apache.logging.log4j.core.LogEvent; import org.apache.logging.log4j.core.config.plugins.Plugin; import org.apache.logging.log4j.core.lookup.StrLookup;/*** Created by chenjiacheng on …...

Linux之RPM和YUM命令

一、RPM命令 1、介绍 RPM(RedHat Package Manager).,RedHat软件包管理工具&#xff0c;类似windows里面的setup,exe是Liux这系列操作系统里而的打包安装工具。 RPMI包的名称格式&#xff1a; Apache-1.3.23-11.i386.rpm “apache’” 软件名称“1.3.23-11” 软件的版本号&am…...

读取硬件板子上的数据

SSCOM工具&#xff0c;先要安装一个插件 这样就可以读到设备数据...

Cesium 实例化潜入潜出

Cesium 实例化潜入潜出 1、WebGL Instance 的原理 狭义的的WebGL 中说使用 Instance, 一般指使用 glDrawArraysInstanced 用于实例化渲染的函数。它允许在一次绘制调用中渲染多个相同的几何体实例&#xff0c;而无需为每个实例发起单独的绘制调用。 Three.js 就是使用这种方…...

java引入jedis并且关于开放redis端口问题

博主主页: 码农派大星. 数据结构专栏:Java数据结构 数据库专栏:数据库 JavaEE专栏:JavaEE 软件测试专栏:软件测试 关注博主带你了解更多知识 目录 1. 引入jedis ​编辑 2. 关于java客户端开放redis端口问题 3. 连接redis服务器 redis服务器在官网公开了使用的协议: resp…...

【人工智能】用Python实现情感分析:从简单词典到深度学习方法的演进

《Python OpenCV从菜鸟到高手》带你进入图像处理与计算机视觉的大门! 解锁Python编程的无限可能:《奇妙的Python》带你漫游代码世界 情感分析是自然语言处理(NLP)中的一个重要任务,其目的是通过分析文本内容,识别出其中的情感极性,如正面、负面或中性。随着技术的不断…...

关系型数据库的完整性和一致性

完整性 1.实体完整性 - 每一个实体都是独一无二的&#xff0c;没有冗余 --主键/唯一索引 2.参照完整性 - 外键 3.域完整性 - 存储的数据都是有效的数据 --数据类型/数据长度/非空约束/检查约束/ 检查约束&#xff1a; alter table tb_score add constraint ck_score_scmar…...

设计模式の命令访问者迭代器模式

文章目录 前言一、命令模式二、访问者模式三、迭代器模式 前言 本篇是关于设计模式中命令模式、访问者模式、以及迭代器模式的学习笔记。 一、命令模式 命令模式是一种行为型设计模式&#xff0c;其核心目的在于将命令的发送者和接受者解耦&#xff0c;提供一个中间层对命令进行…...

信息系统项目管理 -冲突管理

信息系统项目管理题 冲突管理&#xff1a; 项目管理信息系统包括&#xff08;&#xff09;软件&#xff0c;用于监督资源的使用情况&#xff0c;协助确保合适的资源适时、适地的用于合适活动。 A资源管理或进度计划 BCRM系统 C采购系统或智能分析 DBOM系统 答案&#xff1a;A …...

Gmsh有限元网格剖分(Python)---点、直线、平面的移动

Gmsh有限元网格剖分(Python)—点、直线、平面的移动和旋转 最近在学习有限元的网格剖分算法&#xff0c;主要还是要参考老外的开源Gmsh库进行&#xff0c;写一些博客记录下学习过程&#xff0c;方便以后回忆嘞。 Gmsh的官方英文文档可以参考&#xff1a;gmsh.pdf 但咋就说&a…...

山景BP1048增加AT指令,实现单片机串口控制播放音乐(一)

1、设计目的 山景提供的SDK是蓝牙音箱demo&#xff0c;用户使用ADC按键或者IR遥控器&#xff0c;进行人机交互。然而现实很多场景&#xff0c;需要和单片机通信&#xff0c;不管是ADC按键或者IR接口都不适合和单片机通信。这里设计个AT指令用来和BP1048通信。AT指令如下图所示…...

SMMU软件指南SMMU编程之全局错误和最小配置

安全之安全(security)博客目录导读 目录 一、全局错误 二、最小配置 一、全局错误 与编程接口相关的全局错误会报告到适当的 SMMU_(*_)GERROR 寄存器&#xff0c;而不是通过基于内存的事件队列。这些错误通常是严重的&#xff0c;例如导致 SMMU 停止向前推进。例如&#xf…...

CPU条件下Pytorch、jupyter环境配置

一、创建虚拟环境 查看虚拟环境 conda env list 创建python虚拟环境 conda create -n minist python3.11 激活虚拟环境 conda activate minist 查看虚拟环境下有哪些包 pip list 二、安装pytorch 切换清华源 conda config --add channels https://mirrors.tuna.tsing…...

【自用】通信内网部署rzgxxt项目_01,后端pipeDemo部署(使用nssm.exe仿照nohup)

做完这些工作之后&#xff0c;不要忘记打开 Windows Server 的防火墙端口&#xff0c;8181、8081、8080、22、443、1521 做完这些工作之后&#xff0c;不要忘记打开 Windows Server 的防火墙端口&#xff0c;8181、8081、8080、22、443、1521 做完这些工作之后&#xff0c;不要…...

C++实现分布式网络通信框架RPC(3)--rpc调用端

目录 一、前言 二、UserServiceRpc_Stub 三、 CallMethod方法的重写 头文件 实现 四、rpc调用端的调用 实现 五、 google::protobuf::RpcController *controller 头文件 实现 六、总结 一、前言 在前边的文章中&#xff0c;我们已经大致实现了rpc服务端的各项功能代…...

golang循环变量捕获问题​​

在 Go 语言中&#xff0c;当在循环中启动协程&#xff08;goroutine&#xff09;时&#xff0c;如果在协程闭包中直接引用循环变量&#xff0c;可能会遇到一个常见的陷阱 - ​​循环变量捕获问题​​。让我详细解释一下&#xff1a; 问题背景 看这个代码片段&#xff1a; fo…...

【HarmonyOS 5.0】DevEco Testing:鸿蒙应用质量保障的终极武器

——全方位测试解决方案与代码实战 一、工具定位与核心能力 DevEco Testing是HarmonyOS官方推出的​​一体化测试平台​​&#xff0c;覆盖应用全生命周期测试需求&#xff0c;主要提供五大核心能力&#xff1a; ​​测试类型​​​​检测目标​​​​关键指标​​功能体验基…...

可靠性+灵活性:电力载波技术在楼宇自控中的核心价值

可靠性灵活性&#xff1a;电力载波技术在楼宇自控中的核心价值 在智能楼宇的自动化控制中&#xff0c;电力载波技术&#xff08;PLC&#xff09;凭借其独特的优势&#xff0c;正成为构建高效、稳定、灵活系统的核心解决方案。它利用现有电力线路传输数据&#xff0c;无需额外布…...

线程与协程

1. 线程与协程 1.1. “函数调用级别”的切换、上下文切换 1. 函数调用级别的切换 “函数调用级别的切换”是指&#xff1a;像函数调用/返回一样轻量地完成任务切换。 举例说明&#xff1a; 当你在程序中写一个函数调用&#xff1a; funcA() 然后 funcA 执行完后返回&…...

【机器视觉】单目测距——运动结构恢复

ps&#xff1a;图是随便找的&#xff0c;为了凑个封面 前言 在前面对光流法进行进一步改进&#xff0c;希望将2D光流推广至3D场景流时&#xff0c;发现2D转3D过程中存在尺度歧义问题&#xff0c;需要补全摄像头拍摄图像中缺失的深度信息&#xff0c;否则解空间不收敛&#xf…...

在 Nginx Stream 层“改写”MQTT ngx_stream_mqtt_filter_module

1、为什么要修改 CONNECT 报文&#xff1f; 多租户隔离&#xff1a;自动为接入设备追加租户前缀&#xff0c;后端按 ClientID 拆分队列。零代码鉴权&#xff1a;将入站用户名替换为 OAuth Access-Token&#xff0c;后端 Broker 统一校验。灰度发布&#xff1a;根据 IP/地理位写…...

Qt Http Server模块功能及架构

Qt Http Server 是 Qt 6.0 中引入的一个新模块&#xff0c;它提供了一个轻量级的 HTTP 服务器实现&#xff0c;主要用于构建基于 HTTP 的应用程序和服务。 功能介绍&#xff1a; 主要功能 HTTP服务器功能&#xff1a; 支持 HTTP/1.1 协议 简单的请求/响应处理模型 支持 GET…...

vue3 定时器-定义全局方法 vue+ts

1.创建ts文件 路径&#xff1a;src/utils/timer.ts 完整代码&#xff1a; import { onUnmounted } from vuetype TimerCallback (...args: any[]) > voidexport function useGlobalTimer() {const timers: Map<number, NodeJS.Timeout> new Map()// 创建定时器con…...

C# 类和继承(抽象类)

抽象类 抽象类是指设计为被继承的类。抽象类只能被用作其他类的基类。 不能创建抽象类的实例。抽象类使用abstract修饰符声明。 抽象类可以包含抽象成员或普通的非抽象成员。抽象类的成员可以是抽象成员和普通带 实现的成员的任意组合。抽象类自己可以派生自另一个抽象类。例…...