上一篇文章 「前端面试题系列8」数组去重(10 种浓缩版) 的最后,简单介绍了 lodash 中的数组去重方法 _.uniq
,它能够实现咱们平常工做中的去重需求,可以去重 NaN
,并保留 {...}
。前端
今天要讲的,是我从 _.uniq 的源码实现文件 baseUniq.js 中学到的几个很基础,却又容易被忽略的知识点。面试
让咱们先从三个功能相近的 API 讲起,他们分别是:_.uniq
、_.uniqBy
、_.uniqWith
。它们三个背后的实现文件,都指向了 .internal 下的 baseUniq.js。编程
区别在于 _.uniq 只需传入一个源数组 array, _.uniqBy 相较于 _.uniq 要多传一个迭代器 iteratee,而 _.uniqWith 要多传一个比较器 comparator。iteratee
和 comparator
的用法,会在后面说到。数组
以 _.uniqWith 为例,它是这样调用 _.baseUniq 的:bash
function uniqWith(array, comparator) {
comparator = typeof comparator == 'function' ? comparator : undefined
return (array != null && array.length)
? baseUniq(array, undefined, comparator)
: []
}
复制代码
baseUniq 的源码并很少,但比较绕。先贴一下的源码。app
const LARGE_ARRAY_SIZE = 200
function baseUniq(array, iteratee, comparator) {
let index = -1
let includes = arrayIncludes
let isCommon = true
const { length } = array
const result = []
let seen = result
if (comparator) {
isCommon = false
includes = arrayIncludesWith
}
else if (length >= LARGE_ARRAY_SIZE) {
const set = iteratee ? null : createSet(array)
if (set) {
return setToArray(set)
}
isCommon = false
includes = cacheHas
seen = new SetCache
}
else {
seen = iteratee ? [] : result
}
outer:
while (++index < length) {
let value = array[index]
const computed = iteratee ? iteratee(value) : value
value = (comparator || value !== 0) ? value : 0
if (isCommon && computed === computed) {
let seenIndex = seen.length
while (seenIndex--) {
if (seen[seenIndex] === computed) {
continue outer
}
}
if (iteratee) {
seen.push(computed)
}
result.push(value)
}
else if (!includes(seen, computed, comparator)) {
if (seen !== result) {
seen.push(computed)
}
result.push(value)
}
}
return result
}
复制代码
为了兼容刚才说的三个 API,就产生了很多的干扰项。若是先从 _.uniq 入手,去掉 iteratee 和 comparator 的干扰,就会清晰很多。ide
function baseUniq(array) {
let index = -1
const { length } = array
const result = []
if (length >= 200) {
const set = createSet(array)
return setToArray(set)
}
outer:
while (++index < length) {
const value = array[index]
if (value === value) {
let resultIndex = result.length
while (resultIndex--) {
if (result[resultIndex] === value) {
continue outer
}
}
result.push(value)
} else if (!includes(seen, value)) {
result.push(value)
}
}
return result
}
复制代码
这里有 2 个知识点。函数式编程
在源码中有一个判断 value === value
,乍一看,会以为这是句废话!?!但其实,这是为了过滤 NaN 的状况。函数
MDN 中对 NaN 的解释是:它是一个全局对象的属性,初始值就是 NaN。它一般都是在计算失败时,做为 Math 的某个方法的返回值出现的。post
判断一个值是不是 NaN,必须使用 Number.isNaN() 或 isNaN(),在执行自比较之中:NaN,也只有 NaN,比较之中不等于它本身。
NaN === NaN; // false
Number.NaN === NaN; // false
isNaN(NaN); // true
isNaN(Number.NaN); // true
复制代码
因此,在源码中,当遇到 NaN
的状况时,baseUniq 会转而去执行 !includes(seen, value)
的判断,去处理 NaN 。
在源码的主体部分,while 语句以前,有一行 outer:
,它是干什么用的呢? while 中还有一个 while 的内部,有一行 continue outer
,从语义上理解,好像是继续执行 outer
,这又是种什么写法呢?
outer:
while (++index < length) {
...
while (resultIndex--) {
if (result[resultIndex] === value) {
continue outer
}
}
}
复制代码
咱们都知道 Javascript 中,经常使用到冒号的地方有三处,分别是:A ? B : C 三元操做符、switch case 语句中、对象的键值对组成。
但其实还有一种并不常见的特殊做用:标签语句
。在 Javascript 中,任何语句均可以经过在它前面加上标志符和冒号来标记(identifier: statement
),这样就能够在任何地方使用该标记,最经常使用于循环语句中。
因此,在源码中,outer 只是看着有点不习惯,多看两遍就行了,语义上仍是很好理解的。
_.uniqBy 可根据指定的 key 给一个对象数组去重,一个官网的例子以下:
// The `_.property` iteratee shorthand.
_.uniqBy([{ 'x': 1 }, { 'x': 2 }, { 'x': 1 }], 'x');
// => [{ 'x': 1 }, { 'x': 2 }]
复制代码
这里的 'x'
是 _.property('x')
的缩写,它指的就是 iteratee。
从给出的例子和语义上看,还挺好理解的。可是为何 _.property 就能实现对象数组的去重了呢?它又是如何实现的呢?
@param {Array|string} path The path of the property to get.
@returns {Function} Returns the new accessor function.
function property(path) {
return isKey(path) ? baseProperty(toKey(path)) : basePropertyDeep(path)
}
复制代码
从注释看,property 方法会返回一个 Function
,再看 baseProperty 的实现:
@param {string} key The key of the property to get.
@returns {Function} Returns the new accessor function.
function baseProperty(key) {
return (object) => object == null ? undefined : object[key]
}
复制代码
咦?怎么返回的仍是个 Function
?感受它什么也没干呀,那个参数 object
又是哪里来的?
纯函数,是函数式编程中的概念,它表明这样一类函数:对于指定输出,返回指定的结果。不存在反作用。
// 这是一个简单的纯函数
const addByOne = x => x + 1;
复制代码
也就是说,纯函数的返回值只依赖其参数,函数体内不能存在任何反作用。若是是一样的参数,则必定能获得一致的返回结果。
function baseProperty(key) {
return (object) => object == null ? undefined : object[key]
}
复制代码
baseProperty 返回的就是一个纯函数,在符合条件的状况下,输出 object[key]
。在函数式编程中,函数是“一等公民”,它能够只是根据参数,作简单的组合操做,再做为别的函数的返回值。
因此,在源码中,object 是调用 baseProperty 时传入的对象。 baseProperty 的做用,是返回指望结果为 object[key] 的函数。
仍是先从官网的小例子提及,它会彻底地给对象中全部的键值对,进行比较。
var objects = [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }, { 'x': 1, 'y': 2 }];
_.uniqWith(objects, _.isEqual);
// => [{ 'x': 1, 'y': 2 }, { 'x': 2, 'y': 1 }]
复制代码
而在 baseUniq 的源码中,能够看到最终的实现,须要依赖 arrayIncludesWith 方法,如下是它的源码:
function arrayIncludesWith(array, target, comparator) {
if (array == null) {
return false
}
for (const value of array) {
if (comparator(target, value)) {
return true
}
}
return false
}
复制代码
arrayIncludesWith 没什么复杂的。comparator 做为一个参数传入,将 target
和 array
的每一个 value 进行处理。从官网的例子看,_.isEqual 就是 comparator,就是要比较它们是否相等。
接着就追溯到了 _.isEqual 的源码,它的实现文件是 baseIsEqualDeep.js。在里面看到一个让我犯迷糊的写法,这是一个判断。
/** Used to check objects for own properties. */
const hasOwnProperty = Object.prototype.hasOwnProperty
...
const objIsWrapped = objIsObj && hasOwnProperty.call(object, '__wrapped__')
复制代码
hasOwnProperty ?call, 'wrapped' ?
再次查找到了 MDN 的解释:全部继承了 Object 的对象都会继承到 hasOwnProperty 方法。它能够用来检测一个对象是否含有特定的自身属性;会忽略掉那些从原型链上继承到的属性。
o = new Object();
o.prop = 'exists';
o.hasOwnProperty('prop'); // 返回 true
o.hasOwnProperty('toString'); // 返回 false
o.hasOwnProperty('hasOwnProperty'); // 返回 false
复制代码
call 的用法能够参考这篇 细说 call、apply 以及 bind 的区别和用法。
那么 hasOwnProperty.call(object, '__wrapped__')
的意思就是,判断 object 这个对象上是否存在 'wrapped' 这个自身属性。
wrapped 是什么属性?这就要说到 lodash 的延迟计算方法 _.chain,它是一种函数式风格,从名字就能够看出,它实现的是一种链式的写法。好比下面这个例子:
var names = _.chain(users)
.map(function(user){
return user.user;
})
.join(" , ")
.value();
复制代码
若是你没有显样的调用value方法,使其当即执行的话,将会获得以下的LodashWrapper延迟表达式:
LodashWrapper {__wrapped__: LazyWrapper, __actions__: Array[1], __chain__: true, constructor: function, after: function…}
复制代码
由于延迟表达式的存在,所以咱们能够屡次增长方法链,但这并不会被执行,因此不会存在性能的问题,最后直到咱们须要使用的时候,使用 value()
显式当即执行便可。
因此,在 baseIsEqualDeep 源码中,才须要作 hasOwnProperty 的判断,而后在须要的状况下,执行 object.value()
。
阅读源码,在一开始会比较困难,由于会遇到一些看不明白的写法。就像一开始我卡在了 value === value 的写法,不明白它的用意。一旦知道了是为了过滤 NaN 用的,那后面就会通畅不少了。
因此,阅读源码,是一种很棒的重温基础知识的方式。遇到看不明白的点,不要放过,多查多问多看,才能不断地夯实基础,读懂更多的源码思想,体会更多的原生精髓。若是我在一开始看到 value === value 时就放弃了,那或许就不会有今天的这篇文章了。
PS:欢迎关注个人公众号 “超哥前端小栈”,交流更多的想法与技术。