
当从外部输入源(如键盘事件监听器捕获的条码扫描数据)构建字符串时,我们可能会遇到一个看似矛盾的现象:string.length返回的值远大于我们预期中的可见字符数量,同时,基于该字符串的正则表达式匹配也总是失败。
例如,在一个处理条码扫描的React Native组件中,componentDidMount监听按键事件并拼接this.barcode:
componentDidMount() {
KeyEvent.onKeyDownListener((keyEvent: KeyEventProps) => {
if (keyEvent.keyCode === 61) { // Tab键作为结束符
const barcodeType = this.getBarcodeType(this.barcode);
if (barcodeType) {
this.props.barcodeSubject.next({barcode: this.barcode});
}
this.setState({barcode: this.barcode, barcodeType});
this.barcode = ''; // 重置条码
return;
}
this.barcode += keyEvent.pressedKey;
});
}
getBarcodeType(input: string): Format | undefined {
const ean13Pattern = /^[0-9]{13}$/gm;
// ... 其他正则表达式
console.log(
'***',
input,
typeof input,
input.length,
input.match(ean13Pattern),
);
if (ean13Pattern.test(input)) {
return 'EAN';
}
// ... 其他匹配逻辑
return undefined;
}当扫描一个13位的条码(如"5449000214911")时,console.log可能会输出以下结果:
LOG *** 5449000214911 string 26 null
这里,我们期望的长度是13,但实际输出却是26。更关键的是,input.match(ean13Pattern)返回null,表明正则表达式匹配失败。这明确指出字符串input中包含了额外的、不可见的字符。
为了进一步诊断这些不可见字符,我们可以将字符串展开成字符数组并用逗号连接打印:
console.log([...input].join(','));通过这种方式,通常可以发现字符串中夹杂着\x00、\x01等ASCII控制字符,这些字符虽然不可见,但会计入字符串的length,并可能干扰正则表达式的匹配。
解决此问题的核心在于识别并移除这些非打印字符。最有效的方法是使用正则表达式替换。我们可以针对ASCII控制字符范围进行匹配和替换。
ASCII字符集中的控制字符通常位于以下两个范围:
因此,我们可以构造一个正则表达式/[\x00-\x1F\x7F-\x9F]/g来匹配这些字符,并用空字符串替换它们。
const cleanedInput = input.replace(/[\x00-\x1F\x7F-\x9F]/g, '');
将字符串清理步骤集成到getBarcodeType函数或其他处理输入字符串的地方,确保在进行任何格式校验或逻辑处理之前,字符串已经是“干净”的。
以下是优化后的getBarcodeType函数示例:
getBarcodeType(input: string): Format | undefined {
// 1. 清理输入字符串,移除不可见字符
const cleanedInput = input.replace(/[\x00-\x1F\x7F-\x9F]/g, '');
// 2. 定义正则表达式
const ean13Pattern = /^[0-9]{13}$/; // 注意:移除了'g'和'm'标志,因为test方法通常不需要全局或多行匹配,且全局匹配可能影响连续调用
const ean8Pattern = /^[0-9]{8}$/;
const ean128Pattern = /^[0-9A-Za-z]+$/; // 修正ean128Pattern,原问题中括号内多余
// 调试清理后的字符串
console.log(
'*** Cleaned Input:',
cleanedInput,
typeof cleanedInput,
cleanedInput.length,
cleanedInput.match(ean13Pattern),
);
// 3. 进行格式匹配
if (ean13Pattern.test(cleanedInput)) {
return 'EAN';
} else if (ean8Pattern.test(cleanedInput)) {
return 'EAN';
} else if (ean128Pattern.test(cleanedInput)) {
return 'CODE128';
}
return undefined;
}注意事项:
在React Native等环境中处理来自外部硬件设备的字符串输入时,不可见控制字符是一个常见的陷阱。它们会导致字符串length属性失真,并使基于字符串内容的正则表达式匹配失败。通过在字符串处理的早期阶段,利用正则表达式/[\x00-\x1F\x7F-\x9F]/g移除这些非打印字符,我们可以有效地净化输入数据,确保后续的逻辑判断和格式校验能够准确无误地执行。这不仅提升了应用的健壮性,也避免了因数据异常导致的潜在业务逻辑错误。
以上就是React Native中字符串长度异常与不可见字符处理的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号