bug记录:由运算符优先级导致的bug

记录:由运算符优先级导致的bug

错误代码以及问题现象

组件日期格式始终显示为 YYYY-MM-DD HH:mm:ss,即使已显式传入 format 参数

jsx 复制代码
<DatePicker 
  format={format ?? showTime ? 'YYYY-MM-DD HH:mm:ss' : 'YYYY-MM-DD'}
/>

原因分析

运算符优先级混淆

JavaScript 运算符优先级表(部分):

运算符 优先级 结合性
?? (空值合并) 5 从左到右
?: (三元运算) 4 从右到左

实际执行逻辑:

js 复制代码
// 等效于
(format ?? showTime) ? '...HH:mm:ss' : '...YYYY-MM-DD'

format 存在值时:

  1. ?? 先执行返回非空值
  2. 将非空值转换为布尔值进行三元运算
  3. 任何真值都会触发选择时间格式

解决方案

正确写法

jsx 复制代码
// ✅ 正确使用括号明确优先级
<DatePicker
  format={format ?? (showTime ? 'YYYY-MM-DD HH:mm:ss' : 'YYYY-MM-DD')}
/>

执行逻辑分解

  1. 优先处理三元表达式:

    js 复制代码
    const defaultFormat = showTime ? '...HH:mm:ss' : '...YYYY-MM-DD';
  2. 执行空值合并运算:

    js 复制代码
    finalFormat = format ?? defaultFormat;

验证示例

测试用例 原代码结果 修正后结果
format="YYYY/MM/DD" HH:mm:ss 格式 YYYY/MM/DD
format=null, showTime=true HH:mm:ss 格式 HH:mm:ss 格式
format=undefined, showTime=false HH:mm:ss 格式 YYYY-MM-DD

最佳实践

  1. 括号优先原则:混合使用逻辑运算符时显式使用括号

  2. 分步处理 :对复杂逻辑进行分解

    jsx 复制代码
    const getDefaultFormat = () => 
      showTime ? 'YYYY-MM-DD HH:mm:ss' : 'YYYY-MM-DD';
    
    <DatePicker format={format ?? getDefaultFormat()} />
  3. 类型守卫 :严格判断空值

    js 复制代码
    const finalFormat = typeof format === 'string' ? format : defaultFormat;

关键点:当 ???: 混合使用时,永远显式使用括号明确优先级关系。

相关推荐
@大迁世界5 分钟前
TypeScript 的本质并非类型,而是信任
开发语言·前端·javascript·typescript·ecmascript
GIS之路14 分钟前
GDAL 实现矢量裁剪
前端·python·信息可视化
是一个Bug18 分钟前
后端开发者视角的前端开发面试题清单(50道)
前端
Amumu1213819 分钟前
React面向组件编程
开发语言·前端·javascript
持续升级打怪中41 分钟前
Vue3 中虚拟滚动与分页加载的实现原理与实践
前端·性能优化
GIS之路1 小时前
GDAL 实现矢量合并
前端
hxjhnct1 小时前
React useContext的缺陷
前端·react.js·前端框架
冰暮流星1 小时前
javascript逻辑运算符
开发语言·javascript·ecmascript
前端 贾公子1 小时前
从入门到实践:前端 Monorepo 工程化实战(4)
前端
菩提小狗1 小时前
Sqlmap双击运行脚本,双击直接打开。
前端·笔记·安全·web安全