Lodash 浓缩
Lodash 是个十分有用的工具库,但我们往往只需要其中的一小部分函数。这时,整个 lodash 库就显得十分庞大,我们需要减小 lodash 的体积。
cherry-pick 方法
Lodash 官方在 npm 上为每个方法维护了独立的包,便于我们直接引用。举个例子,如果我们需要lodash.chunk方法,可以直接安装lodash.chunk这个 npm 包,在代码中 import 即可:
1 |
import _chunk from 'lodash.chunk'; |
这样的弊端很明显,我们如果需要使用一个新方法,就要重新 install 一个新包并 require 进来。另外,每个包之间会有一部分重复(公共)方法会被重复打包,也会增大最终的打包体积。更好的解决办法,是直接从 lodash 完整库内导入方法:
1 |
import flatten from 'lodash/flatten'; |
由于 lodash 库中,每个方法都以独立模块封装,这样的引入方法,在 webpack 打包时,只会按需取指定的两个模块进行打包。从flatten 和 flatMap) 的代码可以看出,二者都用到了./.internal/baseFlatten.js这个模块,在 webpack 打包时,这个模块就变成了公共模块,这就避免了引入两个独立方法的包带来的重复。这种引入方式,官方叫做 “cherry-pick methods”,就像捡樱桃一样,捡自己需要的方法。
babel-plugin-lodash
然而,cherry-pick 方法的书写习惯,和我们平时使用第三方库的书写习惯并不一致。我们更习惯于直接 import 整个库:
1 |
import _ from 'lodash'; |
这时,就是babel-plugin-lodash库发挥作用的时候。在 babel 编译代码时,这个插件可以自动把这种代码
1 |
import _ from 'lodash'; |
转化为
1 |
import _add from 'lodash/fp/add'; |
不过,这里有一些额外限制:必须要引入 babel(6以上版本);必须要使用 ES2015 的方式引入包(即 import 的方式)。
lodash-webpack-plugin
在这一步的基础上,我们还可以使用 lodash-webpack-plugin 进一步减包。它的原理是将我们用到的方法(和间接用到的内部方法)使用更简单的方法替代。具体的方法对应关系,参照源代码中的mapping.js。使用方法也非常简单,只需要在 webpack 配置中引入该 plugin ,即可将所有的 cheery-pick 引入的方法自动进行转换。
需要注意的问题
一般介绍 lodash 打包优化的文章,介绍到这里就算结束了。但是,lodash-webpack-plugin 还有一系列的 feature 配置,它们的作用又是什么?实际上,经过 lodash-webpack-plugin 转化的 lodash 函数,已经和文档中对应的函数不完全等价,因此,如果我们用到了一些特定的特性,我们需要通过调整这些配置,保证我们的代码正常运行。
要关注这些配置的作用,只需要关注mapping.js这个文件,其中配置了features和overrides两个变量。features中每一项对应一个开关,每个开关则对应着若干组替换规则。以cloning为例,当对应配置关闭(默认即为关闭)时,插件将把内部的_._baseClone(暂且这样表示)方法替换为_.identity方法。而overrides变量中的每一项表示:只要我们引用了对应方法,则打开对应配置开关。
了解了这一点,我们再谈谈这个配置是怎样影响对应方法的。以_.sortBy方法为例,在它的源代码中(注:由于我们使用 npm 安装,因此需要查看 es 分支的源代码),我们可以看到它使用了四个内部库:
1 |
import baseFlatten from './_baseFlatten.js'; |
在flattening规则中,内部_baseFlatten将会被替换为_.head方法。本来根据_.sortBy文档,下列代码可以正常运行:
1 |
var users = [ |
经过插件转化后,这种写法将会抛出异常。我们只能使用下列写法:
1 |
_.sortBy(users, [function(o) { return o.user; }]);
|
收到类似影响的函数还有_.overArgs,_.flow等几个方法。同理,_.get,_.has等方法,也会_baseGet方法被替换受到影响。
要解决这个问题,我们需要校验自己的代码中,有没有用到官方描述的Feature Sets中类似的特性,然后更改自己的代码,或者开启对应的特性配置。一般情况下,开启 flattening 和 paths 配置可以解决大部分问题。
Lodash 浓缩的更多相关文章
- 延迟求值-如何让Lo-Dash再提速x100?
「注释」作者在本文里没有说明这么一个事实: 目前的版本Lo-Dash v2.4.1并没有引入延迟求值的特性,Lo-Dash 3.0.0-pre中部分方法进行了引入,比如filter(),map(),r ...
- 函数式 js 接口实现原理,以及 lodash/fp 模块
函数式 js 接口 之前在 youtube 上看到一个技术视频,讲“underscore.js的接口为什么不好用”,以及什么样的接口更好用.演讲者是 lodash.js 的作者,他提出了一种“全面函数 ...
- lodash
lodash常用函数一 <!DOCTYPE html> <html lang="en"> <head> <meta charset=&qu ...
- lodash 替换 underscore
不少知名项目都在用lodash替换underscore lodash Lazy evaluation 英文原文:http://filimanjaro.com/blog/2014/introducin ...
- JavaScript工具库之Lodash
你还在为JavaScript中的数据转换.匹配.查找等烦恼吗?一堆看似简单的foreach,却冗长无趣,可仍还在不停的repeat it!也许你已经用上了Underscore.js,不错,你已经进步很 ...
- inoic start projectname sidemenu报错 - Error: Cannot find module 'lodash._baseslice'
inoic start projectname sidemenu报错 - Error: Cannot find module 'lodash._baseslice' 在公司的电脑上出现过这个错误,后来 ...
- lodash链式使用
chain var _ = require('lodash'); var user1 = { name: 'zhangsan', height: 180, weight: 120 }; var use ...
- lodash常用方法1--查询
1.find var _ = require('lodash'); var user1 = { name: 'zhangsan', height: 180, weight: 120 }; var us ...
- 高性能的JavaScript库---Lodash
上周在仿做Nodejs社区的时候,遇到了lodash这个javascript库,很惭愧,那也是我第一次听说lodash.人嘛,对于新鲜的事物总是会或多或少感到些好奇的,于是就毫不犹豫地去lodash官 ...
随机推荐
- spark + yarn调试问题java.nio.channels.ClosedChannelException
spark客户端提交任务至yarn,后台抛错,FinalStatus:UNDEFINED. ./spark-submit --class org.apache.spark.examples.Spar ...
- 谈谈let与const
let 命令 let命令用于声明变量,但是与传统var命令的不同之处在于拥有以下特性: 使用let命令声明的变量只在let命令所在的代码块内有效(我将之称为变量绑定): 不存在变量提升: 存在暂时性死 ...
- linux按照进程名杀掉进程
1.按照进程名杀掉进程 ps -ef | grep sftp | grep mysql |grep -v grep | awk '{print("kill -9 ", ...
- 数据结构——堆(Heap)大根堆、小根堆
目录 Heap是一种数据结构具有以下的特点: 1)完全二叉树: 2)heap中存储的值是偏序: Min-heap: 父节点的值小于或等于子节点的值: Max-heap: 父节点的值大于或等于子节点的值 ...
- Python学习札记(十八) 高级特性4 生成器
参考:生成器 Note 1.通过列表生成式,我们可以直接创建一个列表.但是,受到内存限制,列表容量肯定是有限的,且容易造成空间浪费.所以,如果列表元素可以按照某种算法推算出来,那我们可以在循环的过程中 ...
- Zookeeper Zkclient客户端
Zkclient是对Zookeeper的原生API进行了包装,实现了超时重连.Watcher反复注册等功能,它可以实现递归创建,删除节点,但是zkClient不能递归给节点赋值. 主要的api如下: ...
- angular 当使用ng-repeat 时出现 $$hashKey的键值对
小问题 把: ng-repeat="item in items " 改成 : ng-repeat="item in items track by $index"
- angular之自定义 directive
1,指令的创建至少需要一个带有@Directive装饰器修饰的控制器类.@Directive装饰器指定了一个选择器名称,用于指出与此指令相关联的属性的名字. 2,创建一个highlight.direc ...
- [mybatis]Record与Example的用法
一.Record 一个Record是一个Dao对象(继承Mapper接口),tkmybatis会将record自动映射成sql语句,record中所有非null的属性都作为sql语句,如: 映射的sq ...
- nginx 日志搜集解决方案
# nginx 日志搜集解决方案 ## 系统环境描述 ``` java8 logstash --监控nginx日志文件 ``` ## 技术描述 ``` 通过logstash监控nginx access ...