20190329期
怎样完成一个Loader?
我们在上几节有讲过loader,本日我们来深切相识它们,最暴力的体式格局莫过于着手完成它们
好了,回到正题, 先来回忆一下loader
loader定义: 用于对模块的源代码举行转换。loader 能够使你在 import 或”加载”模块时预处置惩罚文件
简朴运用
module.exports = {
//...
module: {
rules: [
{
test: /\.js$/,
use: [
{
// loader 是导出为一个a函数的 node 模块。该函数在 loader 转换资本的时刻挪用
// 给定的函数将挪用 loader API,并经由过程 this 上下文接见
loader: path.resolve('loader.js'),
options: {/* ... */}
}
]
}
]
}
};
回忆了loader的定义及简朴运用后,我们再来剖析一下完成loader的思绪
- 单一职责,一个loader只做一件事
- 链式组合,链中的每一个 loader 会将转换应用在已处置惩罚过的资本上
- 模块化,是导出为一个函数的 node 模块
- 参数兼并,loader 能够经由过程 options 对象设置
基于上面剖析的几点,我们最先着手
// 这个就是一个最简朴loader,
// 假如我们的loader有依靠别的模块,也得以module的写法将在在顶部引入
import fs from 'fs';
export default function(source){
return source
}
我们发明上面直接运用了return,是因为是同步类的loader且返回的内容唯一,假如你愿望你的loader支撑链式挪用,将效果返给下一个loader继承运用,这时刻就需要用webpack供应的api
这里我们简朴看一下this.callback的定义,一个能够同步或许异步挪用的能够返回多个效果的函数。预期的参数是
this.callback(
err: Error | null,
content: string | Buffer,
sourceMap?: SourceMap,
meta?: any
)
// loader-utils 它供应了许多有效的东西
// 最经常使用的一个就是猎取传入 loader 的 options
import { getOptions } from 'loader-utils';
export default function(source, other) {
const options = getOptions(this)
// do whatever you want
// ...
this.callback(null, source, other)
}
手写一个loader对没有研究过的听上去彷佛有点难,事实上, 控制上面所引见的内容及头脑,就能够最先写一个简朴的 Loader 了, 我们再来用简朴的代码绥一下loader究竟是什么?
// 起首loader它是一个node模块,这很好明白
export const lessToCss = function(source, other) {
// source 就是你即将要转换的文件源
// TODO
// 将转换好的文件源流转至一个管道
this.callback(null, source, other)
}
让你的loader更好用
loader api中有几个好用的家伙这里就趁便带一下
- this.cacheable() 从进步实行效力上,怎样处置惩罚应用缓存是极其重要的, webpack 中this.cacheable就能够轻松将loader缓存了
- this.async() 当一个loader无依靠时,我们应当异步的去返回效果
案例剖析
下方贴上less-loader的源码,代码很简约,连系上方我们所剖析的,也很轻易明白
import processResult from './processResult';
const render = pify(less.render.bind(less));
function lessLoader(source) {
const loaderContext = this;
const options = getOptions(loaderContext);
const done = loaderContext.async();
const isSync = typeof done !== 'function';
if (isSync) {
throw new Error(
'Synchronous compilation is not supported anymore. See https://github.com/webpack-contrib/less-loader/issues/84'
);
}
processResult(loaderContext, render(source, options));
}
总结
- loader是一个node模块
- 编写loader时要遵照单一准绳,每一个loader只做一种”转义”事情
- webpack为我们供应了雄厚的loader api
- webpack为我们还供应了东西函数集——loader-utils
关于JS逐日一题
JS逐日一题能够看成是一个语音答题社区
天天应用碎片时候采纳60秒内的语音情势来完成当天的考题
群主在越日0点推送当天的参考答案
- 注 毫不仅限于完成当天使命,更多是查漏补缺,进修群内别的同砚优异的答题思绪