PHP 软删除,Laravel 实现数据软删除功能

protected function performDeleteOnModel()

{

if ($this->forceDeleting) {

$this->exists = false;

return $this->newModelQuery()->where(

$this->getKeyName(), $this->getKey()

)->forceDelete();

}

return $this->runSoftDelete();

}

protected function runSoftDelete()

{

$query = $this->newModelQuery()

->where($this->getKeyName(), $this->getKey());

$time = $this->freshTimestamp();

$columns = [$this->getDeletedAtColumn() => $this->fromDateTime($time)];

$this->{$this->getDeletedAtColumn()} = $time;

if ($this->timestamps && ! is_null($this->getUpdatedAtColumn())) {

$this->{$this->getUpdatedAtColumn()} = $time;

$columns[$this->getUpdatedAtColumn()] = $this->fromDateTime($time);

}

$query->update($columns);

}

Model查询过滤删除数据

Laravel中允许在Model中 static::addGlobalScope方法添加全局的 Scope 。这样就可以在查询条件中添加一个全局条件。Laravel中软删除数据的过滤也是使用这种方式实现的。

SoftDeletes trait中加入了 Illuminate\Database\Eloquent\SoftDeletingScope全局的 Scope 。并在 SoftDeletingScope 中实现查询自动过滤被删除数据,指定查询已删除数据功能。

public static function bootSoftDeletes()

{

static::addGlobalScope(new SoftDeletingScope);

}

远程关联数据的软删除处理

Scope的作用只在于当前模型,以及关联模型操作上。如果是远程关联,则还需要额外的处理。Laravel远程关联关系通过 hasManyThrough 实现。里面有两个地方涉及到软删除的查询。

protected function performJoin(Builder $query = null)

{

$query = $query ?: $this->query;

$farKey = $this->getQualifiedFarKeyName();

$query->join($this->throughParent->getTable(), $this->getQualifiedParentKeyName(), '=', $farKey);

if ($this->throughParentSoftDeletes()) {

$query->whereNull(

$this->throughParent->getQualifiedDeletedAtColumn()

);

}

}

public function throughParentSoftDeletes()

{

return in_array(SoftDeletes::class, class_uses_recursive(

get_class($this->throughParent)

));

}

public function getRelationExistenceQueryForSelfRelation(Builder $query, Builder $parentQuery, $columns = ['*'])

{

$query->from( $query->getModel()->getTable().' as '

.$hash = $this->getRelationCountHash()

);

$query->join($this->throughParent->getTable(),

$this->getQualifiedParentKeyName(), '=', $hash.'.'.$this->secondLocalKey

);

if ($this->throughParentSoftDeletes()) {

$query->whereNull($this->throughParent->getQualifiedDeletedAtColumn());

}

$query->getModel()->setTable($hash);

return $query->select($columns)->whereColumn(

$parentQuery->getQuery()->from.'.'.$query->getModel()->getKeyName(), '=', $this->getQualifiedFirstKeyName()

);

}

performJoin 中通过中间模型关联远程模型,会根据 throughParentSoftDeletes判断中间模型是否有软删除,如果有软删除会过滤掉中间模型被删除的数据。

以上就是Laravel实现软删除的大概逻辑。这里有一个细节,Laravel中软删除的标记是一个时间格式的字段,默认 delete_at。通过是否为null判断数据是否删除。

但是有的时候,项目中会使用一个整形的字段标记数据是否删除。在这样的场景下,需要对Laravel的软删除进行修改才能够实现。

主要的方案是:

1.自定义 SoftDeletes trait,修改字段名称,修改更新删除标记操作;

2.自定义 SoftDeletingScope 修改查询条件

3.自定义 HasRelationships trait,在自定义的 HasRelationships 中重写 newHasManyThrough 方法,实例化自定义的 HasManyThrough 对象

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值