当前位置:编程学习 > > 正文

laravel数据库操作方式(Laravel 实现数据软删除功能)

时间:2021-11-08 16:22:49类别:编程学习

laravel数据库操作方式

Laravel 实现数据软删除功能

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

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

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

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

4.已删除数据可以恢复

Model的软删除功能实现

  • ?
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • Illuminate\Database\Eloquent\Model 中delete方法源码:
  •  
  • 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,所以, SoftDeletes performDeleteOnModel 方法会覆盖父类的方法,最终通过  runSoftDelete 方法更新删除标记。

  • ?
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 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 中实现查询自动过滤被删除数据,指定查询已删除数据功能。

  • ?
  • 1
  • 2
  • 3
  • 4
  • public static function bootSoftDeletes()
  • {
  •  static::addGlobalScope(new SoftDeletingScope);
  • }
  • 远程关联数据的软删除处理

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

  • ?
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 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 对象

    总结

    以上所述是小编给大家介绍的Laravel 实现数据软删除功能,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!

    原文链接:https://mp.weixin.qq.com/s/zasGG9Ik4UZ7p9ZEv4YAkg

    上一篇下一篇

    猜您喜欢

    热门推荐