前言
如何判断一个对象为空是我们在开发中经常会遇到的问题,今天我们来聊聊几种经常使用的方法,以及在不同的场景下我们如何去使用。
1. JSON.stringify
JSON.stringify
方法可以使对象序列化,转为相应的 JSON 格式。
1const obj = {};
2
3console.log(JSON.stringify(obj) === '{}')
缺点:如果存在 undefined
、任意的函数以及 symbol 值,在序列化过程中会被忽略(出现在非数组对象的属性值中时)或者被转换成 null
(出现在数组中时)。更多
如下示例:
1const obj = {
2 a: undefined,
3 b: function() {},
4 c: Symbol()
5}
6
7console.log(JSON.stringify(obj) === '{}')
2. for in 配合 hasOwnProperty
使用 for in
对当前对象进行遍历:
1const obj = {}
2Object.prototype.a = 1
3
4function isEmptyObj(obj) {
5 let flag = true
6 for (let o in obj) {
7 flag = false
8 break
9 }
10 return flag
11}
12
13console.log(isEmptyObj(obj))
由于 for in
在进行对象遍历时,会遍历对象原型上的属性,而我们只希望得到其自身的属性,这时可以使用 hasOwnProperty
来实现,如下:
1const obj = {}
2Object.prototype.a = 1
3
4function isEmptyObj(obj) {
5 let flag = true
6 for (let o in obj) {
7 if (obj.hasOwnProperty(o)) {
8 flag = false
9 break
10 }
11 }
12 return flag
13}
14
15console.log(isEmptyObj(obj))
缺点:for in
不能遍历不可枚举的属性。
3. Object.keys
Object.keys
会返回对象自身可枚举属性组成的数组,而不会遍历原型上的属性。
1const obj = {}
2Object.prototype.a = 1
3
4console.log(Object.keys(obj).length === 0)
缺点:Object.keys
和 for in
都只能遍历可枚举属性,不能遍历不可枚举的属性。
我们使用 Object.defineProperty
将属性 enumerable
设置为 false
来进行测试,示例如下:
1const obj = {}
2Object.defineProperty(obj, 'a', {
3 value: 1,
4 enumerable: false
5})
6
7console.log(obj.a)
8console.log(isEmptyObj(obj))
9console.log(Object.keys(obj).length === 0)
4. Object.getOwnPropertyNames
使用 Object.getOwnPropertyNames
可以得到对象自身的所有属性名组成的数组(包括不可枚举属性)。
1const obj = {}
2Object.defineProperty(obj, 'a', {
3 value: 1,
4 enumerable: false
5})
6
7console.log(Object.getOwnPropertyNames(obj))
缺点:不能获取 Symbol
值作为名称的属性,以上的 JSON.stringify
、for in
以及 Object.keys
方法也不能获取Symbol
值作为名称的属性,示例如下:
1const a = Symbol()
2const obj = {
3 [a]: 1
4}
5
6console.log(obj)
7console.log(Object.getOwnPropertyNames(obj).length === 0)
8console.log(JSON.stringify(obj) === '{}')
9console.log(isEmptyObj(obj))
10console.log(Object.keys(obj).length === 0)
5. Object.getOwnPropertyNames 结合 Object.getOwnPropertySymbols
已知 Object.getOwnPropertyNames
唯一的缺点是不能获取 Symbol
值作为名称的属性,而 Object.getOwnPropertySymbols
只能获取由 Symbol
值作为名称的属性,两者相结合是不是就可以完美解决了。我们来简单测试一下:
1const a = Symbol()
2const obj1 = {
3 [a]: 1
4}
5const obj2 = {b: 2}
6const obj3 = {}
7Object.defineProperty(obj3, 'a', {
8 value: 1,
9 enumerable: false
10})
11const obj4 = {}
12
13function getLength(obj) {
14 return Object.getOwnPropertyNames(obj).concat(Object.getOwnPropertySymbols(obj)).length
15}
16
17console.log(getLength(obj1) === 0)
18console.log(getLength(obj2) === 0)
19console.log(getLength(obj3) === 0)
20console.log(getLength(obj4) === 0)
经过测试,上面这种方法的确可以解决,但是比较繁琐,那有没有更好的方法呢?答案是有的。
6. Reflect.ownKeys
Reflect.ownKeys
方法返回一个由目标对象自身的属性组成的数组,它的返回值等同于 Object.getOwnPropertyNames(target).concat(Object.getOwnPropertySymbols(target))
,示例如下:
1const a = Symbol()
2const obj1 = {
3 [a]: 1
4}
5const obj2 = {b: 2}
6const obj3 = {}
7Object.defineProperty(obj3, 'a', {
8 value: 1,
9 enumerable: false
10})
11const obj4 = {}
12
13console.log(Reflect.ownKeys(obj1).length === 0)
14console.log(Reflect.ownKeys(obj2).length === 0)
15console.log(Reflect.ownKeys(obj3).length === 0)
16console.log(Reflect.ownKeys(obj4).length === 0)
总结
判断一个对象是否为空时,使用 Reflect.ownKeys
方法最为完美。