Python设计模式:MVC模式

1. 什么是 MVC 模式?

MVC 模式是一种软件架构模式,用于将应用程序分为三个主要部分:模型(Model)、视图(View)和控制器(Controller)。这种分离使得应用程序的结构更加清晰,便于管理和扩展。

MVC 模式的组成部分:

  • 模型(Model)

    • 模型是应用程序的核心部分,负责管理数据和业务逻辑。它与数据库或其他数据源进行交互,处理数据的存储、检索和更新。
  • 视图(View)

    • 视图是用户界面的部分,负责显示数据和接收用户输入。它将模型中的数据呈现给用户,并在用户进行操作时更新显示。
  • 控制器(Controller)

    • 控制器是模型和视图之间的中介,负责处理用户输入并更新模型和视图。它接收用户的操作(如按钮点击),调用相应的模型方法,并更新视图。

MVC 模式的优点:

  • 分离关注点:MVC 模式将应用程序的不同部分分离,使得每个部分可以独立开发和维护。这种分离使得代码更加清晰,易于理解。
  • 可扩展性:由于模型、视图和控制器之间的解耦,开发者可以轻松地添加新功能或修改现有功能,而不影响其他部分。
  • 可测试性:MVC 模式使得单元测试变得更加容易,因为模型和控制器可以独立于视图进行测试。

2. 示例1:运动会计时器

实现一个运动会计时器应用,用户可以通过三个按钮来控制计时:开始计时、记录当前耗时和停止计时。每次点击"记录"按钮时,当前的耗时将被记录并显示在列表中。点击"停止"按钮将结束计时并显示最终耗时。

python 复制代码
import tkinter as tk
from tkinter import messagebox
import time


# 模型
class TimerModel:
    def __init__(self):
        self.start_time = None
        self.elapsed_time = 0
        self.records = []

    def start_timer(self):
        self.start_time = time.time()
        self.elapsed_time = 0  # 重置计时器为 0
        self.records.clear()  # 清除之前的记录

    def record_time(self):
        if self.start_time is not None:
            current_time = time.time()
            elapsed = current_time - self.start_time + self.elapsed_time
            self.records.append(elapsed)
            return elapsed
        return None

    def stop_timer(self):
        if self.start_time is not None:
            self.elapsed_time = time.time() - self.start_time + self.elapsed_time
            self.start_time = None
            return self.elapsed_time
        return None

    @staticmethod
    def format_time(seconds):
        milliseconds = int((seconds - int(seconds)) * 1000)
        seconds = int(seconds)
        minutes = seconds // 60
        seconds = seconds % 60
        return f"{minutes}m {seconds}s {milliseconds}ms"


# 视图
class TimerView:
    def __init__(self, master):
        self.master = master
        self.master.title("Sports Timer")
        self.master.geometry("400x400")

        self.record_listbox = tk.Listbox(master, font=("Arial", 14))
        self.record_listbox.pack(pady=20, fill=tk.BOTH, expand=True)

        self.start_button = tk.Button(master, text="Start", command=self.start)
        self.start_button.pack(pady=5)

        self.record_button = tk.Button(master, text="Record", command=self.record)
        self.record_button.pack(pady=5)

        self.stop_button = tk.Button(master, text="Stop", command=self.stop)
        self.stop_button.pack(pady=5)

        self.message_label = tk.Label(master, text="", font=("Arial", 12))
        self.message_label.pack(pady=10)

    def start(self):
        return True

    def record(self, elapsed_time):
        if elapsed_time is not None:
            formatted_time = TimerModel.format_time(elapsed_time)
            self.record_listbox.insert(tk.END, formatted_time)  # 显示记录的时间
            self.record_listbox.see(tk.END)  # 自动滚动到最新记录

    def stop(self, total_time):
        if total_time is not None:
            formatted_time = TimerModel.format_time(total_time)
            self.record_listbox.insert(tk.END, f"Total Time: {formatted_time}")  # 显示总时间
            self.record_listbox.see(tk.END)  # 自动滚动到最新记录

    def display_start_message(self):
        self.message_label.config(text="Timer started! Click 'Record' to log the time.")

    def clear_start_message(self):
        self.message_label.config(text="")  # 清除提示信息


