如何用C#实现数据库的软删除模式?如何配置?

通过添加IsDeleted字段并结合EF Core实现软删除,首先在实体中增加bool类型IsDeleted属性,默认为false;然后在OnModelCreating中使用HasQueryFilter过滤已删除数据;接着重写SaveChanges方法,将Delete转为更新IsDeleted为true;可选扩展DeletedAt和DeletedBy记录删除信息;特殊场景下可用IgnoreQueryFilters查询已删除数据。

如何用c#实现数据库的软删除模式?如何配置?

在C#中实现数据库的软删除模式,核心是标记数据为“已删除”而不是真正从数据库中移除。通常通过添加一个 IsDeleted 字段来实现,并结合 Entity Framework Core(EF Core)进行查询过滤和操作控制。

1. 添加 IsDeleted 字段

在实体类中增加一个布尔类型的 IsDeleted 属性,用于标识该记录是否已被软删除。

public class Product{    public int Id { get; set; }    public string Name { get; set; }    public bool IsDeleted { get; set; } = false;}

2. 配置全局查询过滤器(EF Core)

使用 EF Core 的 HasQueryFilter 方法,在 OnModelCreating 中配置,使所有查询自动排除已删除的数据。

protected override void OnModelCreating(ModelBuilder modelBuilder){    modelBuilder.Entity()        .HasQueryFilter(p => !p.IsDeleted);}

这样每次查询 Product 时,EF Core 会自动加上 WHERE IsDeleted = 0 条件。

3. 重写 SaveChanges 实现软删除逻辑

拦截 SaveChanges 方法,将删除操作转换为更新 IsDeleted 字段。

public override int SaveChanges(){    foreach (var entry in ChangeTracker.Entries())    {        if (entry.State == EntityState.Deleted)        {            entry.Property("IsDeleted").CurrentValue = true;            entry.State = EntityState.Modified;        }    }    return base.SaveChanges();}

这段代码检查所有被标记为删除的实体,改为设置 IsDeleted = true 并改为修改状态,避免物理删除。

4. 可选:支持删除时间和删除人

如果需要审计信息,可以扩展字段:

public class Product{    public int Id { get; set; }    public string Name { get; set; }    public bool IsDeleted { get; set; } = false;    public DateTime? DeletedAt { get; set; }    public string DeletedBy { get; set; }}

SaveChanges 中补充赋值:

