php 乐观锁原理,浅谈Yii乐观锁的使用及原理

本文介绍了Yii乐观锁的使用及原理,自己做个学习笔记,也分享给大家,希望对大家有用处

原理:

数据表中使用一个int类型的字段来存储版本号,即该行记录的版本号。更新数据时,对比版本号是否一致

sql查询代码如下(伪代码)

update `test_ver` set `name`="lili" and `ver`=2 where `id`=1 and `ver`=1

即在更新时的where查询条件中,带上之前查询记录时得到的版本号,如果其他线程已经修改了该记录,则版本号势必不会一致,则更新失败

示例

数据表

假设有如下数据表

9a014e84e253a1c4837bfa9021229709.png

模型类

appmodelsTestVer

该模型类,重写BaseActiveRecord类中的optimisticLock方法

声明用于记录版本号的字段

/**

* 乐观锁

* @return string

*/

public function optimisticLock()

{

return 'ver';

}

public function updateRecord(){

$ver = self::findOne(['id'=>1]);

$ver->name = "lili";

$res = $ver->update();

return $res;

}

updateRecord修改id为1的记录

控制器

控制器中调用updateRecord方法

public function actionVersion(){

$testVer = new TestVer();

$res = $testVer->updateRecord();

return $this->render('version');

}

Yii Debugger结果

查看database选项,可以查看到实际执行的sql语句。

有一条语句如下

UPDATE `test_ver` SET `name`='lili', `ver`='2' WHERE (`id`='1') AND (`ver`='1')

Yii乐观锁实现原理

实现原理在yiidbBaseActiveRecord::updateInteranl()方法

protected function updateInternal($attributes = null)

{

if (!$this->beforeSave(false)) {

return false;

}

// 获取等下要更新的字段及新的字段值

$values = $this->getDirtyAttributes($attributes);

if (empty($values)) {

$this->afterSave(false, $values);

return 0;

}

// 把原来ActiveRecord的主键作为等下更新记录的条件,

// 也就是说,等下更新的,最多只有1个记录。

$condition = $this->getOldPrimaryKey(true);

// 获取版本号字段的字段名,比如 ver

$lock = $this->optimisticLock();

// 如果 optimisticLock() 返回的是 null,那么,不启用乐观锁。

if ($lock !== null) {

// 这里的 $this->$lock ,就是 $this->ver 的意思;

// 这里把 ver+1 作为要更新的字段之一。

$values[$lock] = $this->$lock + 1;

// 这里把旧的版本号作为更新的另一个条件

$condition[$lock] = $this->$lock;

}

$rows = $this->updateAll($values, $condition);

// 如果已经启用了乐观锁,但是却没有完成更新,或者更新的记录数为0;

// 那就说明是由于 ver 不匹配,记录被修改过了,于是抛出异常。

if ($lock !== null && !$rows) {

throw new StaleObjectException('The object being updated is outdated.');

}

$changedAttributes = [];

foreach ($values as $name => $value) {

$changedAttributes[$name] = isset($this->_oldAttributes[$name]) ? $this->_oldAttributes[$name] : null;

$this->_oldAttributes[$name] = $value;

}

$this->afterSave(false, $changedAttributes);

return $rows;

}

从上面的代码中,我们不难得出:

当 optimisticLock() 返回 null 时,乐观锁不会被启用。

版本号只增不减。

通过乐观锁的条件有2个,一是主键要存在,二是要能够完成更新。

当启用乐观锁后,只有下列两种情况会抛出 StaleObjectException 异常:

当记录在被别人删除后,由于主键已经不存在,更新失败。

版本号已经变更,不满足更新的第二个条件。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值