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,所以,SoftDeletes
的performDeleteOnModel
方法会覆盖父类的方法,最终通过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