`??` 和 `||` 搞混,线上用户头像全挂了

问题场景

上周五快下班,客服群突然炸了:

"好多用户头像显示空白!" "不是全部,大概 30% 的账号头像没了"

查前端日志,头像接口返回正常,url 字段有值。再看渲染层代码:

js 复制代码
const avatarUrl = user.avatar?.url || '/default-avatar.png'

逻辑看似没毛病:有头像用头像,没有就用默认图。但诡异的是,有头像的用户也显示了默认图

原因分析

问题出在 || 运算符的"激进"行为上。

|| 会把所有假值 (falsy) 都当成"没有"来对待。JavaScript 中的假值包括:

类型
'' 空字符串
0 数字零
false 布尔假
null
undefined 未定义
NaN 非数字

排查后发现,这批头像异常的用户的 url 值刚好是 空字符串 '' --- 头像服务在某些情况下会返回 url: ""(代表用户上传过头像但处理失败)。

js 复制代码
// 实际数据
user.avatar = { url: '' }  // 空字符串,但 ≠ 没头像

// 执行逻辑
const avatarUrl = '' || '/default-avatar.png'
// → 空字符串是假值,被 || 吞掉,返回默认图 ❌

用户明明上传过头像(数据库有记录),只是因为图片处理失败返回了空串,就被 || 强制替换成默认图------这就成了"有头像但显示默认"的 bug。

解决方案

方案一:改用 ??(空值合并运算符)

js 复制代码
const avatarUrl = user.avatar?.url ?? '/default-avatar.png'

?? 只会在左操作数为 nullundefined 时才取右侧值。''0false 都会原样保留。

注意: ?? 不能与 &&|| 混用不括弧:

js 复制代码
// ❌ 语法错误
const a = b ?? c || d

// ✅ 正确写法
const a = (b ?? c) || d

方案二:显式判断

js 复制代码
const avatarUrl = user.avatar?.url && user.avatar.url.length > 0
  ? user.avatar.url
  : '/default-avatar.png'

更保险,但啰嗦。

方案三:后端统一语义

后端对 url 字段做约束:没有头像 → 不返回字段或返回 null;有头像(包括处理失败)都正确定义。前端只信任 ??

深入:还有哪些地方容易踩?

场景 1:分数/数值 0

js 复制代码
// 用户完成了 0 道题
const count = data.completedCount || '无数据'
// 0 被 || 吞掉,显示"无数据" ❌
const count = data.completedCount ?? '无数据'  // ✅ 显示 0

场景 2:CSS class 拼接

js 复制代码
// 某个状态的 className 可能是空串 ''
const cls = statusClass || 'default-status'
// 空串时被覆盖 ❌
const cls = statusClass ?? 'default-status'  // ✅

场景 3:函数默认参数

js 复制代码
function setTheme(color = '#333') { ... }

setTheme('')   // '' 是假值吗?不,函数默认参数只对 undefined 生效
// → 传入 '' 不会触发默认值,color = ''

函数默认参数的行为类似 ??,只对 undefined 生效。很多人误以为它和 || 一样。

要点总结

| 运算符 | 触发替换的条件 | 适用场景 |
|--------|----------------------|-----------------|------|------------------|
| ?? | null / undefined | 后端可能缺失的字段、可选链取值 |
| ` | | ` | 所有假值 | 兜底默认值(明确想过滤空串/0) |
| 函数默认参数 | 仅 undefined | 参数缺省传值 |

一句话黄金法则:

当你要表达"没有这个数据就用默认值"时,99% 的场景应该用 ??; 只有 1% 的场景(如用户输入为空时填默认文案)才刻意用 ||

修完这个 bug 后的复盘会上,全组把 eslint 加了一条规则:

json 复制代码
// .eslintrc
{
  "rules": {
    "no-unneeded-ternary": "error",
    "@typescript-eslint/prefer-nullish-coalescing": "error"
  }
}

并在代码 review checklist 里写死了一条:"见到 || 接变量,先想想是不是应该用 ??"

从此,头像空白 bug 再没出现过。

相关推荐
kyriewen2 小时前
白宫前脚下了限制令,OpenAI 后脚就把 GPT-5.6 发了
前端·gpt·openai
用户40269244819083 小时前
CRMEB Pro 新增后台接口全链路:路由、权限、验证器、返回格式一次讲清
前端·后端
泉城老铁3 小时前
springboot+vue+ ffmpeg 实现视频的拉流播放
前端
PedroQue994 小时前
uni-router v1.8.0新增冷启动守卫补执行
前端·uni-app
xiaok4 小时前
部署之后,本地浏览器还在读取旧缓存导致页面一直显示loading中
前端
用户059540174464 小时前
Redis缓存一致性踩坑实录:线上故障排查6小时,我用pytest+内存快照把它永久关进了笼子
前端·css
星栈4 小时前
我用 Rust + Dioxus 做了个全栈跨平台笔记应用:第一版先把列表和详情跑通
前端·rust·前端框架
用户1733598075374 小时前
Vue 3 SPA 首屏优化:从 3s 到 1.2s 的 5 个实践
前端·vue.js
咖啡无伴侣4 小时前
基础骨架:30 分钟搭好 pnpm workspace,完成双项目 Monorepo 迁入
前端