
本教程详细阐述了如何在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 } // 获取第一个对象的键,作为所有待校验的“列”