ABP天坑--UOW自动保存修改

2024-02-01 07:48
文章标签 保存 自动 修改 abp 天坑 uow

本文主要是介绍ABP天坑--UOW自动保存修改,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

ABP项目也做了挺久了,实际上也没碰上太多的问题,但这两天被ABP的自动保存修改(Automatically Saving Changes)这个天坑给恶心坏了

因为项目设计上的问题,原先为了方便加上环境限制,没有文件服务器来进行文件持久化服务,这样多个系统之间如何共享文件就成了个问题,所以当初设计上采取了个偷懒的方式,直接将文件保存到MySql中,所有系统要用这些文件时,都去数据库中读取……

上面描述了问题产生的原因,现在项目已经上线,且运行了一段时间,发现文件存在数据库中导致数据库增长速度很快,而且未来迁移也不方便,所以考虑在原有直接存储到数据库中的方式,增加其他存储方式(注意是增加,不是全部替换),于是很简单的,直接给相应数据表增加一个字段标志存储方式(StorageMode),而原先存储文件数据的字段(Data)则用于存储其他方式持久化文件后对应的定位标志(比如文件路径等)

为了方便未来扩展出其他存储方式,所以我定义了如下存储接口

    /// <summary>/// 定义文件保存和获取的方式/// </summary>public interface IStorage{/// <summary>/// 文件保存,并返回保存后的编码结果/// </summary>/// <param name="fileType"></param>/// <param name="data"></param>/// <param name="fileName">要保存的文件名,如果不设置则随机命名</param>/// <returns></returns>byte[] Save(FileType fileType, byte[] data, string fileName = null);/// <summary>/// 根据请求编码获取对应的文件数据/// </summary>/// <param name="fileType"></param>/// <param name="input"></param>/// <returns></returns>byte[] Get(FileType fileType, byte[] input);/// <summary>/// 获取文件数据对应的文件名,如不支持则返回null/// </summary>/// <param name="fileType"></param>/// <param name="input"></param>/// <returns></returns>string GetFileName(FileType fileType, byte[] input);/// <summary>/// 删除文件数据/// </summary>/// <param name="fileType"></param>/// <param name="input"></param>/// <returns></returns>bool Remove(FileType fileType, byte[] input);}

其中FileType是按业务进行划分的文件类型,然后我依次定义了InDatabaseStorage(数据库存储)以及LocalDiskStorage(本地磁盘存储)两种实现,当然这部分在这里就不细写了,毕竟不是本文的关键……

下来到问题产生的重点了,为了统一文件的管理,以及符合ABP的设计风格,所以我们在Core层定义了FileRefManager这样一个Manager类来统一封装处理文件的管理,也就是在这里,我们完整的封装了如何根据存储方式来获取对应的IStorage,然后又如何在存储时进行转化,最后在获取时,为了尽可能少的修改其它代码,我们采用了一个取巧的方式,将文件通过对应途径获取数据流后,赋值回Data字段并予以返回,这样原先通过Data字段来获取文件流的其它代码就不用进行修改

想法是美好的,事实是残酷的,测试时,当我将某种FileType设置为存储到本地磁盘时,读取出来居然报“路径存在非法字符”,反复DEBUG,存进去没错,但读取就是错,这什么鬼?难道DEBUG后程序还做了其它什么我不知道的事情?

徒劳一天,一无所获后,第二天终于有所突破,因为可以确定,在保存成功时,不管是本地磁盘,还是数据库里的byte长度,都是正确的,而且不管保存后多久,只要我不通过程序查询,直接在数据库中查询,Data字段始终正确,而一旦我在程序中查询了,随后就可以立刻发现:数据库中的Data字段变了,又变成了以数据库存储时,存储的文件本身!!!而正是这个现象,脑中突然灵光一闪,当初看ABP资料时,貌似有看到过ABP会自动保存,赶紧去ABP官网一查,果然在UOW对应的说明中有如此一段,既然找到了问题,那解决问题相对就简单了,解决的途径也有好几种可以采纳

a)禁用UOW,但此方法可行性不高,因为按照ABP说明,UOW是否启用是按最外层调用方决定的,而UOW在ABP中其扩散范围并不仅仅只在Core层中

b)Data字段保持不变,返回时另外增加一个FactData字段,但这样需要将所有涉及文件存储的地方均进行代码修改,改动量太大,有些得不偿失

c)在返回文件实体时,既然不能对原数据实体进行修改,那么为何不干脆直接深Copy一个新的数据实体,然后在新实体上进行Data修正,最终返回新实体呢?此方案最终尝试果然可行,而且这样也无需修改其它涉及文件存储的代码

