
本文介绍在 laravel 中遍历对象数组时,如何避免因使用 `next()` 导致的“trying to get property of non-object”错误,并提供更健壮、可读性更强的拆分方案——通过预提取唯一 cnpj 值实现精准分组。
在 Laravel(或纯 PHP)中处理对象数组时,若需根据相邻元素差异进行逻辑判断(例如按 cnpj 字段拆分数组),直接依赖 next() 函数极易引发运行时错误。根本原因在于:foreach 是值拷贝式遍历,内部指针与 next() 所操作的数组指针不同步;且当遍历至最后一个元素时,next() 返回 false,导致 $next->cnpj 尝试访问非对象属性,触发致命错误。
更可靠的做法是分离关注点:先提取关键标识(如 cnpj),再基于该标识进行分类。以下为优化后的完整实现:
// 步骤 1:提取所有 CNPJ 并去重(保留顺序)
$cnpjs = array_unique(array_column($finances, 'cnpj'));
sort($cnpjs); // 确保索引有序,便于后续判断
// 步骤 2:初始化分组容器
$aiuaEd = []; // 默认组(匹配第一个 CNPJ)
$aiua = []; // 第二组(匹配第二个 CNPJ,仅当存在时)
// 步骤 3:按 CNPJ 分类(兼容 1 或 2 种 CNPJ 的场景)
foreach ($finances as $finance) {
if (count($cnpjs) > 1 && $finance->cnpj === $cnpjs[1]) {
$aiua[] = $finance;
} else {
$aiuaEd[] = $finance;
}
}✅ 优势说明:
- 安全:完全规避 next() 指针异常与 null/false 访问风险;
- 清晰:逻辑分三步,语义明确,易于维护和测试;
- 高效:array_column() + array_unique() 时间复杂度为 O(n),远优于嵌套循环;
- 鲁棒:自动适配单 CNPJ(全部归入 $aiuaEd)或双 CNPJ 场景。
⚠️ 注意事项:
- 本方案假设业务约束为「最多两种不同 CNPJ」。若需支持 N 组动态拆分,推荐改用 collect($finances)->groupBy('cnpj')->values()->toArray()(Laravel 集合方法),返回多维分组数组;
- 若 $finances 可能为空,请在开头添加 if ($finances->isEmpty()) { return [$aiuaEd, $aiua]; } 防御性检查;
- 使用严格比较 === 替代 ==,避免字符串与数字隐式转换导致误判。
总结:在数组遍历中依赖“下一个元素”时,应优先考虑预处理 + 映射分类,而非操纵内部指针。这不仅是解决当前问题的最佳实践,更是构建可扩展、易调试 Laravel 数据处理逻辑的关键原则。










