.NET Core MongoDB数据仓储和工作单元模式实操

前言

  上一章节我们主要讲解了MongoDB数据仓储和工作单元模式的封装,这一章节主要讲的是MongoDB用户管理相关操作实操。如:获取所有用户信息、获取用户分页数据、通过用户ID获取对应用户信息、添加用户信息、事务添加用户信息、用户信息修改、用户信息删除等实战教程。

MongoDB从入门到实战的相关教程

MongoDB从入门到实战之MongoDB简介👉

MongoDB从入门到实战之MongoDB快速入门👉

MongoDB从入门到实战之Docker快速安装MongoDB👉

MongoDB从入门到实战之MongoDB工作常用操作命令👉

MongoDB从入门到实战之.NET Core使用MongoDB开发ToDoList系统(1)-后端项目框架搭建👉

MongoDB从入门到实战之.NET Core使用MongoDB开发ToDoList系统(2)-Swagger框架集成👉

MongoDB从入门到实战之.NET Core使用MongoDB开发ToDoList系统(3)-系统数据集合设计👉

MongoDB从入门到实战之.NET Core使用MongoDB开发ToDoList系统(4)-MongoDB数据仓储和工作单元模式封装👉

MongoDB从入门到实战之.NET Core使用MongoDB开发ToDoList系统(5)-MongoDB数据仓储和工作单元模式实操👉

YyFlight.ToDoList项目源码地址

欢迎各位看官老爷review,有帮助的别忘了给我个Star哦💖!!!

GitHub地址:GitHub - YSGStudyHards/YyFlight.ToDoList: 【.NET8 MongoDB 待办清单系统】.NET8 MongoDB从入门到实战基础教程,该项目后端使用的是.NET8、前端页面使用Blazor、使用MongoDB存储数据,更多相关内容大家可以看目录中的MongoDB从入门到实战的相关教程。该系列教程可作为.NET Core入门项目进行学习,感兴趣的小伙伴可以关注博主和我一起学习共同进步。

MongoRepository地址:https://github.com/YSGStudyHards/YyFlight.ToDoList/tree/main/Repository/Repository

MongoDB事务使用前提说明

参阅MongoDB的事务👉

说明:

MongoDB单机服务器不支持事务【使用MongoDB事务会报错:Standalone servers do not support transactions】,只有在集群情况下才支持事务,因为博主接下来都是在单机环境下操作,所以无法来演示Mongo事务操作,但是方法都已经是封装好了的,大家可以自己搭建集群实操。

原因:

MongoDB在使用分布式事务时需要进行多节点之间的协调和通信,而单机环境下无法实现这样的分布式协调和通信机制。但是,在MongoDB部署为一个集群(cluster)后,将多个计算机连接为一个整体,通过协调和通信机制实现了分布式事务的正常使用。从数据一致性和可靠性的角度来看,在分布式系统中实现事务处理是至关重要的。而在单机环境下不支持事务,只有在集群情况下才支持事务的设计方式是为了保证数据一致性和可靠性,并且也符合分布式系统的设计思想。

创建EntityBase公共类

