
最好使用 chunkbyid 而不是 chunk 以避免批量更新时丢失行。使用 chunk 可以在更新行后移动后续查询的偏移量,从而导致跳过未处理的行。
例如:
post::where('processed', 0)->chunk(100, function($posts) {
foreach($posts as $post) {
$post->processed = 1;
$post->save();
}
});
上面的代码生成以下查询。
select * from `posts` where `processed` = 0 limit 100 offset 0 select * from `posts` where `processed` = 0 limit 100 offset 100 ...
第一个块更新 100 行。第二个查询没有意识到这一点,因此跳过 100 个未处理的行,因为它仍然使用偏移量。
以上由
详细解释
太阮雄
当尝试使用 laravel 的 chunk() 方法处理有限数量的行时,我们可能期望以下代码以 2 为一组仅处理 5 个用户:
一套专门解决旅行社网上预定、发布、管理线路的强大系统,系统基于ASP+ACCESS数据库开发,功能强大,操作方便,系统设计完全符合旅行社的运做模式。系统着重体现易操作性,只要您会打字,便操作。系统由以下几个模块组成:1、线路的类别发布和管理2、线路的发布和管理3、线路的属性管理(是精品线路、还是普通线路)4、客户预定线路订单管理,人性化的区分为未处理订但和处理订单5、线路查询功能6、网站留言功能,
0
$query = \app\models\user::query()->take(5);
$query->chunk(2, function ($users) {
// process users
});
但是,这将处理数据库中的所有用户,一次两个。发生这种情况是因为 laravel 的 chunk() 方法忽略了应用于查询的 take() 限制,导致所有行都以块的形式处理。
为了确保只在块中处理有限数量的行(例如 5 个用户),我们可以实现一个自定义索引计数器,该计数器将在达到指定限制后中断分块循环。以下代码实现了这一点:
class UserProcessor
{
private int $index = 0;
private int $limit = 5;
public function process()
{
$query = \App\Models\User::query();
$query->chunk(2, function ($users) {
foreach ($users as $user) {
$this->index++;
// Process each user here
// Example: $user->processData();
if ($this->index >= $this->limit) {
// Stop processing after reaching the limit
return false; // This will stop chunking
}
}
});
}
}
注意
$index 和 $limit 是类的属性,而不是通过 use($index, $limit) 传递给闭包的方法变量。这是因为通过 use() 传递的变量会按值复制到闭包对象中。因此,闭包内的任何修改都不会影响原始值,这就是为什么它们必须是类属性才能正确更新和跟踪迭代之间的更改。
以上就是优化 Laravel 查询:分块数据的正确方法的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号