优秀的编程知识分享平台

网站首页 > 技术文章 正文

JS对象判空的几种方式,你真的会了吗?

nanyue 2025-06-10 17:03:26 技术文章 3 ℃

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

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

if(isEmpty(userInfo)){
// 跳转登录页
}

四种主流检测方案对比

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

function isEmpty(obj) {
return Object.keys(obj).length === 0;
}

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

致命缺陷

  • 漏检不可枚举属性
  • 无视Symbol类型键值
  • 对特殊对象处理不当
function isEmpty(obj) {
return JSON.stringify(obj) === '{}';
}

三大隐患

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

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

function isStrictEmpty(obj) {
return Reflect.ownKeys(obj).length === 0;
}

核心优势

  • 捕获所有类型键值(含Symbol)
  • 检测不可枚举属性

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

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

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. 数组误判问题
isEmpty([]) // 返回true,但空数组≠空对象!

2.特殊对象陷阱

isEmpty(new Date()) // 多数方案返回true

3.循环引用崩溃

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

最佳实践推荐

// 通用场景检测
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;
}
最近发表
标签列表