# 控制器
class TimerController:
    def __init__(self, model, view):
        self.model = model
        self.view = view

        # 绑定按钮事件
        self.view.start_button.config(command=self.start_timer)
        self.view.record_button.config(command=self.record_time)
        self.view.stop_button.config(command=self.stop_timer)

    def start_timer(self):
        self.model.start_timer()
        self.view.record_listbox.delete(0, tk.END)  # 清除之前的记录
        self.view.display_start_message()  # 显示开始计时的提示

    def record_time(self):
        elapsed_time = self.model.record_time()
        self.view.record(elapsed_time)

    def stop_timer(self):
        total_time = self.model.stop_timer()
        self.view.stop(total_time)
        self.view.clear_start_message()  # 清除开始计时的提示


if __name__ == "__main__":
    root = tk.Tk()  # 创建主窗口
    model = TimerModel()  # 创建模型
    view = TimerView(root)  # 创建视图
    controller = TimerController(model, view)  # 创建控制器
    root.mainloop()  # 运行主循环
  1. 模型(Model)

    • TimerModel 类负责管理计时数据。它包含开始时间、经过时间和记录列表。
    • start_timer 方法开始计时,重置计时器为 0,并清除之前的记录。
    • record_time 方法计算当前经过的时间并将其添加到记录列表中。
    • stop_timer 方法计算总的经过时间并返回。
    • format_time 方法将时间格式化为"分钟:秒:毫秒"的格式。
  2. 视图(View)

    • TimerView 类负责用户界面的显示和用户输入的处理。它创建了一个列表框用于显示记录的时间和三个按钮(开始、记录、停止)。
    • start 方法用于开始计时。
    • record 方法接收经过的时间并将其格式化后显示在列表框中,同时调用 self.record_listbox.see(tk.END) 来自动滚动到最新记录。
    • stop 方法接收总的经过时间并将其显示在列表框中,同样调用 self.record_listbox.see(tk.END) 来确保最新记录可见。
  3. 控制器(Controller)

    • TimerController 类负责协调模型和视图之间的交互。它处理用户输入并更新模型和视图。
    • start_timer 方法调用模型的开始计时方法,并清除列表框中的记录,同时显示开始计时的提示。
    • record_time 方法调用模型的记录时间方法,并将结果传递给视图。
    • stop_timer 方法调用模型的停止计时方法,并将结果传递给视图,同时清除开始计时的提示。
  4. 客户端代码

    • if __name__ == "__main__": 块中,创建模型、视图和控制器的实例,并运行主循环。

运行上述代码时,将看到一个运动会计时器界面,包含一个显示记录时间的列表框和三个按钮(开始、记录、停止)。用户可以点击"开始"按钮开始计时,点击"记录"按钮记录当前的耗时(保留到毫秒),点击"停止"按钮结束计时并显示总耗时。同时,在点击"开始"时,文本面板会显示"计时器已启动!点击'记录'以记录时间。"在点击"停止"时,提示信息将被清除,并且列表框会自动滚动以显示最新记录。

相关推荐
xiaoh_722 分钟前
解决视频处理中的 HEVC 解码错误:Could not find ref with POC xxx【已解决】
python·ffmpeg·音视频
明月与玄武1 小时前
Python编程的真谛:超越语法,理解编程本质
python·编程语言
CodeCraft Studio1 小时前
Excel处理控件Aspose.Cells教程:使用 Python 在 Excel 中进行数据验
开发语言·python·excel
拾忆-eleven1 小时前
C语言实战:用Pygame打造高难度水果消消乐游戏
c语言·python·pygame
旦莫2 小时前
Python 教程:我们可以给 Python 文件起中文名吗?
开发语言·python
豌豆花下猫2 小时前
Python 潮流周刊#99:如何在生产环境中运行 Python?(摘要)
后端·python·ai
小杨4042 小时前
python入门系列二十(peewee)
人工智能·python·pycharm
弧襪2 小时前
FlaskRestfulAPI接口的初步认识
python·flaskrestfulapi
船长@Quant2 小时前
文档构建:Sphinx全面使用指南 — 进阶篇
python·markdown·sphinx·文档构建
cloudy4912 小时前
强化学习:历史基金净产值,学习最大化长期收益
python·强化学习