Laravel 过滤多个 WhereIn 数组

Posted

技术标签:

【中文标题】Laravel 过滤多个 WhereIn 数组【英文标题】:Laravel Filter Multiple WhereIn Array 【发布时间】:2018-07-15 23:05:16 【问题描述】:

我正在尝试为电子商务商店创建过滤器。用户可以按标签过滤产品,每个风味标签都附加到一个数组中:$tags_array

即 显示所有包含“猕猴桃”AND“薄荷醇”标签的产品

dd($tags_array);

array:2 [
  0 => "kiwi"
  1 => "menthol"
]

查询的过滤部分

$query = $tags_array ? $query
 ->join('product_product_tag', 'products.id', '=', 'product_product_tag.product_id')
 ->join('product_tags', 'product_tags.id', '=', 'product_product_tag.tag_id')
 ->whereIn('product_tags.value', $tags_array) : $query;

-一个产品有很多标签

-一个标签BelongsToMany产品

product_tags

id | value 
------------------
1  | kiwi 
2  | apple
3  | menthol 
4  | strawberry 

-数据透视表:product_product_tag

product_product_tag

id | product_id | tag_id 
----------------------------
1  |    1       |    1
2  |    1       |    3
----------------------------
3  |    2       |    1
4  |    2       |    2
5  |    2       |    3
----------------------------
6  |    3       |    1
7  |    3       |    4
----------------------------
8  |    4       |    3

问题

当前查询显示所有具有奇异果 OR 薄荷醇标签的产品。

实际结果:产品:1、2、3、4

预期结果:产品:1、2

当只应用一个标签时,此查询可以正常工作。如何使此过滤器包含应用的 多个 标签(产品具有奇异果标签 AND 薄荷醇标签)。用户可以按 0,1,2,3,... 标签进行过滤

是否有处理动态标签数量的最佳实践?

编辑:我正在使用 Laravel DB 查询生成器

第二次编辑:尝试 btl 的回答

  $query = $tags_array ? $query
      ->join('product_product_tag', 'products.id', '=', 'product_product_tag.product_id')
      ->join('product_tags', 'product_tags.id', '=', 'product_product_tag.tag_id') : $query;
  if($tags_array) 
      foreach ($tags_array as $tag) 
          $query->where('product_tags.value', '=', $tag);
      
  

第三次修改:转回以下内容:

$query = $tags_array ? $query
 ->join('product_product_tag', 'products.id', '=', 'product_product_tag.product_id')
 ->join('product_tags', 'product_tags.id', '=', 'product_product_tag.tag_id')
 ->where('product_tags.value', $tags_array) : $query;

这仍然是不正确的。当前过滤器返回找到的第一个标签的结果。将继续努力寻求更动态的解决方案

【问题讨论】:

【参考方案1】:

whereIn('product_tags.value', $tags_array) 将匹配数组中至少有一个标签的所有内容,因此它的行为类似于OR。如果只需要匹配所有选中标签的产品,则需要设置多个 where 子句,效果如下:

$query = Product::with('tags');
foreach ($tags_array as $tag) 
     $query->whereHas('tags', function($q) use ($tag) 
         $q->where('id', $tag);
     );
 
 $query->get();

编辑

$query = $tags_array ? $query
 ->join('product_product_tag', 'products.id', '=', 'product_product_tag.product_id')
 ->join('product_tags', 'product_tags.id', '=', 'product_product_tag.tag_id');

foreach ($tags_array as $tag) 
    $query->where('product_tags.id', '=', $tag);

【讨论】:

Ty,我理解这里的逻辑,但我的查询是使用 Laravel db builder 编写的。关于如何转换它的任何想法?我从标签中删除了 eloquent,如有任何混淆,请见谅 我不认为它会改变太多,你仍然会为查询起别名,然后遍历标签。看看我的编辑,看看是否有效。 查看我实施您的答案的第二次编辑。这适用于一个风味标签,但一旦添加第二个,就会有 0 个结果。

以上是关于Laravel 过滤多个 WhereIn 数组的主要内容,如果未能解决你的问题,请参考以下文章

laravel:如何从数据库端过滤记录

如何使 Laravel whereIn 不自动排序

Laravel PHP通过多个键值过滤多维数组

laravel:如何从数据库端过滤记录

将多个选择选项加载到数组中

Laravel Eloquent 不区分大小写的查询