如图,一个服务(service)对应一个用户(user),一个用户对应多个标签(tag),同时一个tag也可以通过中间表(pivot)对应对个用户。
现在业务需求如下:查service,这些service对应的user同时拥有tag1、tag2、tag3、tag4标签。
一个很容易犯的错误如下:
$tags = ['tag1', 'tag2', 'tag3', ''tag4]; $query = ServiceModel::query() $query->whereHas('user.tags', function($query) use($tags) {
foreach($tags as $tag){
$query->where('tag_name', '=', $tag);
}
}); $query-get();
这翻译过来是:找到某service,该service有user,且user有tag,这些tag的tag_name还要满足既等于tag1又等于tag2又等于tag3还等于tag4。
很显然,查询的结果为空,因为一个tag只有一个tag_name。
那换种方式:
$tags = ['tag1', 'tag2', 'tag3', ''tag4]; $query = ServiceModel::query() $query->whereHas('user.tags', function($query) use($tags) {
$query->whereIn('tag_name', $tags);
}); $query-get();
这样也不对,因为要同时拥有4种标签,这种写法标签间是OR的关系
翻译过来是:找到某service,该service有user,且user有tag,这些tag的tag_name只要出现在tag1、tag2、tag3、tag4中就选出来。
那如何找到某service,该service对应的user的标签既有tag1又有tag2又有tag3还有tag4呢?
其实只要把foreach置于外部即可:
$tags = ['tag1', 'tag2', 'tag3', ''tag4]; $query = ServiceModel::query() foreach($tags as $tag){
$query->whereHas('user.tags', function($query) use($tag) {
$query->where('tag_name', '=', $tag);
});
} $query-get();
这样翻译过来:(第一个循环)找到某service,该service有user,且user有tag,tag为tag1 ;循环加入的条件都是AND的关系。
其实还是化繁为简的思想:
找到某service,其对应user同时拥有多个tag的查询太难了,挺绕,那先找到只拥有1个tag(如tag1)的全部service记录总可以吧:
$query = ServiceModel::query() $query->whereHas('user.tags', function($query) use($tag) {
$query->where('tag_name', '=', 'tag1');
});
这和laravel手册上基于存在的关联查询例子几乎一样,那找到只拥有tag2的service记录,不就是把where里面的tag1换成tag2吗,同理其他tag,那这不就相当于对tag数组循环并添加whereHas条件吗...