[英]Best way to exclude results from a big db query in Laravel 5.2
我正在使用Laravel 5.2来构建一个大型产品库存,而我在查看将显示产品的问题时遇到问题。 这是一个例子:
我有很多产品 - 在我的数据库中准确地超过20k。 其中大多数我想每天展示,但大约500-1000我只需要在一周的某些日子显示。
Currenly我已经回顾了以下方法,每个方法都有其缺点:
创建表product_availability,其中每行包含product_id,weekday和available(true / false)列。 然而,根据这种结构,我有7行提供1个产品的信息,整个表格将包含~14万行,我认为这不是一个好主意 - 因为大多数产品无论如何都会每天都可用而且只有500 -1000不会。
我的第二个猜测是简单地在products表中添加一个'availability'列,它将包含以下格式的JSON字符串:
{“mo”:“true”,“tu”:“false”,“we”:“false”,“th”:“true”,“fr”:“false”,“sa”:“false”,“ su“:”true“}
然后我会这样查询:
$weekday = date('N');
switch ($weekday) {
case '1':
$builder->where('$availability->mo', 'true');
break;
case '2':
$builder->where('$availability->tu', 'true');
break;
case '3':
$builder->where('$availability->we', 'true');
break;
case '4':
$builder->where('$availability->th', 'true');
break;
case '5':
$builder->where('$availability->fr', 'true');
break;
case '6':
$builder->where('$availability->sa', 'true');
break;
case '7':
$builder->where('$availability->su', 'true');
break;
}
但IMO对于这个解决方案的问题在于它也将是一个缓慢而沉重的方法。
所以我的问题是:
什么是最快的查询来获得今天可用的所有产品,请记住,只有很小一部分产品不可用(如何排除它们,也许)?
我认为你的第一个选择是最好的解决方案(虽然有一些变化)。
我建议product_availability
的表结构类似于:
| id | product_id | mon | tue | wed | thu | fri | sat | sun |
| 1 | 13403 | 1 | 0 | 1 | 0 | 1 | 0 | 0 |
然后你可以查询类似的东西:
SELECT
products.*
FROM
products
LEFT JOIN
product_availability
ON
product_availability.product_id = products.id
WHERE
# show if no entry here (i.e. always to be shown)
product_availability.id IS NULL
OR
# only want to show specific days
product_availability.id IS NOT NULL AND CASE
WHEN DAYOFWEEK(CURDATE()) = 1 THEN `sun` = 1
WHEN DAYOFWEEK(CURDATE()) = 2 THEN `mon` = 1
WHEN DAYOFWEEK(CURDATE()) = 3 THEN `tue` = 1
WHEN DAYOFWEEK(CURDATE()) = 4 THEN `wed` = 1
WHEN DAYOFWEEK(CURDATE()) = 5 THEN `thu` = 1
WHEN DAYOFWEEK(CURDATE()) = 6 THEN `fri` = 1
WHEN DAYOFWEEK(CURDATE()) = 7 THEN `sat` = 1
END
这将显示所有产品。 如果product_availability
有产品ID的条目,那么它将获得当前的一周(如果您愿意,可以使用其他方法替换它),并查看相应的日期是否有1
来显示它。
只有很小一部分不可用
在这种情况下,您只需要在product_availability
添加一个条目。
这是一个小提琴: http ://sqlfiddle.com/#!9/ c2bbf1/3
希望这可以帮助 :)
编辑:您可以用来转换为雄辩查询的一些方法:
->leftJoin('product_availability', 'products.id', '=', 'product_availability.product_id')
->whereNull('product_availability.id')
->orWhere(function ($query) {
$query
->whereRaw('(product_availability.id IS NOT NULL AND CASE
WHEN DAYOFWEEK(CURDATE()) = 1 THEN `sun` = 1
WHEN DAYOFWEEK(CURDATE()) = 2 THEN `mon` = 1
WHEN DAYOFWEEK(CURDATE()) = 3 THEN `tue` = 1
WHEN DAYOFWEEK(CURDATE()) = 4 THEN `wed` = 1
WHEN DAYOFWEEK(CURDATE()) = 5 THEN `thu` = 1
WHEN DAYOFWEEK(CURDATE()) = 6 THEN `fri` = 1
WHEN DAYOFWEEK(CURDATE()) = 7 THEN `sat` = 1
END)');
});
或者你可以做DB :: raw查询
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.