Laravel中自增实现方案
admin
2023-01-19 21:00:59
0

工作中经常会对一列数据进行自增操作,设想一个场景。
总共发放10张优惠券(以下的操作以时间为序列)
1)用户1请求领取1张优惠券;
2)进程1查询到数据库中剩余10张,代码更新优惠券数量,此时为内存中优惠券数量为9;
3)此时,用户2请求领取1张优惠券。进程1并未将9写入到数据库中,所以此时进程2取到的优惠券的数量还是10。进程2计算数量,内存中的优惠券数量为9;
4)进程1更新数据库优惠券的数量,number = 9;
5)进程2更新数据库优惠券的数量,number = 9;
实际上已经发出去了两张优惠券,但是数据库中还剩9张优惠券。

所以呢,将数据load到内存中加减完成再入库是有点风险的。
两种解决方案:
1)加锁,别管是加个乐观锁还是悲观锁,这个不细聊了;
2)不在内存中加,直接在数据库层面进行set number = number - 1;在Laravel中,有increment和decrement封装来实现操作。
increment:

public function increment($column, $amount = 1, array $extra = [])
{
        if (! is_numeric($amount)) {
                throw new InvalidArgumentException('Non-numeric value passed to increment method.');
        }
        $wrapped = $this->grammar->wrap($column);
        $columns = array_merge([$column => $this->raw("$wrapped + $amount")], $extra);
        return $this->update($columns);
}

decrement:

public function decrement($column, $amount = 1, array $extra = [])
{
        if (! is_numeric($amount)) {
                throw new InvalidArgumentException('Non-numeric value passed to decrement method.');
        }
        $wrapped = $this->grammar->wrap($column);
        $columns = array_merge([$column => $this->raw("$wrapped - $amount")], $extra);
        return $this->update($columns);
}

都调用了update方法,看一下update:

public function update(array $values)
    {
        $sql = $this->grammar->compileUpdate($this, $values);
        var_dump($sql); // 打印一下sql语句
        return $this->connection->update($sql, $this->cleanBindings(
            $this->grammar->prepareBindingsForUpdate($this->bindings, $values)
        ));
    }

用两种方式进行数据库修改:
1)直接修改数量

$res = self::where('id', $data['id'])->update(['number' => self::raw('number + 1')] );

查看输出的sql语句:

 update  coupon  set  number  =  ?  where  id  = ?

2)用decrement

$res = self::where('id', $data['id'])->decrement('number', 2);

查看sql输出结果:

update coupon set number = number - 2 where id = ?;

所以并发较多的情况下建议使用increment和decrement。

相关内容

热门资讯

【今日要闻】“官方正版炸金花.... 【今日要闻】“官方正版炸金花.到底有挂吗?”太坑了原来有挂您好,官方正版炸金花这个游戏其实有挂的,确...
今日重磅消息“昆明文山麻将.开... 家人们!今天小编来为大家解答昆明文山麻将透视挂怎么安装这个问题咨询软件客服徽9784099的挂在哪里...
今日重大消息“闲来贵州麻将.辅... 今日重大消息“闲来贵州麻将.辅助器?”必胜开挂神器您好,闲来贵州麻将这个游戏其实有挂的,确实是有挂的...
玩家分享攻略“炫龙牛牛.到底有... 家人们!今天小编来为大家解答炫龙牛牛透视挂怎么安装这个问题咨询软件客服徽9752949的挂在哪里买很...
【第一财经】“西兵互娱牛牛.怎... 家人们!今天小编来为大家解答西兵互娱牛牛透视挂怎么安装这个问题咨询软件客服徽9752949的挂在哪里...
重磅消息“星禾卡五星.开挂神器... 有 亲,根据资深记者爆料星禾卡五星是可以开挂的,确实有挂(咨询软件无需打...
玩家分享攻略“家乡大贰.怎么装... 您好:家乡大贰这款游戏可以开挂,确实是有挂的,需要了解加客服微信【9784099】很多玩家在这款游戏...
我来教教您“七彩丹霞.是不是有... 家人们!今天小编来为大家解答七彩丹霞透视挂怎么安装这个问题咨询软件客服徽9784099的挂在哪里买很...
今日重大通报“闽悦麻将.怎么装... 您好:闽悦麻将这款游戏可以开挂,确实是有挂的,需要了解加客服微信【4282891】很多玩家在这款游戏...
玩家攻略科普“新海贝之城拼三张... 家人们!今天小编来为大家解答新海贝之城拼三张透视挂怎么安装这个问题咨询软件客服徽9752949的挂在...