虽然问题解决了,但最终对于ABP自动保存这种设计思路还是觉得有些不可思议,就算认为我对实体做的任何修改,都可能应该进行持久化,但你总得有个途径让我告知系统不要自动保存吧,而禁用UOW这个方案又存在那么大的限制……


2018-06-11补充:

根据@protossyk 同学的评论,现在增加新的方案:

d)ABP自所以能自动保存,主要依赖的就是EF的实体跟踪,所以可以通过指定AsNoTracking来解决自动保存问题,写法示例如下

repository.GetAll().AsNoTracking().FirstOrDefaultAsync()


这篇关于ABP天坑--UOW自动保存修改的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/666576

相关文章

一文详解MySQL如何设置自动备份任务

《一文详解MySQL如何设置自动备份任务》设置自动备份任务可以确保你的数据库定期备份,防止数据丢失,下面我们就来详细介绍一下如何使用Bash脚本和Cron任务在Linux系统上设置MySQL数据库的自... 目录1. 编写备份脚本1.1 创建并编辑备份脚本1.2 给予脚本执行权限2. 设置 Cron 任务2

SQL Server修改数据库名及物理数据文件名操作步骤

《SQLServer修改数据库名及物理数据文件名操作步骤》在SQLServer中重命名数据库是一个常见的操作,但需要确保用户具有足够的权限来执行此操作,:本文主要介绍SQLServer修改数据... 目录一、背景介绍二、操作步骤2.1 设置为单用户模式(断开连接)2.2 修改数据库名称2.3 查找逻辑文件名

Java使用HttpClient实现图片下载与本地保存功能

《Java使用HttpClient实现图片下载与本地保存功能》在当今数字化时代,网络资源的获取与处理已成为软件开发中的常见需求,其中,图片作为网络上最常见的资源之一,其下载与保存功能在许多应用场景中都... 目录引言一、Apache HttpClient简介二、技术栈与环境准备三、实现图片下载与保存功能1.

MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理)

《MyBatisPlus中update_time字段自动填充失效的原因分析及解决方案(最新整理)》在使用MyBatisPlus时,通常我们会在数据库表中设置create_time和update... 目录前言一、问题现象二、原因分析三、总结:常见原因与解决方法对照表四、推荐写法前言在使用 MyBATis

Python使用smtplib库开发一个邮件自动发送工具

《Python使用smtplib库开发一个邮件自动发送工具》在现代软件开发中,自动化邮件发送是一个非常实用的功能,无论是系统通知、营销邮件、还是日常工作报告,Python的smtplib库都能帮助我们... 目录代码实现与知识点解析1. 导入必要的库2. 配置邮件服务器参数3. 创建邮件发送类4. 实现邮件

Oracle修改端口号之后无法启动的解决方案

《Oracle修改端口号之后无法启动的解决方案》Oracle数据库更改端口后出现监听器无法启动的问题确实较为常见,但并非必然发生,这一问题通常源于​​配置错误或环境冲突​​,而非端口修改本身,以下是系... 目录一、问题根源分析​​​二、保姆级解决方案​​​​步骤1:修正监听器配置文件 (listener.

Linux中修改Apache HTTP Server(httpd)默认端口的完整指南

《Linux中修改ApacheHTTPServer(httpd)默认端口的完整指南》ApacheHTTPServer(简称httpd)是Linux系统中最常用的Web服务器之一,本文将详细介绍如何... 目录一、修改 httpd 默认端口的步骤1. 查找 httpd 配置文件路径2. 编辑配置文件3. 保存

Python使用pynput模拟实现键盘自动输入工具

《Python使用pynput模拟实现键盘自动输入工具》在日常办公和软件开发中,我们经常需要处理大量重复的文本输入工作,所以本文就来和大家介绍一款使用Python的PyQt5库结合pynput键盘控制... 目录概述:当自动化遇上可视化功能全景图核心功能矩阵技术栈深度效果展示使用教程四步操作指南核心代码解析

SpringBoot实现文件记录日志及日志文件自动归档和压缩

《SpringBoot实现文件记录日志及日志文件自动归档和压缩》Logback是Java日志框架,通过Logger收集日志并经Appender输出至控制台、文件等,SpringBoot配置logbac... 目录1、什么是Logback2、SpringBoot实现文件记录日志,日志文件自动归档和压缩2.1、

SpringCloud使用Nacos 配置中心实现配置自动刷新功能使用

《SpringCloud使用Nacos配置中心实现配置自动刷新功能使用》SpringCloud项目中使用Nacos作为配置中心可以方便开发及运维人员随时查看配置信息,及配置共享,并且Nacos支持配... 目录前言一、Nacos中集中配置方式?二、使用步骤1.使用$Value 注解2.使用@Configur