WinForms 中使用 MVVM 模式构建应用:实现登录页面、页面导航及 SQLite 数据库连接完整框架搭建过程

前言

在传统的 WinForms 应用程序开发中,很多开发者使用事件驱动的设计模式,直接将业务逻辑编写在界面代码中。然而,随着应用程序的复杂性增加,单一的界面文件变得臃肿,难以测试和维护。借鉴 WPF 中 MVVM(Model-View-ViewModel)模式的设计思想,可以帮助我们更好地管理业务逻辑和数据绑定。本文将介绍如何在 WinForms 中构建一个 MVVM 框架的登录页面示例,并实现页面导航、SQLite 数据库连接及依赖注入管理。

一、项目设计与依赖引用

1. 新增winform项目

在这里插入图片描述

2. 创建项目结构

项目结构:按模块创建以下文件夹:

  • Models:存放数据实体类。
  • ViewModels:包含视图模型,负责处理业务逻辑和数据绑定。
  • Views:放置WinForms窗体,充当UI界面。
  • Services:用于数据库服务操作。
  • IoC:配置依赖注入容器。
  • Commands:配置执行命令。
    在这里插入图片描述
  1. 安装所需库
    • 使用 Microsoft.Extensions.DependencyInjection 来实现依赖注入。
    • 使用 Dapper 库来连接和操作 SQLite 数据库。
    • 使用SQLitePCLRaw.bundle_e_sqlite3库来处理和连接SQLite数据库。
      在这里插入图片描述

二、创建数据实体 Model

首先创建一个 User 类来表示数据库中的用户信息。我们假设用户表 Users 包含 IdUsernamePassword 三个字段。

namespace WinFormMVVM.Models
{public class User{public int Id { get; set; }public string Username { get; set; }public string Password { get; set; }}
}

三、创建服务层 Service

创建一个初始化数据库的服务类DatabaseInitializer,配置默认用户和密码

