前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Laravel 如何实现数据的软删除

Laravel 如何实现数据的软删除

作者头像
写PHP的老王
发布2019-08-19 11:41:35
2.3K0
发布2019-08-19 11:41:35
举报
文章被收录于专栏:写PHP的老王写PHP的老王

对于任何一个模型,如果需要使用软删除功能,需要在模型中使用 Illuminate\Database\Eloquent\SoftDeletes这个 trait。软删除功能需要实现的功能有以下几点:

1.模型执行删除操作,只标记删除,不执行真正的数据删除

2.查询的时候自动过滤已经标记为删除的数据

3.可以设置是否查询已删除的数据,可以设置只查询已删除的数据

4.已删除数据可以恢复

Model的软删除功能实现

Illuminate\Database\Eloquent\Model中delete方法源码:

代码语言:javascript
复制
public function delete()
{
  if (is_null($this->getKeyName())) {
    throw new Exception('No primary key defined on model.');
  }
  if (! $this->exists) {
    return;
  }
  if ($this->fireModelEvent('deleting') === false) {
    return false;
  }
  $this->touchOwners();
  $this->performDeleteOnModel();
  $this->fireModelEvent('deleted', false);
  return true;
}

protected function performDeleteOnModel()
{
  $this->setKeysForSaveQuery($this->newModelQuery())
  ->delete();
  $this->exists = false;
}

因为在子类中使用了SoftDeletes trait,所以,SoftDeletesperformDeleteOnModel方法会覆盖父类的方法,最终通过 runSoftDelete方法更新删除标记。

代码语言:javascript
复制
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中实现查询自动过滤被删除数据,指定查询已删除数据功能。

代码语言:javascript
复制

public static function bootSoftDeletes()
{
  static::addGlobalScope(new SoftDeletingScope);
}

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

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

代码语言:javascript
复制
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)
  ));
}
代码语言:javascript
复制
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对象

具体内容,后续文章介绍。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-08-15,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 写PHP的老王 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Model的软删除功能实现
  • Model查询过滤删除数据
  • 远程关联数据的软删除处理
相关产品与服务
数据库
云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档