当前位置:首页 > CMS教程 > 其它CMS > 列表

Laravel 实现数据软删除功能

发布:smiling 来源: PHP粉丝网  添加日期:2021-12-11 20:49:52 浏览: 评论:0 

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

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

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

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

4.已删除数据可以恢复

Model的软删除功能实现

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

  1. public function delete() 
  2.  if (is_null($this->getKeyName())) { 
  3.   throw new Exception('No primary key defined on model.'); 
  4.  } 
  5.  if (! $this->exists) { 
  6.   return
  7.  } 
  8.  if ($this->fireModelEvent('deleting') === false) { 
  9.   return false; 
  10.  } 
  11.  $this->touchOwners(); 
  12.  $this->performDeleteOnModel(); 
  13.  $this->fireModelEvent('deleted', false); 
  14.  return true; 
  15. protected function performDeleteOnModel() 
  16.  $this->setKeysForSaveQuery($this->newModelQuery()) 
  17.  ->delete(); 
  18.  $this->exists = false; 

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

  1. protected function performDeleteOnModel() 
  2.  if ($this->forceDeleting) { 
  3.   $this->exists = false; 
  4.   return $this->newModelQuery()->where( 
  5.     $this->getKeyName(), $this->getKey() 
  6.   )->forceDelete(); 
  7.  } 
  8.  return $this->runSoftDelete(); 
  9.  
  10. protected function runSoftDelete() 
  11.  $query = $this->newModelQuery() 
  12.       ->where($this->getKeyName(), $this->getKey()); 
  13.  $time = $this->freshTimestamp(); 
  14.  $columns = [$this->getDeletedAtColumn() => $this->fromDateTime($time)]; 
  15.  $this->{$this->getDeletedAtColumn()} = $time
  16.  if ($this->timestamps && ! is_null($this->getUpdatedAtColumn())) { 
  17.   $this->{$this->getUpdatedAtColumn()} = $time
  18.   $columns[$this->getUpdatedAtColumn()] = $this->fromDateTime($time); 
  19.  } 
  20.  $query->update($columns); 

Model查询过滤删除数据

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

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

  1. public static function bootSoftDeletes() 
  2.  static::addGlobalScope(new SoftDeletingScope); 

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

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

  1. protected function performJoin(Builder $query = null) 
  2.  $query = $query ?: $this->query; 
  3.  $farKey = $this->getQualifiedFarKeyName(); 
  4.  $query->join($this->throughParent->getTable(), $this->getQualifiedParentKeyName(), '='$farKey); 
  5.  if ($this->throughParentSoftDeletes()) { 
  6.   $query->whereNull( 
  7.    $this->throughParent->getQualifiedDeletedAtColumn() 
  8.   ); 
  9.  } 
  10.  
  11. public function throughParentSoftDeletes() 
  12.  return in_array(SoftDeletes::class, class_uses_recursive( 
  13.   get_class($this->throughParent) 
  14.  )); 
  15. public function getRelationExistenceQueryForSelfRelation(Builder $query, Builder $parentQuery$columns = ['*']) 
  16.  $query->from( $query->getModel()->getTable().' as ' 
  17.   .$hash = $this->getRelationCountHash() 
  18.  ); 
  19.  $query->join($this->throughParent->getTable(),  
  20.   $this->getQualifiedParentKeyName(), '='$hash.'.'.$this->secondLocalKey 
  21.  ); 
  22.  if ($this->throughParentSoftDeletes()) { 
  23.   $query->whereNull($this->throughParent->getQualifiedDeletedAtColumn()); 
  24.  } 
  25.  $query->getModel()->setTable($hash); 
  26.  return $query->select($columns)->whereColumn( 
  27.   $parentQuery->getQuery()->from.'.'.$query->getModel()->getKeyName(), '='$this->getQualifiedFirstKeyName() 
  28.  ); 

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

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

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

主要的方案是:

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

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

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

Tags: Laravel数据软删除

分享到: