- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个用 Spring Boot + Angular 编写的电子商务网站。我需要在我的产品表中维护一个计数器来跟踪已售出的数量。但是,当许多用户同时订购同一件商品时,计数器有时会变得不准确。
在我的服务代码中,我有以下事务声明:@Transactional(propagation = Propagation.REQUIRES_NEW, isolation = Isolation.READ_COMMITTED)
其中,在保留顺序(使用 CrudRepository.save()
)之后,我执行了一个选择查询来总结到目前为止订购的数量,希望选择查询能够计算所有已提交的订单。但似乎并非如此,时不时地,计数器小于实际数量。
我的其他用例也会出现同样的问题:数量限制产品。我使用相同的事务隔离设置。在代码中,我将执行选择查询以查看已售出的数量,并在我们无法履行订单时抛出缺货错误。但是对于热门商品,我们有时会超卖该商品,因为每个线程都看不到其他线程中刚刚提交的订单。READ_COMMITTED
也是如此适合我的用例的隔离级别?或者我应该为这个用例做悲观锁定?
更新 05/13/17
我选择了 Ruben 的方法,因为我对 Java 的了解比数据库多,所以我选择了更简单的方法。这就是我所做的。
@Transactional(propagation = Propagation.REQUIRES_NEW, isolation = Isolation.SERIALIZABLE)
public void updateOrderCounters(Purchase purchase, ACTION action)
最佳答案
根据您检索已售商品总数的方式,可用选项可能会有所不同:
1. 如果您通过 sum
动态计算已售商品数量订单查询
我相信在这种情况下,您可以选择使用 SERIALIZABLE
事务的隔离级别,因为这是唯一一个支持 range locks
并防止 phantom reads
.
但是,我真的不建议使用这种隔离级别,因为它对您的系统有重大的性能影响(或者仅在设计良好的地方谨慎使用)。
友情链接:https://dev.mysql.com/doc/refman/5.7/en/innodb-transaction-isolation-levels.html#isolevel_serializable
2. 如果您在产品或与产品相关的其他行上维护一个计数器
在这种情况下,我可能会推荐使用 row level locking
例如 select for update
在检查产品可用性并增加已售商品计数的服务方法中。植入式广告的高级算法可能类似于以下步骤:
select for update
检索存储剩余/已售商品数量的行查询( @Lock(LockModeType.PESSIMISTIC_WRITE)
在存储库方法上)。 select for update
上执行 id
查询来获取锁)。您可以通过调用“entityManager.refresh(entity)”来实现这一点。 count
行的字段,如果该值符合您的业务规则,则增加/减少它。
@Transactional
public Product performPlacement(@Nonnull final Long id) {
Assert.notNull(id, "Product id should not be null");
entityManager.flush();
final Product product = entityManager.find(Product.class, id, LockModeType.PESSIMISTIC_WRITE);
// Make sure to get latest version from database after acquiring lock,
// since if a load was performed in the same hibernate session then hibernate will only acquire the lock but use fields from the cache
entityManager.refresh(product);
// Execute check and booking operations
// This method call could just check if availableCount > 0
if(product.isAvailableForPurchase()) {
// This methods could potentially just decrement the available count, eg, --availableCount
product.registerPurchase();
}
// Persist the updated product
entityManager.persist(product);
entityManager.flush();
return product;
}
pay
时,就在结账处理程序中)。另一个最小化锁定影响的有用技巧是不将“计数”列添加到产品本身,而是添加到与产品相关联的不同表上。这将阻止您锁定产品行,因为将在不同的行/表组合上获取锁,这些行/表组合仅在结帐阶段使用。
关于spring - 使用哪个 spring 事务隔离级别来维护已售产品的计数器?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43672684/
Android 项目中最低(最低 sdk)和最高(目标 sdk)级别是否有任何影响。这些东西是否会影响项目的可靠性和效率。 最佳答案 没有影响,如果您以 SDK 级别 8 为目标,那么您的应用将以 9
我将现有的 android 项目升级到 API 级别 31。我使用 Java 作为语言。我改变了 build.gradle compileSdkVersion 31 defaultConfig {
我正在使用 ionic 3 创建一个 android 应用程序,当我尝试上传到 playstore 时,我收到一个错误,提示我的应用程序以 api 25 为目标,当我检查我的 project.prop
我刚刚尝试将应用程序的目标和编译 API 级别更新为 29 (Android 10),并注意到我无法再编译,因为 LocationManager.addNmeaListener 只接受 OnNmeaM
我的代码没有在 Kitkat 上显示工具栏。 这是我的两个 Android 版本的屏幕截图。 Kitkat 版本: Lollipop 版: 这背后的原因可能是什么。 list 文件
我正在构建面向 API 级别 8 的 AccessabilityService,但我想使用 API 级别 18 中引入的功能 (getViewIdResourceName())。这应该可以通过使用 A
当我想在我的电脑上创建一个 android 虚拟机时,有两个选项可以选择目标设备。它们都用于相同的 API 级别。那么我应该选择哪一个呢?它们之间有什么区别? 最佳答案 一个是基本的 Android,
当我选择 tagret 作为 Android 4.2.2(API 级别 17)时,模拟器需要很长时间来加载和启动。 所以我研究它并通过使用 找到了解决方案Intel Atom(x86) 而不是 ARM
我有一个使用 Android Studio 创建的 Android 项目。我在项目中添加了一些第三方依赖项,但是当我尝试在 Android Studio 中编译时,我遇到了以下错误: Error:Ex
如上所述,如何使用 API 8 获取移动设备网络接口(interface)地址? 最佳答案 NetworkInterface.getInetAddresses() 在 API8 中可用。 关于andr
我想显示 Snackbar并使用图像而不是文本进行操作。 我使用以下代码: val imageSpan = ImageSpan(this, R.drawable.star) val b
我有一个用 python 编写的简单命令行程序。程序使用按以下方式配置的日志记录模块将日志记录到屏幕: logging.basicConfig(level=logging.INFO, format='
使用下面的代码,实现游戏状态以控制关卡的最简单和最简单的方法是什么?如果我想从标题画面开始,然后加载一个关卡,并在完成后进入下一个关卡?如果有人能解释处理这个问题的最简单方法,那就太好了! impor
我想创建一个可以找到嵌套树结构深度的属性。下面的静态通过递归找出深度/级别。但是是否可以将此函数作为同一个类中的属性而不是静态方法? public static int GetDepth(MenuGr
var myArray = [{ title: "Title 1", children: [{ title: "Title 1.1", children: [{
通过下面的代码,实现游戏状态来控制关卡的最简单、最容易的方法是什么?如果我想从标题屏幕开始,然后加载一个关卡,并在完成后进入下一个关卡?如果有人可以解释处理这个问题的最简单方法,那就太好了! impo
我有一个树结构,其中每个节点基本上可以有无限个子节点,它正在为博客的评论建模。 根据特定评论的 ID,我试图找出该评论在树中的深度/级别。 我正在关注 this guide that explains
考虑任何给定的唯一整数的数组,例如[1,3,2,4,6,5] 如何确定“排序度”的级别,范围从 0.0 到 1.0 ? 最佳答案 一种方法是评估必须移动以使其排序的项目数量,然后将其除以项目总数。 作
我如何定义一个模板类,它提供一个整数常量,表示作为输入模板参数提供的(指针)类型的“深度”?例如,如果类名为 Depth,则以下内容为真: Depth::value == 3 Depth::value
我的场景是:文件接收器应该包含所有内容。另一个接收器应包含信息消息,但需要注意的是 Microsoft.* 消息很烦人,因此这些消息应仅限于警告。两个sink怎么单独配置?我尝试的第一件事是: str
我是一名优秀的程序员,十分优秀!