Laravel关联模型中过滤结果为空的结果集(has和with区别)


Posted in PHP onOctober 18, 2018

首先看代码:

$userCoupons = UserCoupons::with(['coupon' => function($query) use($groupId){
 return $query->select('id', 'group_id', 'cover', 'group_number', 'group_cover')->where([
   'group_id' => $groupId,
 ]);
}])
// 更多查询省略...

数据结构是三张表用户优惠券表(user_coupons)、优惠券表(coupons),商家表(corps),组优惠券表(group_coupons) (为了方便查看,后两项已去除)

这里我本意想用模型关联查出用户优惠券中属于给定组gourpId的所有数据(如果为空该条数据就不返回)。

但有些结果不是我想要的:

array(20) {
 ["id"]=>
 int(6)
 ["user_id"]=>
 int(1)
 ["corp_id"]=>
 int(1)
 ["coupon_id"]=>
 int(4)
 ["obtain_time"]=>
 int(1539739569)
 ["receive_time"]=>
 int(1539739569)
 ["status"]=>
 int(1)
 ["expires_time"]=>
 int(1540603569)
 ["is_selling"]=>
 int(0)
 ["from_id"]=>
 int(0)
 ["sell_type"]=>
 int(0)
 ["sell_time"]=>
 int(0)
 ["sell_user_id"]=>
 int(0)
 ["is_compose"]=>
 int(0)
 ["group_cover"]=>
 string(0) ""
 ["is_delete"]=>
 int(0)
 ["score"]=>
 int(100)
 ["created_at"]=>
 NULL
 ["updated_at"]=>
 NULL
 ["coupon"]=>
 NULL // 注意返回了coupons为空的数据
}

记录中有的coupon有记录,有的为空。想想也是,with只是用sql的in()实现的所谓预加载。无论怎样主user_coupons的数据都是会列出的。

它会有两条sql查询,第一条查主数据,第二条查关联,这里第二条sql如下:

select `id`, `group_id`, `cover`, `group_number`, `group_cover` from `youquan_coupons` where `youquan_coupons`.`id` in (1, 2, 3, 4, 5, 7, 8, 9, 10, 11, 13, 14) and (`group_id` = 1) and `youquan_coupons`.`deleted_at` is null

如果第二条为空,主记录的关联字段就是NULL。

后来看到了Laravel关联的模型的has()方法,has()是基于存在的关联查询,下面我们用whereHas()(一样作用,只是更高级,方便写条件)

这里我们思想是把判断有没有优惠券数据也放在第一次查询逻辑中,所以才能实现筛选空记录。

加上whereHas()后的代码如下

$userCoupons = UserCoupons::whereHas('coupon', function($query) use($groupId){
  return $query->select('id', 'group_id', 'cover', 'group_number', 'group_cover')->where([
   'group_id' => $groupId,
  ]);
 })->with(['coupon' => function($query) use($groupId){
  return $query->select('id', 'group_id', 'cover', 'group_number', 'group_cover');
 }])-> // ...

看下最终的SQL:

select * from `youquan_user_coupons` where exists (select `id`, `group_id`, `cover`, `group_number`, `group_cover` from `youquan_coupons` where `youquan_user_coupons`.`coupon_id` = `youquan_coupons`.`id` and (`group_ids` = 1) and `youquan_coupons`.`deleted_at` is null) and (`status` = 1 and `user_id` = 1)

这里实际上是用exists()筛选存在的记录。然后走下一步的with()查询,因为此时都筛选一遍了,所以with可以去掉条件。

显然区分这两个的作用很重要,尤其是在列表中,不用特意去筛选为空的数据,而且好做分页。

总结

以上所述是小编给大家介绍的Laravel关联模型中过滤结果为空的结果集(has和with区别),希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

