使用chunkById方法时请不要进行排序!

发布时间 - 2020-11-04 00:00:00    点击率:

下面由laravel教程栏目给大家介绍使用chunkbyid方法时请不要进行排序!,希望对需要的朋友有所帮助!

使用 chunkById 方法的时候请不要进行排序

最近在做开发任务的时候碰到了个诡异的问题,于是分享给大家

问题说明

由于需要批量处理数据,并且这个数据的量很大,一次全部取出然后执行是不现实的,幸运的是 Laravel 为我们提供了 chunkById 方法来让我们方便的处理。伪代码如下

Student::query()
    ->where('is_delete', false)
    ->orderBy('id', 'DESC')
    ->chunkById(200, function($students) {
            // 在这里进行逻辑处理
    });

咋一眼看上去,并没有什么问题,但是实际执行代码的时候会发现 chunkById 只会执行第一次,第二次以后由于某种原因会停止执行。

查找原因

Laravel 的源码中 chunkById 代码如下

  public function chunkById($count, callable $callback, $column = null, $alias = null)
    {
        $column = is_null($column) ? $this->getModel()->getKeyName() : $column;
        $alias = is_null($alias) ? $column : $alias;
        $lastId = null;
        do {
            $clone = clone $this;
            $results = $clone->forPageAfterId($count, $lastId, $column)->get();
            $countResults = $results->count();
            if ($countResults == 0) {
                break;
            }
            if ($callback($results) === false) {
                return false;
            }
            $lastId = $results->last()->{$alias};
            unset($results);
        } while ($countResults == $count);
        return true;
    }

看起来没什么问题,由于 while 循环是根据 $countResults == $count 来判断的,那么我们 dump 一下这两个变量就会发现, 第一次这两个是一致的,第二次由于数据不一致导致程序停止。

在上面的代码中, $count 是由 $results = $clone->forPageAfterId($count, $lastId, $column)->get(); 来获得的,

继续查看 forPageAfterId 方法

public function forPageAfterId($perPage = 15, $lastId = 0, $column = 'id')
{
    $this->orders = $this->removeExistingOrdersFor($column);
    if (! is_null($lastId)) {
        $this->where($column, '>', $lastId);
    }
    return $this->orderBy($column, 'asc')
                ->take($perPage);
}

我们可以看到,在这里返回的结果是 orderBy 进行升序排列的, 而我们的原始代码是进行降序排列,就会导致 count 不一致,从而使 chunkById 结束执行。

解决方案

把之前的 orderBy('id', 'desc') 移除即可。

Student::query()
    ->where('is_delete', false)
    ->chunkById(200, function($students) {
            // 在这里进行逻辑处理
    });

总结

  • 以后使用 chunkById 或者 chunk 方法的时候不要添加自定义的排序

  • 骚到老,学到老。。。


# laravel  # count  # while  # 循环  # column  # 在这里  # 就会  # 这两个  # 给大家  # 的是  # 升序  # 让我们  # 是由  # 学到老  # 只会 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Windows Hello人脸识别突然无法使用  Laravel N+1查询问题如何解决_Eloquent预加载(Eager Loading)优化数据库查询  Laravel如何实现数据导出到PDF_Laravel使用snappy生成网页快照PDF【方案】  如何在不使用负向后查找的情况下匹配特定条件前的换行符  黑客如何通过漏洞一步步攻陷网站服务器?  Python企业级消息系统教程_KafkaRabbitMQ高并发应用  js代码实现下拉菜单【推荐】  ,南京靠谱的征婚网站?  QQ浏览器网页版登录入口 个人中心在线进入  高防服务器租用指南:配置选择与快速部署攻略  如何用西部建站助手快速创建专业网站?  Laravel怎么使用Intervention Image库处理图片上传和缩放  Windows10怎样连接蓝牙设备_Windows10蓝牙连接步骤【教程】  如何快速搭建个人网站并优化SEO?  微信公众帐号开发教程之图文消息全攻略  如何用搬瓦工VPS快速搭建个人网站?  Laravel怎么使用Session存储数据_Laravel会话管理与自定义驱动配置【详解】  深入理解Android中的xmlns:tools属性  php json中文编码为null的解决办法  WordPress 子目录安装中正确处理脚本路径的完整指南  Java Adapter 适配器模式(类适配器,对象适配器)优缺点对比  Swift中switch语句区间和元组模式匹配  制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?  JS中页面与页面之间超链接跳转中文乱码问题的解决办法  香港服务器选型指南:免备案配置与高效建站方案解析  Angular 表单中正确绑定输入值以确保提交与验证正常工作  Laravel如何记录自定义日志?(Log频道配置)  JavaScript中的标签模板是什么_它如何扩展字符串功能  Laravel Eloquent性能优化技巧_Laravel N+1查询问题解决  如何确保FTP站点访问权限与数据传输安全?  韩国网站服务器搭建指南:VPS选购、域名解析与DNS配置推荐  如何在七牛云存储上搭建网站并设置自定义域名?  韩国代理服务器如何选?解析IP设置技巧与跨境访问优化指南  湖南网站制作公司,湖南上善若水科技有限公司做什么的?  网站制作免费,什么网站能看正片电影?  在线制作视频网站免费,都有哪些好的动漫网站?  高防服务器租用如何选择配置与防御等级?  图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?  高性能网站服务器部署指南:稳定运行与安全配置优化方案  油猴 教程,油猴搜脚本为什么会网页无法显示?  Laravel请求验证怎么写_Laravel Validator自定义表单验证规则教程  微信小程序 scroll-view组件实现列表页实例代码  Laravel与Inertia.js怎么结合_使用Laravel和Inertia构建现代单页应用  javascript如何操作浏览器历史记录_怎样实现无刷新导航  Java解压缩zip - 解压缩多个文件或文件夹实例  JavaScript如何实现类型判断_typeof和instanceof有什么区别  黑客如何利用漏洞与弱口令入侵网站服务器?  Python进程池调度策略_任务分发说明【指导】  昵图网官网入口 昵图网素材平台官方入口  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?