laravel数据库操作方式(Laravel 实现数据软删除功能)
laravel数据库操作方式
Laravel 实现数据软删除功能
对于任何一个模型,如果需要使用软删除功能,需要在模型中使用 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 对象
总结
以上所述是小编给大家介绍的Laravel 实现数据软删除功能,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!
原文链接:https://mp.weixin.qq.com/s/zasGG9Ik4UZ7p9ZEv4YAkg
- mysql创建数据库教程(MySQL创建数据库并支持中文字符的操作方法)
- phpweb应用技术开发与数据库教程(IIS8 使用FastCGI配置PHP环境图文教程)
- 前端模态框数据(amazeui模态框弹出后立马消失并刷新页面)
- python快速数据分类(Python基于滑动平均思想实现缺失数据填充的方法)
- angularjs数据绑定类指令及作用(详解Angular数据绑定及其实现方式)
- sql数据分页如何查询(SQL分页查询方式汇总)
- 火狐查看json数据
- sparksql的string转日期格式(将string类型的数据类型转换为spark rdd时报错的解决方法)
- vue怎么接收后台的数据(Vue封装全局toast组件的完整实例)
- sqlserver数据库基本操作(SQL Server四个系统表的知识讲解)
- mysql常用数据模型(MySQL数据库基于sysbench实现OLTP基准测试)
- 数据库sql语句大全及例题(数据库之SQL技巧整理案例)
- python6个基础数据类型(计算机二级python学习教程3 python语言基本数据类型)
- sqlserver两表查询语句格式(sql server实现在多个数据库间快速查询某个表信息的方法)
- SQLServer数据库中开启CDC导致事务日志空间被占满的原因(SQLServer数据库中开启CDC导致事务日志空间被占满的原因)
- laravel新增数据表(laravel 多图上传及图片的存储例子)
- 不走心的古装造型 舒畅 毁容式 出演,萧蔷雷出新高度(不走心的古装造型)
- 嘉南传 第22集(嘉南传第22集)
- 哪版孙悟空最萌 黄渤躺萌了(哪版孙悟空最萌)
- 融入小人物的喜怒哀乐,黄渤饰演的角色为什么让人观看时欲罢不能(融入小人物的喜怒哀乐)
- 《极限挑战》深访都市夜归人,夜间打工者体验,黄磊录完憔悴了(极限挑战深访都市夜归人)
- Google 推出了一个游戏生成器,让不会编程的你也能自己设计游戏(推出了一个游戏生成器)
热门推荐
- html5+css样式代码(详解HTML5中CSS外观属性)
- python 聚类找出同一类别的数据(Python实现简单层次聚类算法以及可视化)
- python注册码实现(python实现Virginia无密钥解密)
- JavaScriptSerializer对Json对象的序列化和反序列化
- 树莓派vnc设置失败(树莓派安装宝塔面板后VNC无法登陆的问题说明)
- 性能监视器中常用计数器
- python简单代码实例(Python实现 版本号对比功能的实例代码)
- apache协议内容(Apache中rewrite伪静态规则介绍)
- linux lnmp安装教程(LNMP系列教程之 SSL安装WordPress博客程序下载与安装)
- python爬取豆瓣电影评论(python使用requests模块实现爬取电影天堂最新电影信息)
排行榜
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9