gpt4 book ai didi

php - Laravel 5 记录编辑 checkin / checkout 功能(锁定)

转载 作者:行者123 更新时间:2023-11-29 04:00:14 25 4
gpt4 key购买 nike

我正在使用 Laravel 5.1 并尝试在用户打开记​​录编辑 View 时实现记录锁定,以便任何其他用户在释放锁定之前无法打开同一记录的编辑 View 。

显然有时用户永远不会完成编辑,所以我需要能够自动解锁记录(当用户进行任何其他交易时,或在超时事件后,或使用 ajax 后台保持事件状态调用)。

我查看了 lockForUpdate() 并阅读了有关 InnoDB 锁定功能的信息,但我无法获得有关它的真实信息(搜索了 20 多篇文章,他们似乎都在互相背诵,没有真实信息)。文档也没有提供太多信息,探索 Laravel 查询类代码让我取得了以下进展:

  1. 头痛。
  2. 我意识到它只能在查询生成器中使用(我需要它来使用 Eloquent)。
  3. 似乎只是跨国锁定,而不是 checkin / checkout 记录锁定。

似乎对术语和 mysql (innodb) 锁定的实际作用存在很大的混淆,或者可能只是我想弄清楚这一点。

长话短说,InnodDB/Laravel 锁定(根据我的需要)的主要警告是它不是持久的,因此当 php 脚本终止并且用户仍在编辑表单时它将自行重置。它仅用于交易。

我需要实现上述功能(编辑 checkin / checkout ),在我重新发明轮子之前,我很想知道是否已经有内置的 Laravel/PHP 功能。

如果没有,我正在考虑以下方法,并希望获得关于哪种方法最好的意见。

一个。使用 record_id、user_id、timestamp 创建一个 edit_checkins 表。

B.将 locked_at 列添加到记录表中,该列将保存时间戳或空值(类似于 deleted_at 列)。

我主要担心的是性能和“垃圾收集”,因为有时记录会被锁定但从未主动解锁。使用方法 A,我可以在一个简单的查询中从 edit_checkins 表中删除所有用户锁。但是,在检查记录是否被锁定时会有点慢,因为我将不得不进行表连接(我认为它应该可以忽略不计,因为编辑事件比其他事件少)。使用方法 B,检查速度更快,但我没有获得所有信息(例如 user_id),并且在不知道 user_id 的情况下实现解锁所有事件几乎是不可能的(我可能需要在记录中添加一个 locked_by 列)。

我希望我把问题说清楚了。感谢您的宝贵时间。

最佳答案

您要做的是对记录进行应用程序级锁定。您有一个业务级别要求,即一次只能有一个用户查看记录的编辑 View 。此“锁定”可以持续几秒、几分钟、几小时或您希望允许的任何最大超时。

这与记录上的数据库级锁完全不同。需要数据库级锁来确保两个更新语句不会同时在同一条记录上运行。这个锁只会持续到事务需要的时间,通常只有几毫秒。长时间运行或开放式数据库事务不是一个好主意。

您将需要设计自己的应用程序逻辑来执行您想要的操作。我没有看到任何现有的 Laravel 包。有一个叫laravel-record-lock ,但它不会执行您想要执行的操作,并且不会在多个请求中保持锁定。

我认为最灵活的设计是创建一个record_locks 表,然后创建一个与您希望可锁定 的任何模型的多态关系。 Polymorphic relationship documentation .入门指南:

数据库表:

record_locks
- id
- timestamps (if you want)
- lockable_id - integer
- lockable_type - string
- user_id - integer
- locked_at - datetime/timestamp

模型

class RecordLock extends Model
{
/**
* Polymorphic relationship. Name of the relationship should be
* the same as the prefix for the *_id/*_type fields.
*/
public function lockable()
{
return $this->morphTo();
}

/**
* Relationship to user.
*/
public function user()
{
return $this->belongsTo('App\User');
}

// additional functionality
}

现在您可以将多态关系添加到您想要锁定的任何模型:

class Book extends Model
{
/**
* Polymorphic relationship. Second parameter to morphOne/morphMany
* should be the same as the prefix for the *_id/*_type fields.
*/
public function recordLock()
{
return $this->morphOne('App\RecordLock', 'lockable');
}
}

class Car extends Model
{
/**
* Polymorphic relationship. Second parameter to morphOne/morphMany
* should be the same as the prefix for the *_id/*_type fields.
*/
public function recordLock()
{
return $this->morphOne('App\RecordLock', 'lockable');
}
}

最后,作为常规关系使用:

$book = \App\Book::first();
$lock = $book->recordLock; // RecordLock object or null

$car = \App\Car::first();
$lock = $car->recordLock; // RecordLock object or null

/**
* Accessing the relationship from the RecordLock object will
* dynamically return the type of object that was locked.
*/

$lock = \App\RecordLock::find(1);
$lockedObject = $lock->lockable; // \App\Book object

$lock = \App\RecordLock::find(2);
$lockedObject = $lock->lockable; // \App\Car object

最后一个旁注来解决您对查询构建器与 Eloquent 的担忧:Eloquent 模型回退到 Eloquent 查询构建器,而 Eloquent 查询构建器回退到普通查询构建器。如果您在 Eloquent 模型上调用方法,它会尝试在 Eloquent 查询构建器上调用它。如果它在那里不存在,它会尝试在普通查询生成器上调用它。如果它不存在,你会得到一个错误。

所以,如果你要执行 \App\User::lockForUpdate(),方法调用最终会过滤到普通的查询生成器(因为它不存在于 Eloquent 模型中或 Eloquent 查询生成器)。

关于php - Laravel 5 记录编辑 checkin / checkout 功能(锁定),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32021282/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com