Laravel lockforupdate(悲观锁)

新手上路,请多包涵

我试图弄清楚如何正确使用/测试 lockforupdate,但我发现它的功能不像我预期的那样

这只是测试

public function index() {
        return dd(\DB::transaction(function() {
            if (\Auth::guard('user')->check()) {
                $model = \App\Models\User::find(1)->lockForUpdate();
                sleep(60);
                $model->point = 100000;
                $model->save();
            } else {
                $model = \App\Models\User::find(1);
                $model->point = 999;
                $model->save();
            }

            return $model;
        }));
}

我尝试在 2 个浏览器中进行测试,浏览器 1 用户已登录,浏览器 2 未登录,浏览器 1 点击刷新,然后将锁定更新并在更新前休眠 60 秒

在 60 秒内,我进入浏览器 2 并点击刷新,但记录未锁定,我检查 phpmyadmin 并更新记录(在浏览器 1 的 60 秒锁定触发内)

但60秒后,记录又被浏览器1修改(点100000)

所以我误解了 lockforupdate 的用途?还是我测试不正确?

我期望的是浏览器 2 不应在前 60 秒内修改该行(带有加载图标或错误抛出的空白页面?)

https://laravel.com/docs/5.2/queries#pessimistic-locking

我做了一些研究,但仍然无法理解 sharedLock(LOCK IN SHARE MODE) 和 lockForUpdate(FOR UPDATE) 之间的区别

顺便说一句,我确认数据库是 innodb

原文由 user259752 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 687
2 个回答

这项工作,终于,但仍然不明白 sharedLock(LOCK IN SHARE MODE) 和 lockForUpdate(FOR UPDATE) 有什么不同

    public function index() {
        return dd(\DB::transaction(function() {
            if (\Auth::guard('user')->check()) {
                $model = \App\Models\User::lockForUpdate()->find(1);
                sleep(30);
                $model->point = 100000;
                $model->save();
            } else {
                $model = \App\Models\User::lockForUpdate()->find(1);
                $model->point = $model->point + 1;
                $model->save();
            }

            return $model;
        }));
    }

原文由 user259752 发布,翻译遵循 CC BY-SA 3.0 许可协议

所以这是一个老问题,但我相信我的回答可以阐明 ->lockForUpdate() 是如何工作的

来自 Laravel 文档:

共享锁可防止选定的行在您的事务提交之前被修改。

所以正如它所写的那样 - 从您调用它直到您的交易完成,锁将处于活动状态。

记住:

->find(1) works like ->first() , ->get() , ->insert() , ->save() etc. - it executes the query

->lockForUpdate() works like ->where() , ->select() , join() etc. - it adds to the query, but doesn’t execute it

$model = \App\Models\User::find(1)->lockForUpdate(); 你尝试在查询已经执行后添加锁

$model = \App\Models\User::lockForUpdate()->find(1); - 在执行查询之前添加锁,因此锁在事务完成之前一直处于活动状态

不同之处在于,在第一种情况下 ->lockForUpdate() 您教导它是

原文由 NoOorZ24 发布,翻译遵循 CC BY-SA 4.0 许可协议

推荐问题
logo
Stack Overflow 翻译
子站问答
访问
宣传栏