lodash源码分析之Hash缓存

在那小小的梦的暖阁,我为你收藏起整个季节的烟雨。javascript

——洛夫《灵河》java

本文为读 lodash 源码的第四篇,后续文章会更新到这个仓库中,欢迎 star:pocket-lodashgit

gitbook也会同步仓库的更新,gitbook地址:pocket-lodashes6

做用与用法

Hash 顾名思义,就是要有一个离散的序列,根据 key 来储取数据。而在 javascript 中,最适合的无疑是对象了。github

Hash 在 lodash 的 .internal 文件夹中,做为内部文件来使用。lodash 会根据不一样的数据类型选择不一样的缓存方式,Hash 即是其中的一种方式,这种方式只能缓存 key 的类型符合对象键要求的数据。数组

Hash 只接收一个二维数组做为参数,调用方式以下:缓存

new Hash([['tes1', 1],['test2',2],['test3',3]])
复制代码

其中子项中的第一项会做为 key ,第二项是须要缓存的值。微信

Hash 实例化的结果以下:数据结构

{
  size: 3,
  __data__: {
    test1: 1,
    test2: 2,
    test3: 3
  }
}
复制代码

缓存的数量储存在 __data__ 的对象中。ui

Hash与Map

后面将会讲到,除了使用 Hash 方式缓存数据外,还会用到 Map,lodash 在设计 Hash 的数据管理接口时,也与 Map 的接口一致,可是不会包含 Map 的遍历方法。

先来看看这些接口都有那些:

源码

const HASH_UNDEFINED = '__lodash_hash_undefined__'

class Hash {
  constructor(entries) {
    let index = -1
    const length = entries == null ? 0 : entries.length

    this.clear()
    while (++index < length) {
      const entry = entries[index]
      this.set(entry[0], entry[1])
    }
  }
  clear() {
    this.__data__ = Object.create(null)
    this.size = 0
  }
  delete(key) {
    const result = this.has(key) && delete this.__data__[key]
    this.size -= result ? 1 : 0
    return result
  }
  get(key) {
    const data = this.__data__
    const result = data[key]
    return result === HASH_UNDEFINED ? undefined : result
  }
  has(key) {
    const data = this.__data__
    return data[key] !== undefined
  }
  set(key, value) {
    const data = this.__data__
    this.size += this.has(key) ? 0 : 1
    data[key] = value === undefined ? HASH_UNDEFINED : value
    return this
  }
}

export default Hash
复制代码

constructor

constructor(entries) {
    let index = -1
    const length = entries == null ? 0 : entries.length

    this.clear()
    while (++index < length) {
      const entry = entries[index]
      this.set(entry[0], entry[1])
    }
  }
复制代码

constructor 中并无看到初始化 __data__ 属性和 size 属性,这个其实在 clear 方法中初始化了,后面会解释。

接着遍历传入的二维数组,调用 set 方法,初始化缓存的值。将子项的第一项做为 key ,第二项为缓存的值。

clear

clear() {
  this.__data__ = Object.create(null)
  this.size = 0
}
复制代码

clear 的做用是清空缓存,所以须要将 size 重置为 0

将缓存的数据 __data__ 设置为空对象。

这里并无用 this.__data__ = {} 置空,而是调用了 Object.create 方法,而且将 null 做为参数。咱们都知道, Object.create 的第一个参数为建立对象的原型对象,传入 null 的时候,返回的就是一个真空对象,即没有原型的对象,所以不会有原型属性的干扰,用来作缓存对象十分适合。

has

has(key) {
  const data = this.__data__
  return data[key] !== undefined
}
复制代码

has 用来判断是否已经有缓存数据,若是缓存数据已经存在,则返回 true

判断也十分简单,只须要判断取出来的值是否为 undefined 便可。

这个判断有一个坑,后面会讲到。

set

set(key, value) {
  const data = this.__data__
  this.size += this.has(key) ? 0 : 1
  data[key] = value === undefined ? HASH_UNDEFINED : value
  return this
}
复制代码

set 用来增长或者更新须要缓存的值。set 的时候须要同时维护 size 和在缓存的值。

首先调用 has 方法,判断对应的 key 是否已经被缓存过,若是已经缓存过,则 size 保持不变,不然 size1

缓存值其实就是设置缓存对象 this.__data__ 对应 key 属性的值。

has 中说到用 data[key] !== undefined 有一个坑,由于要缓存的值也能够是 undefined ,若是不作处理,确定会致使判断错误。

lodash 的处理方式是将 undefined 的值转换成 HASH_UNDEFINED ,也即一开始便定义的 __lodash_hash_undefined__ 字符串来储存。

因此在缓存中,是用字符串 __lodash_hash_undefined__ 来替代 undefined 的。

set 在最后还将实例 this 返回,以支持链式操做。

get

get(key) {
  const data = this.__data__
  const result = data[key]
  return result === HASH_UNDEFINED ? undefined : result
}
复制代码

get 方法是从缓存中取值。

取值其实就是返回缓存对象中对应 key 的值便可。由于 undefined 在缓存中是以 __lodash_hash_undefined__ 来表示的,所以遇到值为 __lodash_hash_undefined__ 时,返回 undefined

其实这样仍是有小小的问题的,若是须要缓存的值恰好是 __lodash_hash_undefined__,那取出来的值跟预设的就不一致了。可是这样状况应该不多出现吧。

delete

delete(key) {
  const result = this.has(key) && delete this.__data__[key]
  this.size -= result ? 1 : 0
  return result
}
复制代码

delete 方法用来删除指定 key 的缓存。成功删除返回 true, 不然返回 false。 删除操做一样须要维护 size 属性和缓存值。

首先调用 has 方法来判断缓存是否存在,若是存在,用 delete 操做符将 __data__ 中对应的属性删除。

delete 操做符在成功删除属性时会返回 true,若是成功删除,则须要将 size 减小 1

参考

  1. Set 和 Map 数据结构
  2. Object.create()

License

署名-非商业性使用-禁止演绎 4.0 国际 (CC BY-NC-ND 4.0)

最后,全部文章都会同步发送到微信公众号上,欢迎关注,欢迎提意见:

做者:对角另外一面

相关文章
相关标签/搜索