一个公共的具有相同特性和行为的基类。

    public class EntityBase{/// <summary>/// 主键Id/// </summary>[BsonId][BsonRepresentation(BsonType.ObjectId)]public string Id { get; set; }/// <summary>/// 创建时间/// </summary>public DateTime CreateDate { get; set; }/// <summary>/// 更新时间/// </summary>public DateTime UpdateDate { get; set; }}

添加UserInfo用户表实体映射模型

    [Table("yyflight_todolist_user")]public class UserInfo : EntityBase{/// <summary>/// 登录账号/// </summary>public string UserName { get; set; }/// <summary>/// 登录密码/// </summary>public string Password { get; set; }/// <summary>/// 用户昵称/// </summary>public string NickName { get; set; }/// <summary>/// 用户头像/// </summary>public string HeadPortrait { get; set; }/// <summary>/// 用户邮箱/// </summary>public string Email { get; set; }/// <summary>/// 用户状态(0冻结,1正常,2注销)/// </summary>public int Status { get; set; }}

在前面类中,Id属性中的特性的作用:

  • 需要用于将通用语言运行时(CLR)对象映射到MongoDB集合。
  • 用[BsonId]进行注释,使该属性成为文档的主键。
  • 用[BsonRepresentation(BsonType.ObjectId)]进行注释,以允许以字符串类型而不是ObjectId结构传递参数。Mongo处理从字符串到ObjectId的转换。没有此特性序列化时会有如下异常提示:

System.FormatException: An error occurred while deserializing the Id property of class Repository.Domain.User.UserInfo: Cannot deserialize a 'String' from BsonType 'ObjectId'.

知识拓展MongoDB ObjectId类型概述:

每次插入一条数据系统都会自动插入一个_id键,键值不可以重复,它可以是任何类型的,也可以手动的插入,默认情况下它的数据类型是ObjectId,由于MongoDB在设计之初就是用作分布式数据库,所以使用ObjectId可以避免不同数据库中_id的重复(如果使用自增的方式在分布式系统中就会出现重复的_id的值)。
ObjectId使用12字节的存储空间,每个字节可以存储两个十六进制数字,所以一共可以存储24个十六进制数字组成的字符串,在这24个字符串中,前8位表示时间戳,接下来6位是一个机器码,接下来4位表示进程id,最后6位表示计数器。

MongoDB 采用 ObjectId 来表示主键的类型,数据库中每个文档都拥有一个_id 字段表示主键,_id 的生成规则如下:

其中包括4-byte Unix 时间戳,3-byte 机器 ID,2-byte 进程 ID,3-byte 计数器(初始化随机)

601e2b6b  a3203c  c89f   2d31aa↑        ↑       ↑       ↑时间戳    机器码   进程ID   随机数 

创建用户Repository

创建用户IUserRepository接口

    public interface IUserRepository : IMongoRepository<UserInfo>{}

创建用户UserRepository类

    public class UserRepository : MongoBaseRepository<UserInfo>, IUserRepository{public UserRepository(IMongoContext context) : base(context){}}

创建用户管理业务代码

创建IUserOperationExampleServices接口

    public interface IUserOperationExampleServices{/// <summary>/// 获取所有用户信息/// </summary>/// <returns></returns>Task<IEnumerable<UserInfo>> GetAllUserInfos();/// <summary>/// 用户分页数据获取/// </summary>/// <param name="userInfoByPageListReq">userInfoByPageListReq</param>/// <returns></returns>Task<IEnumerable<UserInfo>> GetUserInfoByPageList(UserInfoByPageListReq userInfoByPageListReq);/// <summary>/// 通过用户ID获取对应用户信息/// </summary>/// <param name="id">id</param>/// <returns></returns>Task<UserInfo> GetUserInfoById(string id);/// <summary>/// 添加用户信息/// </summary>/// <param name="userInfo">userInfo</param>/// <returns></returns>Task<UserInfo> AddUserInfo(UserInfoReq userInfo);/// <summary>/// 事务添加用户信息/// </summary>/// <param name="userInfo">userInfo</param>/// <returns></returns>Task<UserInfo> AddUserInfoTransactions(UserInfoReq userInfo);/// <summary>/// 用户信息修改/// </summary>/// <param name="id">id</param>/// <param name="userInfo">userInfo</param>/// <returns></returns>Task<UserInfo> UpdateUserInfo(string id, UserInfoReq userInfo);/// <summary>/// 用户信息删除/// </summary>/// <param name="id">id</param>/// <returns></returns>Task<bool> Delete(string id);}

创建UserOperationExampleServices类

    public class UserOperationExampleServices : IUserOperationExampleServices{private readonly IUnitOfWork _unitOfWork;private readonly IUserRepository _userRepository;/// <summary>/// 依赖注入/// </summary>/// <param name="unitOfWork">unitOfWork</param>/// <param name="userRepository">userRepository</param>public UserOperationExampleServices(IUnitOfWork unitOfWork, IUserRepository userRepository){_unitOfWork = unitOfWork;_userRepository = userRepository;}/// <summary>/// 获取所有用户信息/// </summary>/// <returns></returns>public async Task<IEnumerable<UserInfo>> GetAllUserInfos(){var getAllUserInfos = await _userRepository.GetAllAsync();return getAllUserInfos;}/// <summary>/// 用户分页数据获取/// </summary>/// <param name="userInfoByPageListReq">userInfoByPageListReq</param>/// <returns></returns>public async Task<IEnumerable<UserInfo>> GetUserInfoByPageList(UserInfoByPageListReq request){//创建查询条件构造器FilterDefinitionBuilder<UserInfo> buildFilter = Builders<UserInfo>.Filter;FilterDefinition<UserInfo> filter = buildFilter.Empty;SortDefinition<UserInfo> sort = Builders<UserInfo>.Sort.Ascending(m => m.CreateDate);if (!string.IsNullOrEmpty(request.NickName)){filter = buildFilter.Eq(m => m.NickName, request.NickName);}if (!string.IsNullOrEmpty(request.Id)){filter = buildFilter.Eq(m => m.Id, request.Id);}var list = await _userRepository.FindListByPageAsync(filter, request.PageIndex, request.PageSize, Array.Empty<string>(), sort);return list;}/// <summary>/// 通过用户ID获取对应用户信息/// </summary>/// <param name="id">id</param>/// <returns></returns>public async Task<UserInfo> GetUserInfoById(string id){var getUserInfo = await _userRepository.GetByIdAsync(id);return getUserInfo;}/// <summary>/// 添加用户信息/// </summary>/// <param name="userInfo">userInfo</param>/// <returns></returns>public async Task<UserInfo> AddUserInfo(UserInfoReq userInfo){var addUserInfo = new UserInfo(){Id = ObjectId.GenerateNewId().ToString(),UserName = userInfo.UserName,Email = userInfo.Email,NickName = userInfo.NickName,Password = MD5Helper.MDString(userInfo.Password),Status = 1,HeadPortrait = userInfo.HeadPortrait,CreateDate = DateTime.Now,UpdateDate = DateTime.Now,};await _userRepository.AddAsync(addUserInfo);var queryUserInfo = await _userRepository.GetByIdAsync(addUserInfo.Id);return queryUserInfo;}/// <summary>/// 事务添加用户信息/// </summary>/// <param name="userInfo">userInfo</param>/// <returns></returns>public async Task<UserInfo> AddUserInfoTransactions(UserInfoReq userInfo){using var session = await _unitOfWork.InitTransaction();var addUserInfo = new UserInfo(){Id = ObjectId.GenerateNewId().ToString(),UserName = userInfo.UserName,Email = userInfo.Email,NickName = userInfo.NickName,Password = MD5Helper.MDString(userInfo.Password),Status = 1,HeadPortrait = userInfo.HeadPortrait,CreateDate = DateTime.Now,UpdateDate = DateTime.Now,};await _userRepository.AddTransactionsAsync(session, addUserInfo);//查不到任何信息var queryUserInfo = await _userRepository.GetByIdAsync(addUserInfo.Id);//提交新增用户信息操作await _unitOfWork.Commit(session);//UserInfo只有在提交后才会被添加queryUserInfo = await _userRepository.GetByIdAsync(addUserInfo.Id);return queryUserInfo;}/// <summary>/// 用户信息修改/// </summary>/// <param name="id">id</param>/// <param name="userInfo">userInfo</param>/// <returns></returns>public async Task<UserInfo> UpdateUserInfo(string id, UserInfoReq userInfo){#region 指定字段和条件修改//修改条件var list = new List<FilterDefinition<UserInfo>>{Builders<UserInfo>.Filter.Eq("_id", new ObjectId(id))};var filter = Builders<UserInfo>.Filter.And(list);//指定要修改的字段内容//参考文章:https://chsakell.gitbook.io/mongodb-csharp-docs/crud-basics/update-documentsvar updateDefinition = Builders<UserInfo>.Update.Set(u => u.HeadPortrait, userInfo.HeadPortrait).Set(u => u.NickName, userInfo.NickName).Set(u => u.Status, userInfo.Status);await _userRepository.UpdateAsync(filter, updateDefinition);#endregion#region 指定对象异步修改一条数据//var updateUserInfo = new UserInfo//{//    UserName = userInfo.UserName,//    Password = MD5Helper.MDString(userInfo.Password),//    Status = 1,//    HeadPortrait = userInfo.HeadPortrait,//    Email = userInfo.Email,//    NickName = userInfo.NickName,//    UpdateDate = DateTime.Now,//};//await _userRepository.UpdateAsync(updateUserInfo, id);#endregion#region 数据批量修改示例1.批量修改的条件(把创建时间CreateDate为近五日的用户状态更改为0)//var time = DateTime.Now;//var list = new List<FilterDefinition<UserInfo>>();//list.Add(Builders<UserInfo>.Filter.Gt("CreateDate", time));//大于当前时间//list.Add(Builders<UserInfo>.Filter.Lt("CreateDate", time.AddDays(5)));//小于当前时间+5day//var filter = Builders<UserInfo>.Filter.And(list);2.要修改的字段内容//var dic = new Dictionary<string, string>//{//    { "Status", "0" }//};3.批量修改//await _userRepository.UpdateManayAsync(dic, filter);#endregionreturn await _userRepository.GetByIdAsync(id);}/// <summary>/// 用户信息删除/// </summary>/// <param name="id"></param>/// <returns></returns>public async Task<bool> Delete(string id){await _userRepository.DeleteAsync(id);var testUserInfo = await _userRepository.GetByIdAsync(id);return testUserInfo == null;}}

UserOperationExample控制创建

    /// <summary>/// MongoDB用户管理操作示例/// </summary>[ApiController][Produces("application/json")][Route("api/[controller]/[action]")]public class UserOperationExampleController : ControllerBase{private readonly IUserOperationExampleServices _userOperationExampleServices;/// <summary>/// 依赖注入/// </summary>/// <param name="userOperationExampleServices">userOperationExampleServices</param>public UserOperationExampleController(IUserOperationExampleServices userOperationExampleServices){_userOperationExampleServices = userOperationExampleServices;}/// <summary>/// 获取所有用户信息/// </summary>/// <returns></returns>[HttpGet]public async Task<ActionResult<IEnumerable<UserInfo>>> GetAllUserInfos(){var userInfos = await _userOperationExampleServices.GetAllUserInfos();return Ok(userInfos);}/// <summary>/// 获取用户分页数据/// </summary>/// <param name="userInfoByPageListReq">userInfoByPageListReq</param>/// <returns></returns>[HttpPost]public async Task<ActionResult<IEnumerable<UserInfo>>> GetUserInfoByPageList([FromBody] UserInfoByPageListReq userInfoByPageListReq){var getUserInfoByPageList = await _userOperationExampleServices.GetUserInfoByPageList(userInfoByPageListReq);return Ok(getUserInfoByPageList);}/// <summary>/// 通过用户ID获取对应用户信息/// </summary>/// <param name="id">id</param>/// <returns></returns>[HttpGet("{id}")]public async Task<ActionResult<UserInfo>> GetUserInfoById(string id){var userInfo = await _userOperationExampleServices.GetUserInfoById(id);return Ok(userInfo);}/// <summary>/// 添加用户信息/// </summary>/// <param name="userInfo">userInfo</param>/// <returns></returns>[HttpPost]public async Task<ActionResult<UserInfo>> AddUserInfo([FromBody] UserInfoReq userInfo){var addUserInfo = await _userOperationExampleServices.AddUserInfo(userInfo);return Ok(addUserInfo);}/// <summary>/// 事务添加用户信息/// </summary>/// <param name="userInfo">userInfo</param>/// <returns></returns>[HttpPost]public async Task<ActionResult<UserInfo>> AddUserInfoTransactions([FromBody] UserInfoReq userInfo){//TODO:单机服务器不支持事务使用【使用MongoDB事务会报错:Standalone servers do not support transactions】,只有在集群情况下才能用var addUserInfo = await _userOperationExampleServices.AddUserInfoTransactions(userInfo);return Ok(addUserInfo);}/// <summary>/// 用户信息修改/// </summary>/// <param name="id">id</param>/// <param name="userInfo">userInfo</param>/// <returns></returns>[HttpPut("{id}")]public async Task<ActionResult<UserInfo>> UpdateUserInfo(string id, [FromBody] UserInfoReq userInfo){var updateUserInfo = await _userOperationExampleServices.UpdateUserInfo(id, userInfo);return Ok(updateUserInfo);}/// <summary>/// 用户信息删除/// </summary>/// <param name="id">id</param>/// <returns></returns>[HttpDelete("{id}")]public async Task<ActionResult> Delete(string id){var deleteUser = await _userOperationExampleServices.Delete(id);return Ok(deleteUser);}}

注册数据库基础操作和工作单元

//注册数据库基础操作和工作单元
builder.Services.AddSingleton<IMongoConnection, MongoConnection>();
builder.Services.AddScoped<IMongoContext, MongoContext>();
builder.Services.AddScoped<IUnitOfWork, UnitOfWork>();
builder.Services.AddScoped<IUserRepository, UserRepository>();

注册相关应用服务

builder.Services.AddScoped<IUserOperationExampleServices, UserOperationExampleServices>();

Swagger用户管理操作示例展示

添加用户信息

 添加成功,返回添加成功的用户信息:

通过用户ID获取对应用户信息

拿刚才添加成功的用户ID,查询用户信息:

获取所有用户信息

用户分页数据获取

查询第1页,显示10条数据:

查询第1页,显示2条数据:

 

 用户信息修改

指定要修改的字段内容,修改HeadPortrait、NickName、Status
参考文章:Update - MongoDB C# docs

 

 修改成功:

用户信息删除

输入需要删除的用户ID,点击Execute删除:

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Java程序员面试专栏 Java领域】Java并发 核心面试指引

关于Java 并发部分的核心知识进行一网打尽,主要包括Java为什么会有并发问题,并发问题的解决方案,到最后的锁和JUC包的解决方案,通过一篇文章串联面试重点,并且帮助加强日常基础知识的理解,全局思维导图如下所示 基础概念 关于并发与并行,Java的内存模型,以及并发编程…

使用miniconda管理Python环境

之前经常使用pipenv管理虚拟环境&#xff0c;但是有一个问题就是代码给别人使用的时候&#xff0c;别人使用的Python版本和自己的不一致时&#xff0c;安装依赖包的时候会有问题。所以现在使用miniconda来管理虚拟环境&#xff0c;不仅小巧方便&#xff0c;还能为每个环境指定不…

Eclipse Version: 2023-03 (4.27.0) JDK19 Tomcat10.2

Eclipse Version: 2023-03 (4.27.0) JDK19 Tomcat10.2

Django实战:部署项目 【资产管理系统】,Django完整项目学习研究(项目全解析,部署教程,非常详细)

导言 关于Django&#xff0c;我已经和大家分享了一些知识&#xff0c;考虑到一些伙伴需要在实际的项目中去理解。所以我上传了一套Django的项目学习源码&#xff0c;已经和本文章进行了绑定。大家可以自行下载学习&#xff0c;考虑到一些伙伴是初学者&#xff0c;几年前&#…

前端常见的设计模式

说到设计模式&#xff0c;大家想到的就是六大原则&#xff0c;23种模式。这么多模式&#xff0c;并非都要记住&#xff0c;但作为前端开发&#xff0c;对于前端出现率高的设计模式还是有必要了解并掌握的&#xff0c;浅浅掌握9种模式后&#xff0c;整理了这份文章。 六大原则&…

一起玩儿物联网人工智能小车(ESP32)——63 SD和TF卡模块的使用

摘要&#xff1a;本文介绍SD和TF卡模块的使用方法 前面介绍了非易失性存储的使用方法&#xff0c;由于空间和本身只支持键值对的限制&#xff0c;非易失性存储只适用于少量数据的记录。而不适用于各种声音、图片、大量数据等情况的使用。这时候就需要有文件系统或者更大容量存…

精品jsp+ssm鲜花销售管理系统-购物商城

《[含文档PPT源码等]精品jspssm鲜花销售管理系统[包运行成功]》该项目含有源码、文档、PPT、配套开发软件、软件安装教程、项目发布教程、包运行成功&#xff01; 使用技术&#xff1a; 开发语言&#xff1a;Java 框架&#xff1a;ssm 技术&#xff1a;JSP JDK版本&#x…

C语言scanf函数详解..

1.前言 前面说过了printf函数 他是将二进制表示的整数、浮点数、字符、字符串根据转换规范转换成字符或者字符串 并且打印到了控制台上 那么既然有了输出函数 那么肯定也有输入函数咯 的确如此 他就是scanf函数 他是将字符或者字符串根据转换规范转换成二进制表示的整数、浮点…

2024抖店保证金缴纳标准,不懂的快来看!

我是电商珠珠 新的一年到来&#xff0c;意味着新的开始。 很多人想要去兼职或全职一个足不出户的副业&#xff0c;兜兜转转了解到了抖店。 但是对于新手来说&#xff0c;比较关心的还是抖店的保证金缴纳问题&#xff0c;需不需要保证金&#xff0c;需要多少等。 今天&#…

磁体发条概念

使用磁体发条&#xff08;也称为磁弹簧或磁蓄能器&#xff09;作为储能装置是一个有趣的概念&#xff0c;它利用电磁感应原理来存储和释放能量。磁体发条的基本原理是通过旋转一个强磁体&#xff0c;使其通过一个线圈的中心&#xff0c;从而在线圈中产生电流。当磁体停止旋转时…

解决ubuntu登录密码问题

解决ubuntu登录密码问题 不要随便删除密码&#xff0c;不要随便改密码&#xff0c;很容导致密码过期&#xff0c;或者密码无效。参考了很多人的做法&#xff0c;都没有得到解决。下面的过程&#xff0c;够详细了&#xff0c;我就是这么搞好的。 1、重启虚拟机&#xff0c;不停…

嵌入式内核链表list_head,如何管理不同类型节点的实现

在Linux内核中&#xff0c;提供了一个用来创建双向循环链表的结构 list_head。虽然linux内核是用C语言写的&#xff0c;但是list_head的引入&#xff0c;使得内核数据结构也可以拥有面向对象的特性&#xff0c;通过使用操作list_head 的通用接口很容易实现代码的重用&#xff0…