首页后端开发PHPModel的软删除功能如何实现,代码是什么

Model的软删除功能如何实现,代码是什么

时间2024-03-22 22:54:03发布访客分类PHP浏览788
导读:这篇文章给大家分享的是“Model的软删除功能如何实现,代码是什么”,文中的讲解内容简单清晰,对大家认识和了解都有一定的帮助,对此感兴趣的朋友,接下来就跟随小编一起了解一下“Model的软删除功能如何实现,代码是什么”吧。 对于任何一...
这篇文章给大家分享的是“Model的软删除功能如何实现,代码是什么”,文中的讲解内容简单清晰,对大家认识和了解都有一定的帮助,对此感兴趣的朋友,接下来就跟随小编一起了解一下“Model的软删除功能如何实现,代码是什么”吧。


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

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

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

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

4.已删除数据可以恢复

Model的软删除功能实现

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,所以,SoftDeletesperformDeleteOnModel 方法会覆盖父类的方法,最终通过runSoftDelete 方法更新删除标记。

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 对象

总结


感谢各位的阅读,以上就是“Model的软删除功能如何实现,代码是什么”的内容了,通过以上内容的阐述,相信大家对Model的软删除功能如何实现,代码是什么已经有了进一步的了解,如果想要了解更多相关的内容,欢迎关注网络,网络将为大家推送更多相关知识点的文章。

声明:本文内容由网友自发贡献,本站不承担相应法律责任。对本内容有异议或投诉,请联系2913721942#qq.com核实处理,我们将尽快回复您,谢谢合作!


若转载请注明出处: Model的软删除功能如何实现,代码是什么
本文地址: https://pptw.com/jishu/650932.html
Oracle执行存储过程的方法操作是什么 Oracle进行递归查询子节点的SQL语句怎样写

游客 回复需填写必要信息