Lodash源码阅读-isEmpty

功能概述

能判断各种各样的数据类型是不是"空的"。不管你给它数组、对象、Map、Set,还是字符串,它都能准确告诉你是不是空的。而且实现方式很巧妙,针对不同类型的数据采用不同的判断策略。

源码实现

js 复制代码
function isEmpty(value) {
  if (value == null) {
    return true;
  }
  if (
    isArrayLike(value) &&
    (isArray(value) ||
      typeof value == "string" ||
      typeof value.splice == "function" ||
      isBuffer(value) ||
      isTypedArray(value) ||
      isArguments(value))
  ) {
    return !value.length;
  }
  var tag = getTag(value);
  if (tag == mapTag || tag == setTag) {
    return !value.size;
  }
  if (isPrototype(value)) {
    return !baseKeys(value).length;
  }
  for (var key in value) {
    if (hasOwnProperty.call(value, key)) {
      return false;
    }
  }
  return true;
}

实现原理解析

整体思路概述

这个函数的实现思路非常清晰,就像走一个决策树:

  1. 先处理最简单的 null/undefined 情况
  2. 然后判断是不是数组类型的(包括数组、字符串、类数组等)
  3. 接着看看是不是 Map/Set 类型
  4. 再检查是不是原型对象
  5. 最后处理普通对象的情况

每一步都针对特定类型采用最合适的判空方式,下面我们来详细看看每种情况。

1. null/undefined 判断

js 复制代码
if (value == null) {
  return true;
}

这是最基础的判断,用双等号(==)来同时处理 null 和 undefined:

js 复制代码
console.log(isEmpty(null)); // true
console.log(isEmpty(undefined)); // true

// 但其他假值不一定是空的
console.log(isEmpty(0)); // false
console.log(isEmpty(false)); // false

2. 数组类型判断

js 复制代码
if (
  isArrayLike(value) &&
  (isArray(value) ||
    typeof value == "string" ||
    typeof value.splice == "function" ||
    isBuffer(value) ||
    isTypedArray(value) ||
    isArguments(value))
) {
  return !value.length;
}

这段代码处理所有有 length 属性的类型,包括:

js 复制代码
// 数组
console.log(isEmpty([])); // true
console.log(isEmpty([1, 2])); // false

// 字符串
console.log(isEmpty("")); // true
console.log(isEmpty("hello")); // false

// 类数组对象
function foo() {
  console.log(isEmpty(arguments)); // true
}
foo();

// 类型化数组
console.log(isEmpty(new Int8Array())); // true
console.log(isEmpty(new Int8Array(2))); // false

3. Map/Set 类型判断

js 复制代码
var tag = getTag(value);
if (tag == mapTag || tag == setTag) {
  return !value.size;
}

对于 Map 和 Set,我们看它们的 size 属性:

js 复制代码
// Map
const map = new Map();
console.log(isEmpty(map)); // true
map.set("key", "value");
console.log(isEmpty(map)); // false

// Set
const set = new Set();
console.log(isEmpty(set)); // true
set.add(1);
console.log(isEmpty(set)); // false

4. 原型对象判断

js 复制代码
if (isPrototype(value)) {
  return !baseKeys(value).length;
}

处理原型对象的情况:

js 复制代码
// 创建一个构造函数
function Person() {}
Person.prototype.name = "John";

// 检查原型对象
console.log(isEmpty(Person.prototype)); // false

// 空的原型对象
function Empty() {}
console.log(isEmpty(Empty.prototype)); // true

5. 普通对象判断

js 复制代码
for (var key in value) {
  if (hasOwnProperty.call(value, key)) {
    return false;
  }
}
return true;

最后处理普通对象,看看有没有自己的可枚举属性:

js 复制代码
// 空对象
console.log(isEmpty({})); // true

// 有属性的对象
console.log(isEmpty({ name: "John" })); // false

// 只有不可枚举属性的对象
const obj = Object.defineProperty({}, "name", {
  value: "John",
  enumerable: false,
});
console.log(isEmpty(obj)); // true

总结

通过一系列的条件判断,优雅地处理了各种不同类型的空值判断。不同类型用不同的判空标准:length、size、自有属性等。而且代码结构清晰,判断逻辑层层递进。

相关推荐
wuhen_n4 小时前
JavaScript内置数据结构
开发语言·前端·javascript·数据结构
大鱼前端4 小时前
为什么我说CSS-in-JS是前端“最佳”的糟粕设计?
前端
不爱吃糖的程序媛4 小时前
Capacitor:跨平台Web原生应用开发利器,现已全面适配鸿蒙
前端·华为·harmonyos
AC赳赳老秦4 小时前
2026国产算力新周期:DeepSeek实战适配英伟达H200,引领大模型训练效率跃升
大数据·前端·人工智能·算法·tidb·memcache·deepseek
CHU7290354 小时前
淘宝扭蛋机抽盒小程序前端功能解析:解锁趣味抽盒新体验
前端·小程序
-凌凌漆-4 小时前
【npm】npm的-D选项介绍
前端·npm·node.js
鹿心肺语5 小时前
前端HTML转PDF的两种主流方案深度解析
前端·javascript
海石5 小时前
去到比北方更北的地方—2025年终总结
前端·ai编程·年终总结
一个懒人懒人5 小时前
Promise async/await与fetch的概念
前端·javascript·html
Mintopia5 小时前
Web 安全与反编译源码下的权限设计:构筑前后端一致的防护体系
前端·安全