use*_*752 8 php mysql pessimistic-locking laravel laravel-5
我正在试图弄清楚如何正确使用/测试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;
}));
}
Run Code Online (Sandbox Code Playgroud)
我尝试在2个浏览器中测试,浏览器1用户登录,浏览器2未登录,浏览器1点击刷新,然后锁定更新并在更新前60秒睡眠
在60秒内,我去浏览器2并点击刷新,但记录未锁定,我检查phpmyadmin并更新记录(在浏览器1的60秒锁定触发器内)
但是在60秒之后,浏览器1再次修改了记录(点100000)
所以我误解了lockforupdate用于?或者我测试不正确?
我期望的是在前60秒内浏览器2不应该修改行(带有加载图标的空白页或错误抛出?)
https://laravel.com/docs/5.2/queries#pessimistic-locking
我做了一些研究,但仍然无法理解sharedLock(LOCK IN SHARE MODE)和lockForUpdate(FOR UPDATE)之间的区别
顺便说一句,我确认数据库是innodb
use*_*752 12
这个工作,最后,但仍然不明白什么是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;
}));
}
Run Code Online (Sandbox Code Playgroud)
所以这是一个老问题,但我相信我的回答可以澄清如何->lockForUpdate()
工作
从 Laravel 文档:
共享锁可防止在提交事务之前修改所选行。
因此,正如所写的那样 - 从您调用它直到您的交易完成,锁都将处于活动状态。
记住:
->find(1)
就像->first()
,->get()
,->insert()
,->save()
等-它执行查询
->lockForUpdate()
就像->where()
,->select()
,join()
等等-它增加了查询,但不执行它
$model = \App\Models\User::find(1)->lockForUpdate();
- 您尝试在查询已执行后添加锁
$model = \App\Models\User::lockForUpdate()->find(1);
- 您在执行查询之前添加锁,因此锁在事务完成之前一直处于活动状态
不同之处在于,在第一个场景中->lockForUpdate()
,当您教导它是