JavaScript 空对象检测

📌 前言:为什么空对象检测如此重要?

在开发中我们经常会遇到这样的场景:

js 复制代码
if(isEmpty(userInfo)){
  // 跳转登录页
}

🛠️ 四种主流检测方案对比

方案一:Object.keys() 基础版

js 复制代码
function isEmpty(obj) {
  return Object.keys(obj).length === 0;
}

适用场景:普通数据对象检测

致命缺陷

  • 漏检不可枚举属性
  • 无视Symbol类型键值
  • 对特殊对象处理不当

方案二:JSON.stringify() 取巧法

js 复制代码
function isEmpty(obj) {
  return JSON.stringify(obj) === '{}';
}

⚠️ 三大隐患

  1. 过滤undefined和函数属性
  2. 无法处理循环引用
  3. Date/RegExp等对象误判

方案三:Reflect.ownKeys() 终极方案

js 复制代码
function isStrictEmpty(obj) {
  return Reflect.ownKeys(obj).length === 0;
}

💡 核心优势

  • 捕获所有类型键值(含Symbol)
  • 检测不可枚举属性
  • 精准识别特殊对象

方案四:for...in 循环法(不推荐)

js 复制代码
function isEmptyLoop(obj) {
  for(let key in obj){
    if(obj.hasOwnProperty(key)) return false;
  }
  return true;
}

🚨 重大缺陷

  • 可能误判原型链属性
  • 性能最差(平均慢3倍)

方法对比表格

方法 Symbol 键 不可枚举属性 原型链属性 特殊对象处理
Object.keys()
JSON.stringify()
Reflect.ownKeys()
for...in + 检查

💻 核心API深度对比:Object.keys vs Reflect.ownKeys

js 复制代码
const secretKey = Symbol('SECRET');
const obj = {
  [secretKey]: '绝密数据',
  2: '数字键',
  'name': '张三'
};
Object.defineProperty(obj, 'hiddenProp', {
  value: '隐藏属性',
  enumerable: false
});

console.log('Object.keys:', Object.keys(obj)); 
// 输出: ['2', 'name']

console.log('Reflect.ownKeys:', Reflect.ownKeys(obj));
// 输出: ['2', 'name', 'hiddenProp', Symbol(SECRET)]

Object.keys 与 Reflect.ownKeys 的核心区别

示例说明

🚀 性能实测数据

通过Benchmark.js测试10万次操作:

🛑 常见误区警示

  1. 数组误判问题

    js 复制代码
    isEmpty([]) // 返回true,但空数组≠空对象!
  2. 特殊对象陷阱

    js 复制代码
    isEmpty(new Date()) // 多数方案返回true
  3. 循环引用崩溃

    js 复制代码
    const obj = { self: null };
    obj.self = obj;
    JSON.stringify(obj); // 抛出异常

🔧 最佳实践推荐

js 复制代码
// 通用场景检测
const safeIsEmpty = obj => {
  return Object.prototype.toString.call(obj) === '[object Object]' 
    && Object.keys(obj).length === 0;
}

// 严格模式检测
const strictIsEmpty = obj => {
  return Reflect.ownKeys(obj).length === 0 
    && Object.getPrototypeOf(obj) === Object.prototype;
}
相关推荐
hedley(●'◡'●)2 分钟前
基于cesium和vue的大疆司空模仿程序
前端·javascript·vue.js·python·typescript·无人机
百思可瑞教育4 分钟前
构建自己的Vue UI组件库:从设计到发布
前端·javascript·vue.js·ui·百思可瑞教育·北京百思教育
CappuccinoRose30 分钟前
JavaScript 学习文档(二)
前端·javascript·学习·数据类型·运算符·箭头函数·变量声明
全栈前端老曹1 小时前
【MongoDB】深入研究副本集与高可用性——Replica Set 架构、故障转移、读写分离
前端·javascript·数据库·mongodb·架构·nosql·副本集
NCDS程序员1 小时前
v-model: /v-model/ :(v-bind)三者核心区别
前端·javascript·vue.js
小杨同学呀呀呀呀2 小时前
Ant Design Vue <a-timeline>时间轴组件失效解决方案
前端·javascript·vue.js·typescript·anti-design-vue
qq_532453532 小时前
使用 Three.js 构建沉浸式全景图AR
开发语言·javascript·ar
Mr Xu_10 小时前
告别冗长 switch-case:Vue 项目中基于映射表的优雅路由数据匹配方案
前端·javascript·vue.js
前端摸鱼匠10 小时前
Vue 3 的toRefs保持响应性:讲解toRefs在解构响应式对象时的作用
前端·javascript·vue.js·前端框架·ecmascript
sleeppingfrog11 小时前
zebra通过zpl语言实现中文打印(二)
javascript