if (entry.State == EntityState.Deleted){    entry.Property("IsDeleted").CurrentValue = true;    entry.Property("DeletedAt").CurrentValue = DateTime.UtcNow;    // 可根据上下文设置 DeletedBy,如从用户服务获取    entry.State = EntityState.Modified;}

5. 查询包含已删除数据(特殊场景)

某些管理功能可能需要查看已删除数据,可通过 IgnoreQueryFilters() 绕过过滤:

var deletedProducts = context.Products    .IgnoreQueryFilters()    .Where(p => p.IsDeleted)    .ToList();

基本上就这些。软删除的关键在于统一拦截删除操作并配合查询过滤,确保业务逻辑不会误读已删除数据,同时保留恢复能力。

以上就是如何用C#实现数据库的软删除模式?如何配置?的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1440402.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 17:00:59
下一篇 2025年12月17日 17:01:13

相关推荐

  • 如何使用C#进行数据库单元测试?常用框架有哪些?

    使用内存数据库(如SQLite内存模式)结合EF Core进行C#数据库测试,通过xUnit/NUnit实现测试生命周期管理,Moq用于mock隔离依赖,区分单元与集成测试,确保数据操作逻辑正确且测试高效可重复。 在C#中进行数据库单元测试,核心目标是验证数据访问逻辑的正确性,同时避免依赖真实生产数…

    2025年12月17日
    000
  • 什么是依赖注入?在C#数据库项目中如何用它管理数据库上下文?

    依赖注入通过外部传入DbContext实现解耦,提升测试与维护效率。在C#数据库项目中,安装EF Core包后创建继承DbContext的类,如AppDbContext;在Program.cs中用AddDbContext注册服务并配置连接字符串,默认Scoped生命周期确保每请求单实例;控制器通过构…

    2025年12月17日
    000
  • 什么是数据库上下文工厂?在C#中如何使用它?

    数据库上下文工厂用于集中管理DbContext实例的创建与生命周期,解决直接new DbContext导致的资源泄漏和DI兼容性问题;通过实现IDbContextFactory接口,在EF Core 5.0+中可安全地在后台线程、命令行工具等场景按需创建上下文,适用于多租户、测试、IHostedSe…

    2025年12月17日
    000
  • C# 中的 nameof 表达式在验证中的优势?

    nameof表达式用于返回变量、参数或属性的名称字符串,提升参数验证的准确性和维护性。在方法中检查null值时,使用nameof可避免硬编码字符串错误,确保抛出ArgumentNullException时参数名正确无误。例如:public void ProcessPerson(Person pers…

    2025年12月17日
    000
  • 如何用C#实现数据库事务的隔离级别?如何设置?

    在C#中可通过SqlTransaction或TransactionScope设置事务隔离级别,以控制并发行为。1. 使用SqlConnection.BeginTransaction(IsolationLevel.ReadCommitted)可指定隔离级别,如ReadCommitted防止脏读;2. …

    2025年12月17日
    000
  • C# 中的模式匹配列表模式如何匹配序列?

    列表模式是C#11引入的特性,用于在switch或is表达式中按顺序匹配数组或列表结构。通过方括号[]定义元素模式,支持常量、变量、通配符_和..范围匹配。可实现精确值判断、变量提取、忽略特定位置或捕获子序列,适用于解析固定结构数据、函数式分支和输入验证。示例中Describe方法利用该特性分类数组…

    2025年12月17日
    000
  • 什么是数据库触发器?在C#中如何与触发器交互?

    触发器是数据库中自动执行的特殊存储过程,当表发生INSERT、UPDATE或DELETE操作时被激活,用于保障数据完整性、记录日志、实现级联更新或阻止非法操作。常见类型有AFTER触发器和INSTEAD OF触发器,前者在数据变更后执行,后者替代原操作执行。在C#开发中,无法直接调用触发器,但通过A…

    2025年12月17日
    000
  • C# 中的顶级语句在简单微服务中的用法?

    顶级语句简化微服务启动逻辑,无需编写完整的Program类和Main方法,直接通过几行代码配置WebApplication、添加服务与中间件,使代码更聚焦业务逻辑,适用于轻量级API、快速原型及简单场景,如健康检查等小型服务,配合隐式using更简洁,但在需复杂入口逻辑或团队规范要求时仍建议使用传统…

    2025年12月17日 好文分享
    000
  • C#中如何使用EF Core的种子数据功能?如何配置?

    在C#中使用EF Core的种子数据功能需在OnModelCreating中通过HasData方法配置,如为Role实体添加Id和Name初始化数据,并确保主键明确;该数据通过迁移机制管理,需执行“dotnet ef migrations add”和“update”命令使数据生效;修改种子数据后再次…

    2025年12月17日
    000
  • C#中如何使用EF Core的查询原生SQL插入/更新?

    使用EF Core执行原生SQL可通过ExecuteSqlRaw或ExecuteSqlInterpolated方法实现,推荐使用后者以避免SQL注入;两者均属于DbContext.Database属性,适用于插入、更新等操作,且建议采用异步版本如ExecuteSqlInterpolatedAsync…

    2025年12月17日
    000
  • C#的反射是什么?如何使用?

    C#的反射,简单来说,就是在程序运行时,你可以检查和操作程序集(Assembly)、模块(Module)和类型(Type)的信息。它就像一个探照灯,让你在黑暗中也能看清程序的内部结构。 反射允许你动态地创建对象、调用方法、访问字段和属性,甚至可以发现程序集中定义的类型。这在很多场景下非常有用,比如插…

    2025年12月17日
    000
  • C#性能优化技巧分享

    c#性能优化的核心在于减少不必要的开销,主要从内存管理、集合与循环优化、异步编程和字符串处理等方面入手。首先,避免频繁的对象分配,多用结构体、span和memory减少gc压力;其次,使用对象池如arraypool复用高成本对象;第三,选择合适的集合类型如hashset或dictionary提升查找…

    2025年12月17日
    000
  • C#中如何配置多个数据库连接?如何切换连接?

    答案:通过appsettings.json配置多连接字符串,为不同数据库创建独立DbContext,并在Program.cs中注册服务,运行时可动态传入连接字符串或使用工厂类按条件创建实例,实现灵活切换。 在C#中配置和切换多个数据库连接,通常通过配置文件管理连接字符串,并在运行时根据需要选择对应的…

    2025年12月17日
    000
  • C# 中的扩展方法如何为现有类型添加功能?

    扩展方法允许为现有类型添加新方法而无需修改源码或使用继承,其本质是静态方法但可像实例方法一样调用。定义时需在静态类中创建静态方法,并用 this 关键字修饰第一个参数以指定扩展类型,如 public static bool IsNullOrEmpty(this string str)。调用时看似实例…

    2025年12月17日
    000
  • C#的alias指令如何解决命名冲突?实际怎么用?

    c#命名冲突通常发生在引入多个同名类型的第三方库或模块时,using alias指令可通过为类型或命名空间定义局部别名来精确解决该问题,其作用范围限于当前文件,避免编译器歧义,而global using指令则是全局引入命名空间以减少重复代码,二者作用不同且可互补使用。 在C#的开发实践中,命名冲突是…

    2025年12月17日
    000
  • C# 中的索引器如何简化集合访问?

    索引器允许类通过方括号访问内部数据,如用整数或字符串作为索引封装数组或字典,提升代码可读性和封装性,支持参数类型重载且简化集合操作。 索引器(Indexer)让类像数组一样通过方括号 [] 直接访问内部数据,极大简化了集合操作。它常用于封装集合字段,提供更自然、直观的访问语法。 索引器的基本用法 定…

    2025年12月17日
    000
  • 什么是数据库的Computed Column?在C#中如何映射?

    计算列通过表达式基于其他列动态生成值,可持久化或非持久化,用于减少应用层重复逻辑。主流数据库如SQL Server、PostgreSQL、MySQL均支持。在C#中使用Entity Framework时,通过[DatabaseGenerated(DatabaseGeneratedOption.Com…

    2025年12月17日
    000
  • C#中事务处理的基本步骤是什么?如何确保数据一致性?

    开启事务需调用BeginTransaction()获取SqlTransaction对象,所有SqlCommand指定该事务;在事务中执行多条SQL命令并捕获异常;若全部成功则Commit提交,否则Rollback回滚;结合try-catch-finally和using语句确保资源释放与一致性;合理设…

    2025年12月17日
    000
  • C# 中的背景任务服务如何用于微服务?

    BackgroundService用于微服务中执行异步后台任务,如消息监听、数据同步等。它通过继承基类并重写ExecuteAsync方法实现长周期运行任务,支持依赖注入与CancellationToken优雅关闭,需捕获异常并加入延迟重试机制。在Program.cs中注册为托管服务,并结合健康检查提…

    2025年12月17日
    000
  • ASP.NET Core中的配置绑定是什么?如何实现?

    配置绑定是ASP.NET Core中将配置数据映射到强类型对象的核心机制,通过定义与配置结构匹配的C#类,并在Program.cs中使用services.Configure将IConfiguration节绑定到该类,再通过IOptions在应用中注入使用,实现类型安全、易维护的配置管理;其优势包括类…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信