网站首页 > 技术文章 正文
前言:为什么空对象检测如此重要?
在开发中我们经常会遇到这样的场景:
if(isEmpty(userInfo)){
// 跳转登录页
}四种主流检测方案对比
方案一:Object.keys() 基础版
function isEmpty(obj) {
return Object.keys(obj).length === 0;
}适用场景:普通数据对象检测
致命缺陷:
- 漏检不可枚举属性
 - 无视Symbol类型键值
 - 对特殊对象处理不当
 
function isEmpty(obj) {
return JSON.stringify(obj) === '{}';
}
三大隐患:
- 过滤undefined和函数属性
 - 无法处理循环引用
 - 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万次操作:
常见误区警示
- 数组误判问题
 
isEmpty([]) // 返回true,但空数组≠空对象!2.特殊对象陷阱
isEmpty(new Date()) // 多数方案返回true3.循环引用崩溃
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;
}
    
猜你喜欢
- 2025-09-03 网页数据如何获取,带你走近JS逆向,完全入门级!
 - 2025-09-03 【JavaScript】手撕前端面试题:手写new操作符
 - 2025-06-10 告别JSON.stringify,一行代码搞定JavaScript深拷贝的正确姿势!
 - 2025-06-10 常见web安全问题,SQL注入、XSS、CSRF,基本原理以及如何防御
 - 2025-06-10 web前端之null和undefined区别(前端null与undefined区别)
 - 2025-06-10 Why does Google prepend while(1); to their JSON responses?
 - 2025-06-10 Map与Set:JavaScript的数据魔术师,让你的代码性能原地起飞!
 - 2025-06-10 JavaScript 数据类型详解:从基础到进阶,一篇全搞定!
 - 2024-07-30 JavaScript面向对象核心知识归纳(javascript面向对象编程)
 - 2024-07-30 前端教程:Javascript Boolean对象
 
- 最近发表
 - 
- 聊一下 gRPC 的 C++ 异步编程_grpc 异步流模式
 - [原创首发]安全日志管理中心实战(3)——开源NIDS之suricata部署
 - 超详细手把手搭建在ubuntu系统的FFmpeg环境
 - Nginx运维之路(Docker多段构建新版本并增加第三方模
 - 92.1K小星星,一款开源免费的远程桌面,让你告别付费远程控制!
 - Go 人脸识别教程_piwigo人脸识别
 - 安卓手机安装Termux——搭建移动服务器
 - ubuntu 安装开发环境(c/c++ 15)_ubuntu安装c++编译器
 - Rust开发环境搭建指南:从安装到镜像配置的零坑实践
 - Windows系统安装VirtualBox构造本地Linux开发环境
 
 
- 标签列表
 - 
- cmd/c (90)
 - c++中::是什么意思 (84)
 - 标签用于 (71)
 - 主键只能有一个吗 (77)
 - c#console.writeline不显示 (95)
 - pythoncase语句 (88)
 - es6includes (74)
 - sqlset (76)
 - apt-getinstall-y (100)
 - node_modules怎么生成 (87)
 - chromepost (71)
 - flexdirection (73)
 - c++int转char (80)
 - mysqlany_value (79)
 - static函数和普通函数 (84)
 - el-date-picker开始日期早于结束日期 (76)
 - js判断是否是json字符串 (75)
 - c语言min函数头文件 (77)
 - asynccallback (87)
 - localstorage.removeitem (77)
 - vector线程安全吗 (73)
 - java (73)
 - js数组插入 (83)
 - mac安装java (72)
 - 无效的列索引 (74)
 
 
