
本教程详细阐述了如何在 javaScript 中对嵌套 对象 数组进行列值一致性校验。针对“若某列任一元素有值,则该列所有元素必须有值”的业务需求,文章提出了一种基于 Object.keys、map 和 every 等现代 js 数组方法的解决方案,有效避免了传统多层 循环 的复杂性,提高了代码的可读性和维护性。
引言
在 前端 开发中,我们经常需要处理结构化的数据,尤其是来自 后端 API 的json 数组。对这些数据进行有效性校验是确保应用健壮性的关键一步。本文将探讨一种特定但常见的校验场景:确保对象数组中某一“列”的数据一致性,即如果该列的任何一个“行”有值,那么该列的所有“行”都必须有值。
业务场景与校验规则
假设我们有一个包含校准数据的对象数组,其结构如下所示。每个对象代表一个 Axis 的校准数据,并包含 P1 到 P10 以及 year 等属性。
[{ "Axis": "Profile", "P1": 12, "P2": "", "P3": "", "P4": "", "P5": "", "P6": "", "P7": "", "P8": "", "P9": "", "P10": "", "year": ""}, {"Axis": "Symmetry", "P1": 23, "P2": "", "P3": "", "P4": "", "P5": "", "P6": "", "P7": "", "P8": "", "P9": "", "P10": "", "year": ""}, {"Axis": "Error (%)", "P1": "", "P2": 1, "P3": "", "P4": "", "P5": "", "P6": "", "P7": "", "P8": "", "P9": "", "P10": "", "year": "" }, {"Axis": "Velocity (m/s)", "P1": 22, "P2": "", "P3": "", "P4": "", "P5": "", "P6": "", "P7": "", "P8": "", "P9": "", "P10": "", "year": "" } ]
我们的核心校验规则是:对于任意一个属性(例如 P1、P2 等),如果该属性在数组中的任一对象(行)上有非空值,那么它在所有对象(行)上都必须有非空值。反之,如果该属性在任一对象上为空值,那么它在所有对象上都必须为空值。 简而言之,对于任何一个 P_n 列,其所有值必须要么全部填充,要么全部为空,不允许混用。
立即学习“Java 免费学习笔记(深入)”;
例如,在上述数据中:
- P1 列:Profile 有值,Symmetry 有值,Error (%)为空,Velocity (m/s)有值。这违反了规则,因为 Error (%)为空而其他有值。
- P2 列:Profile 为空,Symmetry 为空,Error (%)有值,Velocity (m/s)为空。这也违反了规则。
传统循环方法的局限性
面对这种校验需求,开发者可能会倾向于使用多层嵌套循环来遍历数据并检查每个条件。然而,这种方法往往导致代码冗长、难以理解和维护,尤其当属性数量(如 P1 到 P10)增多时。原始问题中展示的尝试就是一个例子,它试图通过多个 if 条件和布尔标志来跟踪每个 Axis 和 P_n 的状态,这不仅复杂,而且未能正确实现“列一致性”的逻辑。
采用现代 javascript 数组方法实现列值一致性校验
为了更优雅、高效地解决这个问题,我们可以利用 JavaScript 的 Object.keys()、map()和 every()等高阶函数。这种方法将校验逻辑抽象化,使其更具通用性和可读性。
核心思路
- 识别所有待校验的“列”: 通过获取数组中第一个对象的键(属性名),我们可以得到所有需要检查的“列”。
- 提取每一“列”的数据: 对于每个识别出的键,遍历整个对象数组,提取出该键对应的所有值,形成一个“列”的数组。
- 校验“列”内数据的一致性: 对于每个“列”的数组,检查其所有元素的“真值性”(truthiness)是否与该列第一个元素的真值性保持一致。
示例代码与解析
假设我们的数据存储在一个名为 calibrationDataArray 的变量中。
const calibrationDataArray = [{ Axis: "Profile", P1: 12, P2: "", P3: "", P4: "", P5: "", P6: "", P7: "", P8: "", P9: "", P10: "", year: ""}, {Axis: "Symmetry", P1: 23, P2: "", P3: "", P4: "", P5: "", P6: "", P7: "", P8: "", P9: "", P10: "", year: ""}, {Axis: "Error (%)", P1: "", P2: 1, P3: "", P4: "", P5: "", P6: "", P7: "", P8: "", P9: "", P10: "", year: "" }, {Axis: "Velocity (m/s)", P1: 22, P2: "", P3: "", P4: "", P5: "", P6: "", P7: "", P8: "", P9: "", P10: "", year: "" }, ]; /** * 校验对象数组中各列数据的一致性。* 若某列任一元素有值,则该列所有元素必须有值;反之亦然。* @param {Array<Object>} data - 待校验的对象数组。* @returns {{isValid: boolean, errors: Array<string>}} - 校验结果。*/ function validateColumnConsistency(data) {if (!data || data.length === 0) {console.warn(" 校验数据为空或无效。"); return {isValid: true, errors: [] }; // 认为空数组是合法的,或者根据业务需求返回 false } // 获取第一个对象的键,作为所有待校验的“列”