从 Laravel 5.2 中的大数据库查询中排除结果的最佳方法
Best way to exclude results from a big db query in Laravel 5.2
我正在使用 Laravel 5.2 建立一个大的产品库存,但在显示产品的查询中遇到了问题。这是示例:
我有很多产品 - 在我的数据库中准确地说超过 20k。其中大部分我想每天展示,但大约 500-1000 个我只需要在一周的特定几天展示。
目前我已经回顾了以下方法,每种方法都有其缺点:
创建一个 table product_availability,其中每行将包含 product_id、工作日和可用 (true/false) 列。然而,使用这种结构,我将有 7 行提供 1 种产品的信息,整个 table 将包含 ~ 140 000 行,我认为这不是一个好主意 - 因为大多数产品每天都有货无论如何,只有 500-1000 不会。
我的第二个猜测是简单地向产品 table 添加一个 '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 是它也将是缓慢而繁重的方法。
- 我的第三个猜测是倒推,不是检查产品今天是否有货,而是检查哪一天没有。这样我会得到所有 'availability' 列为空的产品,对于这些产品,它是 运行 一个 if 语句,然后以某种方式添加它们,但我不知道如何构造它要么。
所以我的问题是:
获得今天所有可用产品的最快查询是什么,请记住只有一小部分产品不可用(可能如何排除它们)?
我认为你的第一个选择是最好的解决方案(尽管有一些变化)。
我建议 product_availability
的 table 结构类似于:
| 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
展示一下。
only a very small fraction of them wouldn't be available
在这种情况下,您只需要在 product_availability
.
中添加一个条目
这是一个fiddle:http://sqlfiddle.com/#!9/c2bbf1/3
希望对您有所帮助:)
编辑:您可以使用某些方法转换为 eloquent 查询:
->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 查询
我正在使用 Laravel 5.2 建立一个大的产品库存,但在显示产品的查询中遇到了问题。这是示例:
我有很多产品 - 在我的数据库中准确地说超过 20k。其中大部分我想每天展示,但大约 500-1000 个我只需要在一周的特定几天展示。
目前我已经回顾了以下方法,每种方法都有其缺点:
创建一个 table product_availability,其中每行将包含 product_id、工作日和可用 (true/false) 列。然而,使用这种结构,我将有 7 行提供 1 种产品的信息,整个 table 将包含 ~ 140 000 行,我认为这不是一个好主意 - 因为大多数产品每天都有货无论如何,只有 500-1000 不会。
我的第二个猜测是简单地向产品 table 添加一个 '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 是它也将是缓慢而繁重的方法。
- 我的第三个猜测是倒推,不是检查产品今天是否有货,而是检查哪一天没有。这样我会得到所有 'availability' 列为空的产品,对于这些产品,它是 运行 一个 if 语句,然后以某种方式添加它们,但我不知道如何构造它要么。
所以我的问题是:
获得今天所有可用产品的最快查询是什么,请记住只有一小部分产品不可用(可能如何排除它们)?
我认为你的第一个选择是最好的解决方案(尽管有一些变化)。
我建议 product_availability
的 table 结构类似于:
| 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
展示一下。
only a very small fraction of them wouldn't be available
在这种情况下,您只需要在 product_availability
.
这是一个fiddle:http://sqlfiddle.com/#!9/c2bbf1/3
希望对您有所帮助:)
编辑:您可以使用某些方法转换为 eloquent 查询:
->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 查询