using System.Data;
using Dapper;
using SQLitePCL;namespace WinFormMVVM.Services
{public class DatabaseInitializer{private readonly IDbConnection _dbConnection;public DatabaseInitializer(IDbConnection dbConnection){_dbConnection = dbConnection;}public void InitializeDatabase(){Batteries.Init();const string createTableQuery = @"CREATE TABLE IF NOT EXISTS Users (Id INTEGER PRIMARY KEY AUTOINCREMENT,Username TEXT NOT NULL,Password TEXT NOT NULL);";const string insertUserQuery = @"INSERT INTO Users (Username, Password) VALUES (@Username, @Password)";_dbConnection.Open();_dbConnection.Execute(createTableQuery);// 检查是否已有用户数据,若无则添加var existingUser = _dbConnection.QueryFirstOrDefault("SELECT * FROM Users WHERE Username = @Username", new { Username = "admin" });if (existingUser == null){_dbConnection.Execute(insertUserQuery, new { Username = "admin", Password = "password123" });}_dbConnection.Close();}}
}

Services 文件夹中创建 IUserService 接口及其实现 UserService,用于从 SQLite 数据库中查询用户信息。

using System.Data;
using Dapper;
using WinFormMVVM.Models;namespace WinFormMVVM.Services
{public interface IUserService{User GetUserByUsername(string username);}public class UserService : IUserService{private readonly IDbConnection _dbConnection;public UserService(IDbConnection dbConnection){_dbConnection = dbConnection;}public User GetUserByUsername(string username){string sql = "SELECT * FROM Users WHERE Username = @Username";return _dbConnection.QuerySingleOrDefault<User>(sql, new { Username = username });}}
}

四、Commands命令实现类

RelayCommand 是一种常用的命令实现类,通常在 MVVM 模式中用于实现 ICommand 接口,但 WinForms 中并没有自带该类。如果需要使用它,可以自己定义一个简单的 RelayCommand 实现,或从一些 MVVM 库(如 CommunityToolkit.Mvvm)中引入。以下是一个自定义 RelayCommand 类的实现:

using System.Windows.Input;namespace WinFormMVVM.Commands
{public class RelayCommand : ICommand{private readonly Action _execute;private readonly Func<bool> _canExecute;public event EventHandler CanExecuteChanged;public RelayCommand(Action execute, Func<bool> canExecute = null){_execute = execute ?? throw new ArgumentNullException(nameof(execute));_canExecute = canExecute;}public bool CanExecute(object parameter){return _canExecute == null || _canExecute();}public void Execute(object parameter){_execute();}public void RaiseCanExecuteChanged(){CanExecuteChanged?.Invoke(this, EventArgs.Empty);}}
}

五、创建 ViewModel 类

在 MVVM 模式中,ViewModel 负责处理业务逻辑并将数据传递给视图。这里创建 LoginViewModel 类来处理登录逻辑:

using System.ComponentModel;
using System.Windows.Input;
using WinFormMVVM.Models;
using WinFormMVVM.Services;namespace WinFormMVVM.ViewModels
{public class LoginViewModel : INotifyPropertyChanged{private readonly IUserService _userService;public event PropertyChangedEventHandler PropertyChanged;public string Username { get; set; }public string Password { get; set; }public ICommand LoginCommand { get; }public LoginViewModel(IUserService userService){_userService = userService;LoginCommand = new RelayCommand(Login);}private void Login(){var user = _userService.GetUserByUsername(Username);if (user != null && user.Password == Password){MainForm mainForm = new MainForm();mainForm.Show();}else{// 显示登录失败的消息}}}
}

LoginViewModel 通过 _userService 获取用户信息,验证成功后跳转到主页面 MainForm

六、配置 IoC 容器

IoC 文件夹中创建 IoCContainer 静态类,通过依赖注入容器来管理 IDbConnectionIUserService 和其他ViewModel等依赖关系。

using Microsoft.Data.Sqlite;
using Microsoft.Extensions.DependencyInjection;
using System.Data;
using WinFormMVVM.Services;
using WinFormMVVM.ViewModels;namespace WinFormMVVM.IoC
{public static class IoCContainer{public static ServiceProvider Configure(){var services = new ServiceCollection();services.AddSingleton<IDbConnection>(sp =>new SqliteConnection("Data Source=./database.db")); // 设置SQLite数据库路径services.AddSingleton<DatabaseInitializer>();services.AddTransient<IUserService, UserService>();services.AddSingleton<LoginViewModel>();return services.BuildServiceProvider();}}
}

在这里使用了 SQLiteConnection 连接到本地的 SQLite 数据库,连接字符串 Data Source=./database.db 可以根据实际情况修改。

七、创建 View 和绑定 ViewModel

  1. 登录页面(LoginForm):创建一个 LoginForm 窗体,通过构造函数注入 LoginViewModel 实例并绑定到表单。
    在这里插入图片描述
using WinFormMVVM.ViewModels;namespace WinFormMVVM.Views
{public partial class LoginForm : Form{private readonly LoginViewModel _viewModel;public LoginForm(LoginViewModel viewModel){InitializeComponent();_viewModel = viewModel;}private void btnLogin_Click(object sender, EventArgs e){_viewModel.Username = tb_user.Text.Trim();_viewModel.Password = tb_password.Text.Trim();_viewModel.LoginCommand.Execute(null);this.Hide();}}
}

八、设置程序入口并启动依赖注入

Program.cs 文件中配置依赖注入容器,并通过容器注入 LoginViewModel 进入应用的启动界面 LoginForm

using System;
using System.Windows.Forms;
using Microsoft.Extensions.DependencyInjection;
using WinFormMVVM.IoC;
using WinFormMVVM.ViewModels;
using WinFormMVVM.Views;namespace WinFormMVVM
{static class Program{[STAThread]static void Main(){var serviceProvider = IoCContainer.Configure();Application.EnableVisualStyles();Application.SetCompatibleTextRenderingDefault(false);var loginViewModel = serviceProvider.GetService<LoginViewModel>();var loginForm = new LoginForm(loginViewModel);Application.Run(loginForm);}}
}

创建主页面 MainForm.cs

namespace WinFormMVVM.Views
{public partial class MainForm : Form{public MainForm(){InitializeComponent();}}
}

九、执行程序

通过启动程序,可以实现sqlite数据库自动创建,并且初始化默认数据,同时,通过依赖注入实现服务的运行,和页面ViewModel的注册。输入正确的账号密码,即可登录成功。
在这里插入图片描述

十、总结

本文介绍了如何在 WinForms 中应用 MVVM 模式,并通过 SQLite 进行数据持久化处理。通过引入依赖注入容器,服务类与视图模型的依赖关系可以在应用程序运行时被动态配置,实现了良好的解耦。这样设计的应用不仅具备更好的扩展性和可维护性,还更利于测试和重构。

借助上述框架,可以更清晰地组织 WinForms 项目,将应用逻辑、数据操作、UI 展示解耦,提升代码质量。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/6695.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

windows系统编程 - 静态库和动态库

文章目录 前言一、使用obj保护源码生成obj文件导入并使用obj文件方式一 拖入解决方案方式二 附加依赖项适配C语言文件 二、静态库的概述三、静态库的创建与使用四、动态库的概述五、动态库的创建六、动态库的两种调用方式七、动态链接库的隐式加载__declspec(dllimport) 声明外…

数据结构 ——— 查找链式二叉树中值为X的节点

目录 链式二叉树示意图 手搓一个链式二叉树 查找链式二叉树中值为X的节点 链式二叉树示意图 手搓一个链式二叉树 代码演示&#xff1a; // 数据类型 typedef int BTDataType;// 二叉树节点的结构 typedef struct BinaryTreeNode {BTDataType data; //每个节点的数据struc…

基于SSM的BBS社区论坛系统源码

1.项目介绍 系统角色&#xff1a;管理员、业主&#xff08;普通用户&#xff09;功能模块&#xff1a;管理员&#xff08;用户管理、二手置换管理、报修管理、缴费管理、公告管理&#xff09;、普通用户&#xff08;登录注册、二手置换、生活缴费、信息采集、报事报修&#xf…

python的安装环境Miniconda(Conda 命令管理依赖配置)

这一段时间&#xff0c;对AI大模型 有了兴趣就想研究一下。 在研究之前肯定要先把需要的编程技能掌握了。经过我查阅资料&#xff0c;今天就先学一下 python的 环境安装。 Node.js 包管理工具&#xff1a;npm 依赖配置文件&#xff1a;package.json 环境管理&#xff1a;nvm&am…

出租房管理系统有哪些?

出租房管理系统在现代房产租赁市场中发挥着至关重要的作用&#xff0c;其供应商众多&#xff0c;各具特色。以下是对易收租、寓小二、全房通、水滴管家以及悟空租房管理系统等供应商的详细介绍。 一、深圳合众致达科技有限公司的易收租 深圳合众致达科技有限公司是一家专注于…

【在Linux世界中追寻伟大的One Piece】Socket编程TCP

目录 1 -> TCP socket API 2 -> V1 -Echo Server 2.1 -> 测试多个连接的情况 1 -> TCP socket API socket()&#xff1a; socket()打开一个网络通讯端口&#xff0c;如果成功的话&#xff0c;就像open()一样返回一个文件描述符。应用程序可以像读写文件一样用r…

【spring】IOC与DI

&#x1f490;个人主页&#xff1a;初晴~ &#x1f4da;相关专栏&#xff1a;程序猿的春天 一、IOC&#xff08;Inversion of Control&#xff09; 1、概念 IOC&#xff08;Inversion of Control&#xff0c;控制反转&#xff09;是一种设计原则&#xff0c;它将对象的控制权…

【英特尔IA-32架构软件开发者开发手册第3卷:系统编程指南】2001年版翻译,2-16

文件下载与邀请翻译者 学习英特尔开发手册&#xff0c;最好手里这个手册文件。原版是PDF文件。点击下方链接了解下载方法。 讲解下载英特尔开发手册的文章 翻译英特尔开发手册&#xff0c;会是一件耗时费力的工作。如果有愿意和我一起来做这件事的&#xff0c;那么&#xff…

​​​​​​​PHP类型比较

在php中符号分为两种&#xff0c;一种是&#xff0c;还是一种是 松散比较&#xff1a;使用两个等号 比较&#xff0c;只比较值&#xff0c;不比较类型。 严格比较&#xff1a;用三个等号 比较&#xff0c;除了比较值&#xff0c;也比较类型。 注意&#xff0c;当一个号时&…

Mysql、Dm8达梦数据库通过脚本导出指定库所有表的结构详情信息到

目录 前言二、Mysql三、达梦8 前言 在当今复杂多变的数据环境中&#xff0c;数据库作为信息存储与管理的核心&#xff0c;其重要性不言而喻。随着业务的不断拓展和深化&#xff0c;对于数据库表结构的理解与管理成为了确保数据一致性和准确性的关键。特别是在跨数据库系统的场…

yelp数据集上识别潜在的热门商家

yelp数据集是研究B2C业态的一个很好的数据集&#xff0c;要识别潜在的热门商家是一个多维度的分析过程&#xff0c;涉及用户行为、商家特征和社区结构等多个因素。从yelp数据集里我们可以挖掘到下面信息有助于识别热门商家 用户评分和评论分析 评分均值: 商家的平均评分是反映其…

文献阅读记录6-Toward computer-made artificial antibiotics

摘要 将合成生物学和计算生物学的概念结合起来&#xff0c;可能会产生比现有药物更不容易产生耐药性的抗生素&#xff0c;而且还能对抗耐药感染。事实上&#xff0c;计算机引导策略与大规模并行高通量实验方法相结合&#xff0c;代表了抗生素发现的新范式。耐多药微生物引起的…

【docker compose】docker compose的hello world

安装docker desktop后在终端使用以下命令&#xff0c;代表安装成功&#xff0c;并查看当前安装的版本 docker-compose --version示例docker-compose.yml文件 version: 3.8 # 指定 Docker Compose 文件的版本services:scau_jwc: # 定义一个名为 scau_jwc 的服务image: scau_…

PyTorch核心概念:从梯度、计算图到连续性的全面解析(二)

文章目录 pytorch中的Autograd计算图叶子张量 inplace操作PyTorch的两大特点动态图eager execution PyTorch中的Variable参考文献 pytorch中的Autograd pytorch提供了自动求导机制和对GPU的支持 了解自动求导背后的原理和规则&#xff1a;当使用pytorch中没有的loss function时…

dayseven-因果分析-图模型与结构因果模型

在数学上&#xff0c;​“图”(graph)是顶点&#xff08;vertex&#xff0c;也可以称为节点&#xff09;和边(edge)的集合&#xff0c;表示为图G(V,E)&#xff0c;其中V是节点的集合&#xff0c;E是边的集合&#xff0c;图中的节点之间通过边相连&#xff08;也可以不相连&…

今天强的可怕,AI文风写作再也不用写指令了

AI写作最有用的事情之一就是捕捉特定的写作风格&#xff0c;市面上写作工具模仿文风需要下达复杂的prompt&#xff0c;经过一大段精细的微调才能实现&#xff01; 而现在文思助手只要一个按钮就能输出一篇文风相似的文章&#xff01;超级简单&#xff0c;你再也不用为一大段一大…

Vue2中使用firefox的pdfjs进行文件文件流预览

文章目录 1.使用场景2. 使用方式1. npm 包下载,[点击查看](https://www.npmjs.com/package/pdfjs-dist)2. 官网下载1. 放到public文件夹下面2. 官网下载地址[点我,进入官网](https://github.com/mozilla/pdf.js/tags?afterv3.3.122) 3. 代码演示4. 图片预览5. 如果遇到跨域或者…

哪些因素会影响 DC/DC 转换电路快速测试的性能?-纳米软件

DC/DC 转换电路在现代电子设备中起着至关重要的作用&#xff0c;其性能的快速准确测试对于确保电子系统的可靠性和稳定性至关重要。然而&#xff0c;有许多因素会影响 DC/DC 转换电路快速测试的性能。 电路复杂性和参数多样性 单片 DC/DC 转换器由于功能模块和参数复杂性&…

解线性方程组(二)

实验类型&#xff1a;●验证性实验 ○综合性实验 ○设计性实验 实验目的&#xff1a;进一步熟练掌握用Jacobi迭代法和Gauss-Seidel法解线性方程组的算法&#xff0c;提高编程能力和解算线性方程组问题的实践技能。 实验内容&#xff1a; 1)取初值性x(0)(0,0,0,0)T, 精度要求ε…

跨境电商营销:Pinterest的5个便捷营销工具

Pinterest是消费者寻找创意灵感的首选平台之一&#xff0c;同时&#xff0c;根据Global Web Index的调查数据&#xff0c;人们使用Pinterest的首要原因是寻找新产品和品牌&#xff0c;这意味着用户在使用Pinterest时已经有消费意愿和倾向。 因此&#xff0c;让更多目标受众注意…