PHP 相关文章推荐
繁体中文转换为简体中文的PHP函数
Oct 09 PHP
解决phpmyadmin 乱码,支持gb2312和utf-8
Nov 20 PHP
使用PHPMYADMIN操作mysql数据库添加新用户和数据库的方法
Apr 02 PHP
PHP 无限分类三种方式 非函数的递归调用!
Aug 26 PHP
PHP中读写文件实现代码
Oct 20 PHP
php curl模拟post请求小实例
Nov 13 PHP
PHP中如何实现常用邮箱的基本判断
Jan 07 PHP
php计划任务之ignore_user_abort函数实现方法
Jan 08 PHP
PHP中的数组处理函数实例总结
Jan 09 PHP
Docker配置PHP开发环境教程
Dec 21 PHP
PHP获取文件扩展名的方法实例总结
Jun 10 PHP
php使用fputcsv实现大数据的导出操作详解
Feb 27 PHP
PHP使用glob方法遍历文件夹下所有文件的实例
Oct 17 #PHP
php 读取文件夹下所有图片、文件的实例
Oct 17 #PHP
PHP使Laravel为JSON REST API返回自定义错误的问题
Oct 16 #PHP
实现PHP中session存储及删除变量
Oct 15 #PHP
PHP实现用session来实现记录用户登陆信息
Oct 15 #PHP
PHP中使用CURL发送get/post请求上传图片批处理功能
Oct 15 #PHP
深入理解 PHP7 中全新的 zval 容器和引用计数机制
Oct 15 #PHP
You might like
php 静态页面中显示动态内容
2009/08/14 PHP
Discuz7.2版的faq.php SQL注入漏洞分析
2014/08/06 PHP
PHP中两个float(浮点数)比较实例分析
2015/09/27 PHP
PHP中set_include_path()函数相关用法分析
2016/07/18 PHP
[原创]PHP实现生成vcf vcard文件功能类定义与使用方法详解【附demo源码下载】
2017/09/02 PHP
PHP使用XMLWriter读写xml文件操作详解
2018/07/31 PHP
JS模拟多线程
2007/02/07 Javascript
5秒后跳转效果(setInterval/SetTimeOut)
2013/05/03 Javascript
从数组中随机取x条不重复数据的JS代码
2013/12/24 Javascript
javascript版的in_array函数(判断数组中是否存在特定值)
2014/05/09 Javascript
JavaScript实现随机替换图片的方法
2015/04/16 Javascript
JavaScript中的this使用详解
2016/07/27 Javascript
浅谈jquery采用attr修改form表单enctype不起作用的问题
2016/11/25 Javascript
jQuery实现花式轮播之圣诞节礼物传送效果
2016/12/25 Javascript
js如何获取网页所有图片
2017/05/12 Javascript
mock.js模拟前后台交互
2019/07/25 Javascript
小程序如何自主实现拦截器的示例代码
2019/11/04 Javascript
javascript跳转与返回和刷新页面的实例代码
2019/11/20 Javascript
JS highcharts实现动态曲线代码示例
2020/10/16 Javascript
[01:35:53]完美世界DOTA2联赛PWL S3 Magma vs GXR 第二场 12.13
2020/12/17 DOTA
玩转python爬虫之正则表达式
2016/02/17 Python
判断网页编码的方法python版
2016/08/12 Python
Python+PIL实现支付宝AR红包
2018/02/09 Python
关于Pycharm无法debug问题的总结
2019/01/19 Python
PyQt5实现五子棋游戏(人机对弈)
2020/03/24 Python
django的ORM模型的实现原理
2019/03/04 Python
pycharm设置当前工作目录的操作(working directory)
2020/02/14 Python
HTML5标签嵌套规则详解【必看】
2016/04/26 HTML / CSS
JOSEPH官网:英国奢侈时尚品牌
2018/01/31 全球购物
Eton丹麦官网:精美的男式衬衫
2020/05/27 全球购物
心理咨询承诺书
2014/05/20 职场文书
个人四风问题原因分析及整改措施
2014/09/28 职场文书
2015年双拥工作总结
2015/04/08 职场文书
旅游项目合作意向书
2015/05/08 职场文书
社团招新宣传语
2015/07/13 职场文书
Python查找算法的实现 (线性、二分,分块、插值查找算法)
2022/04/24 Python