- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我一直在试图了解亚马逊将如何处理并发写入问题,例如当多个用户试图购买可用数量只有一个的相同产品时。
考虑到“PRODUCT”表有以下字段:
我正在寻找有关数据库事务和锁定方面的详细解释,或者是用更详细的技术术语来处理这种情况的任何其他方法。
假设当 2 个用户在他们的购物车中添加了相同的产品并且一个用户继续付款时:
一个)
开始交易
SELECT * FROM PRODUCT WHERE ID = 1 AND STATUS = "AVAILABLE"FOR UPDATE;
将产品表中的数量字段更新为“0”和 STATUS =“SOLD OUT”
调用支付网关
如果付款超时回滚
提交
结束交易
============================================= ============================虽然上述交易仍在进行中,但其他用户仍然可以看到可用的产品,因为它仍未提交,他们也可以继续付款。
此外,如果有足够数量的产品可用,锁定产品行可能会阻止其他用户同时尝试购买相同的产品。
============================================= ============================
如何处理这个问题?
我已经阅读了以下关于在数据库上使用乐观锁定的博客,但发现很难深入理解具体如何实现。
最佳答案
让两个事务同时运行上面的代码,分别命名为tx1和tx2。tx1 和 tx2 都会首先尝试运行
SELECT * FROM PRODUCT WHERE ID = 1 AND STATUS = "AVAILABLE" FOR UPDATE;
这个查询也意味着从结果集中获取行上的锁。自然地,只有一个事务可以获取它,因此另一个事务将不得不等待。例如,如果 tx1 获得了锁,tx2 将等待直到 tx1 提交并释放锁。然后,tx2 也将继续并提交,因为它只是将数量设置为 0。因此,上述方法不会按预期工作,因为最终,两个用户都会收到有关他们购买的成功消息。但是,如果您想保留这种方法,不是直接将数量设置为零,而是将其减少 1,并对列进行约束,声明其值不能低于 0。以这种方式,tx2 将失败由于违反约束(记住 ACID 的 C - 一致性)并回滚,因此第二个用户将收到一条消息,表明他/她的购买失败。
或者,如果您不想为这个特定示例实现额外的约束而只是直接设置数量,您可以使用乐观锁定,它使用透明的行版本控制并且在性能方面也更便宜。所以对于id为1的产品(姑且称之为P),tx1生成一个新版本的P(P1),tx2同时生成另一个新版本的P(P2)。当 tx1 和 tx2 中的任何一个尝试提交时,系统会注意到 P 有新的未提交版本,即 P1 和 P2,并且只需要批准其中一个。结果,P1可能被选为接受的P的新版本,从而tx1成功,购买成功,P2被拒绝,tx2被中止并回滚,第二个用户收到关于他/她的购买失败的消息。另请注意,相反的情况也可能发生,即 P2 被接受而 P1 被拒绝。
最后,关于乐观锁和悲观锁考虑以下几点:
就我个人而言,我会选择悲观锁定但修复上述事务,因为对于大于 1 的数量,乐观锁定仍会导致一个事务中止,而使用悲观锁定可以成功地为两个事务提供服务,但会稍微牺牲性能。
While still above transaction is going on, other users can still see the PRODUCT available as it is still not committed and they may also proceed to PAY.
关于这个,可能与用户屏幕的刷新率有关。即使用户能够继续付款(前端的陈旧数据),后端也应验证用户是否有资格进行购买,如果是,则继续购买(所有这些都在一次交易中完成) )
Also, if the sufficient quantity of product is available, taking a lock on the product row may stop other users trying to buy the same product at same time.
这就是悲观锁的重点,性能下降是其主要影响,但只要数量不为0,所有事务都可以服务。乐观锁只有1成功,其他失败,提示用户再次尝试购买。
关于java - 电子商务应用程序设计(例如 : Amazon/ebay/flipkart ): Concurrency issues handling,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53976375/
我想使用foreach 来等待线程终止。但是,出现以下错误,没有实现。请告诉我。 cannot move out of `*handle` which is behind a shared refer
如果在TypoScript中未配置给定的typeNum,则TYPO3将抛出Exception/CMS/1294587217。 背景:从另一个系统迁移到TYPO3后,我们遇到了许多此类异常,因为在那里使
我需要一个带有 2 个 handle 的 slider ,一个可拖动,另一个固定。我正在使用 Jquery UI slider 。这是我到目前为止尝试过的:http://jsfiddle.net/8K
给定文件的HANDLE(例如C:\\FolderA\\file.txt),我想要一个函数,该函数会将HANDLE返回到包含的目录(在前面的示例中,它将是C:\\FolderA的HANDLE)。例如:
我想通过Automic在Unix中检查文件。如果该文件不存在,则应切换主机并检查文件是否存在。 问题是,我现在不执行错误处理。 每当脚本对象正在处理并且找不到文件时,skript都会中止。我在skri
鉴于: fruitid('Apple', 'Granny Smith', 1). fruitid('Apple', 'Cox', 2). fruitid('Pear', 'Bartlett', 3).
我有一个基于Spring的Wicket应用程序。 有一个池化的数据源bean。 现在,当MySQL死了时,我得到了带有堆栈跟踪的默认Wicket错误页面。 我想处理这种情况,只允许某些页面完全显示(静
我希望能够一次查询多个句柄,其中表格具有相同的格式,例如: 句柄:8000,8001,8003表:foo 想要做这样的事情: x:hopen `8000`8001`8003 x select from
我对在Swift 3中引发自定义异常有些困惑。 在C++中,我可以执行此操作以立即停止方法中的进程,抛出错误并进行处理,而无需进一步进行操作。 void foo() { try {
我一直在阅读MSDN开发人员COM指南。但是this page上的代码令人困惑。在此处复制: The following code sample shows the recommended way o
我有一个计划的批处理文件每天都会启动的过程。如果有错误,我需要内置错误处理才能重启进程。所有这些在大多数情况下都有效,但是我每个月都会收到一次超时错误,所以这是不可避免的。该进程不会将错误级别输出到b
我正在尝试从 chartlyrics API 获取歌词。我编写了一个可以运行但不能在循环内运行的 R 函数。我的脚本是: library(httr) library(RCurl) library(XM
在libuv事件循环中调用prepare handle callback和check handle callback的原因是什么? 最佳答案 I/O 操作发生在这两者之间,因此您可能希望在阻塞 I/O
我正在尝试在 R 中安装 BTYplus 包。 devtools::install_github("mplatzer/BTYDplus", dependencies=TRUE) library(BTY
我有一个Arduino,可以使用pySerialTransfer库通过串行与Mac正常通信,并且可以运行数小时。然后是某种形式的串行中断-尽管一夜间发生时我一直无法确定原因,但只要从笔记本电脑上拔下A
我是hooks和async/await的新手。我正在尝试处理Axios调用中的错误,并且不确定如何使用then/catch或try/catch处理我的API调用中的错误。 在基于类的React中,我将
我正在尝试向脚本中添加一些内容,以便让我知道我复制的文件是否已被完全复制。 基本上,我要压缩一堆文件,然后将它们发送到网络上的映射驱动器。然后,一旦文件被成功复制,我将脚本删除原始位置的文件。该脚本可
我有一个圆形 slider ,其中绘制了一条贝塞尔弧,一个圆弧在 slider 的起点和终点有两个 handle ,圆弧是在圆形 slider 中绘制的。 借助开始和结束 handle ,我可以沿着圆
删除 NULL 指针是安全的。 int* p = NULL; delete p; // ok, secure 句柄是什么? HANDLE h = NULL; CloseHandle(h
如果您没有在 dojo.connect 期间返回的“句柄”,您如何删除 dojo 连接事件? 我的示例涉及将一组事件动态分配给一组对象。 (为简单起见,事件是 onclick 和 ondblclick
我是一名优秀的程序员,十分优秀!