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

相关文章

JAVA实现Token自动续期机制的示例代码

《JAVA实现Token自动续期机制的示例代码》本文主要介绍了JAVA实现Token自动续期机制的示例代码,通过动态调整会话生命周期平衡安全性与用户体验,解决固定有效期Token带来的风险与不便,感兴... 目录1. 固定有效期Token的内在局限性2. 自动续期机制:兼顾安全与体验的解决方案3. 总结PS

linux部署NFS和autofs自动挂载实现过程

《linux部署NFS和autofs自动挂载实现过程》文章介绍了NFS(网络文件系统)和Autofs的原理与配置,NFS通过RPC实现跨系统文件共享,需配置/etc/exports和nfs.conf,... 目录(一)NFS1. 什么是NFS2.NFS守护进程3.RPC服务4. 原理5. 部署5.1安装NF

Python函数的基本用法、返回值特性、全局变量修改及异常处理技巧

《Python函数的基本用法、返回值特性、全局变量修改及异常处理技巧》本文将通过实际代码示例,深入讲解Python函数的基本用法、返回值特性、全局变量修改以及异常处理技巧,感兴趣的朋友跟随小编一起看看... 目录一、python函数定义与调用1.1 基本函数定义1.2 函数调用二、函数返回值详解2.1 有返

Nginx屏蔽服务器名称与版本信息方式(源码级修改)

《Nginx屏蔽服务器名称与版本信息方式(源码级修改)》本文详解如何通过源码修改Nginx1.25.4,移除Server响应头中的服务类型和版本信息,以增强安全性,需重新配置、编译、安装,升级时需重复... 目录一、背景与目的二、适用版本三、操作步骤修改源码文件四、后续操作提示五、注意事项六、总结一、背景与

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

基于Redis自动过期的流处理暂停机制

《基于Redis自动过期的流处理暂停机制》基于Redis自动过期的流处理暂停机制是一种高效、可靠且易于实现的解决方案,防止延时过大的数据影响实时处理自动恢复处理,以避免积压的数据影响实时性,下面就来详... 目录核心思路代码实现1. 初始化Redis连接和键前缀2. 接收数据时检查暂停状态3. 检测到延时过

SpringBoot实现RSA+AES自动接口解密的实战指南

《SpringBoot实现RSA+AES自动接口解密的实战指南》在当今数据泄露频发的网络环境中,接口安全已成为开发者不可忽视的核心议题,RSA+AES混合加密方案因其安全性高、性能优越而被广泛采用,本... 目录一、项目依赖与环境准备1.1 Maven依赖配置1.2 密钥生成与配置二、加密工具类实现2.1

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

SQL Server跟踪自动统计信息更新实战指南

《SQLServer跟踪自动统计信息更新实战指南》本文详解SQLServer自动统计信息更新的跟踪方法,推荐使用扩展事件实时捕获更新操作及详细信息,同时结合系统视图快速检查统计信息状态,重点强调修... 目录SQL Server 如何跟踪自动统计信息更新:深入解析与实战指南 核心跟踪方法1️⃣ 利用系统目录