问题来自.Net开发群友

问题

我想做一个自动生成单据号的功能,但是在EFCORE里没有行级锁,请有什么等价方案吗?

ChatGPT回答

在 EF Core 中确实没有提供行级锁(row-level locking)的功能,因为这需要对数据库进行显式的锁定操作,而 EF Core 的设计理念是尽可能不与数据库直接交互,而是通过抽象层面的数据操作来实现,因此不鼓励使用显式的锁定操作。

不过,在 EF Core 中提供了乐观并发控制(Optimistic Concurrency Control)的功能,可以用于处理类似于行级锁的需求。乐观并发控制基于版本号(Version)机制实现,每个实体对象都有一个版本号属性,当两个用户同时更新同一个实体对象时,EF Core 会比较它们的版本号,如果版本号不一致,则表示有冲突发生,需要执行相应的冲突解决逻辑。

因此,如果你想实现自动生成单据号的功能,并保证其唯一性,可以考虑使用乐观并发控制的方案。具体的实现思路如下:

  1. 在你的实体对象中增加一个版本号属性,类型为整数。

  2. 在生成单据号前,使用 EF Core 查询最后一条记录的单据号和版本号,得到最新的单据号和版本号。

  3. 在生成单据号时,将版本号加 1,并将单据号和版本号写入到新的记录中。如果写入数据库时发生冲突,则表示有其他用户同时在生成单据号,此时可以重新获取最新的单据号和版本号,并再次尝试生成。

以下是一个简单的示例代码,演示了如何使用乐观并发控制来生成唯一的单据号

public class Order
{
public int Id { get; set; }
public string OrderNo { get; set; }
public int Version { get; set; }
} public class OrderContext : DbContext
{
public DbSet<Order> Orders { get; set; } protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<Order>()
.Property(o => o.Version)
.IsConcurrencyToken();
}
} public class OrderService
{
private readonly OrderContext _context; public OrderService(OrderContext context)
{
_context = context;
} public string GenerateOrderNo()
{
string lastOrderNo = _context.Orders
.OrderByDescending(o => o.Id)
.Select(o => o.OrderNo)
.FirstOrDefault(); int lastVersion = _context.Orders
.OrderByDescending(o => o.Id)
.Select(o => o.Version)
.FirstOrDefault(); string newOrderNo = GenerateNewOrderNo(lastOrderNo); var order = new Order
{
OrderNo = newOrderNo,
Version = lastVersion + 1
}; _context.Orders.Add(order); try
{
_context.SaveChanges();
}
catch (DbUpdateConcurrencyException ex)
{
// 处理并发冲突
ex.Entries.Single().Reload();
lastOrderNo = _context.Orders
.OrderByDescending(o => o.Id)
.Select(o=> o.OrderNo) .FirstOrDefault(); lastVersion = _context.Orders .OrderByDescending(o => o.Id) .Select(o => o.Version) .FirstOrDefault(); newOrderNo = GenerateNewOrderNo(lastOrderNo); order.OrderNo = newOrderNo; order.Version = lastVersion + 1; _context.SaveChanges(); } return newOrderNo; } private string GenerateNewOrderNo(string lastOrderNo) { // 生成新的单据号逻辑 // ... }
}

反思

EFCORE 和 行级锁,如果提问者不知道这两者的含义根本没法问,就算问了,回答的东西,所以chatGPT对提问者也是有要求的

再没有chatGPT以前,我们码农如果遇到这个问题如何解决了?

很简单,使用各种搜索引擎,或者询问同行,搜集零碎的信息然后最终拼凑起来

程序员在有了chatGPT之后,有什么变化?

显而易见,缩短了搜索的时间

总结

其实不管是以前的面向搜索引擎编程,还是面向群友编程,其实都在暗中隐喻了程序员的未来——架构师

开发软件应用,最终是为了人类某个行业提供多一种可选方式

(例如外卖,古代也有外卖,让店小二提餐走路送去大户人家家里

  后来有了马车,单车,直到现在的摩托车,什么变了?方式变了,最终目的并没有改变

做开发其实也是一个道理,我们并不是做一个XX软件去替代,而是为了原本的最终目的提供多一种方式,而程序员则是这多出来的一种方式的设计师,让这个新的方式更加的容易被大众接纳和方便使用

要不然,我原本方式很简单,为什么要用你给我提供的方式呢?近些年开发行业也经常出现这种本末倒置的情况,很多码农都不知道自己开发这些软件应用的目的到底是为什么,对于我们开发者而言

不管什么技术,只是为了达成最终目的的手段,而不是最终目的

这也是我实在不喜欢背诵一些照本宣科的书面理论知识,而是更热衷提出和发现程序的到设计思路

说了这么多只是想说,程序员的未来肯定会是架构师,我们所有的技术积累我个人看来只是为了让我们的思维从普通人转换成程序员思维,而且我相信在未来,编程思维会如同吃饭一样,成为每个人的必备

最新文章

  1. Autofac 及 Autofac.WebApi 与MVC版本兼容问题
  2. 我对自己提的几个关于cocos2dx的几个问题
  3. java写的简单通用线程池demo
  4. linux常用命令6:关机重启命令
  5. subclipse svn 在64位win7下报Failed to load JavaHL Library
  6. [翻译][MVC 5 + EF 6] 1:创建数据模型
  7. linux install zh_CN(ubuntu)
  8. try..catch..finally执行顺序return
  9. java窗口按钮位置设置
  10. 51nod 1423 最大二“货” 单调栈
  11. SharePoint Framework 在web部件中使用已存在的JavaScript库 - JavaScript库的格式
  12. .NET HttpPost 上传文件图片到服务器
  13. PAT 1058 A+B in Hogwarts
  14. c++简单实现对mysql数据库操作
  15. Linux下Redis安装使用教程
  16. Ant打包可运行的Jar包(加入第三方jar包)
  17. LuoguP3948 数据结构
  18. php中文件操作常用函数有哪些
  19. jQuery下ajax事件的简单分析
  20. 可视化库-seaborn-热力图(第五天)

热门文章

  1. C温故补缺(九):字节对齐与排序
  2. 【消息队列面试】11-14:kafka高可靠、高吞吐量、消息丢失、消费模式
  3. 【每日一题】2021年12月6日-剑指 Offer 22. 链表中倒数第k个节点
  4. [FCC] Cash Register 计算找零
  5. 记一次 .NET 某安全生产信息系统 CPU爆高分析
  6. texlive编译lshort-zh-cn
  7. java8新特性学习笔记
  8. 旋转卡壳(求凸包直径)学习笔记 | 题解 P1452 [USACO03FALL]Beauty Contest G /【模板】旋转卡壳
  9. 使用Lighthouse更好推动项目性能优化,性能指标详解,优化方法,需要关注指标分析
  10. Redis 正则扫描key并删除