gulp 是一个使用“流”来实现自动化的工具,正如 官方文档 首页展示的这副动图一样,以“流动”的状态去处理 TypeScript、PNG、Markdown 资源。

与webpack比较

类别webpackgulp
核心理念module bundlertask runner
执行任务模块化定义任务、等待执行
构建方式loader文件Stream
支持插件

gulp 相比 webpack 思想会更加简单、易用,更适合一些自动化任务(比如Jquery、编译文件上传到服务器等功能),但它默认不支持模块化,所以大型的项目(Vue、React、Angle)并不会使用它。

执行任务

安装依赖

webpack 一样,gulp 可以全局安装,也可以在项目局部安装,这里通过 npm install gulp 仅在当前项目安装

配置文件

webpack 中默认的配置文件为 webpack.config.js,而 gulp 中默认配置文件为 gulpfile.js,根目录创建文件,定义一个函数,将它导出,这就定义了一个简单的任务。

const foo = () => {
  console.log('foo');
};

module.exports = {
  foo,
};
编译文件

执行 npx gulp foo 命令编译任务,在 gulp 后面跟上函数名 foo 告知执行的任务名称

但此时编译会报错,提醒任务没有完成,是否忘记了发送异步任务完成的信号

异步任务

这是因为默认情况所有任务是异步的,想要结束有两种方式

  1. 返回固定的内容,stream、promise、event emitter、child process 或 observable 类型
  2. 接受一个 callback 作为参数,调用 callback 函数任务结束

这里使用第二种方式

const foo = (cb) => {
  console.log('foo');
  cb();
};

再次执行 npx gulp foo,此时编译成功,执行时间 2.59ms

默认任务

上面执行命令时,gulp 后面跟了任务名称 foo,如果不跟名称时,执行的是默认任务,需要在 gulpfile.js 中导出默认任务

module.exports.default = (cb) => {
  console.log('default task');
  cb();
};

此时可以看到输出默认任务

串行并行

当存在多个任务需要进行组合时,可以通过 gulp 提供的方法,series 表示串行,parallel 表示并行。

const { series, parallel } = require('gulp');
const task1 = (cb) => {
  setTimeout(() => {
    console.log('task1');
    cb();
  }, 2000);
};

const task2 = (cb) => {
  setTimeout(() => {
    console.log('task2');
    cb();
  }, 2000);
};

const task3 = (cb) => {
  setTimeout(() => {
    console.log('task3');
    cb();
  }, 2000);
};

const seriesTask = series(task1, task2, task3);
const parallelTask = parallel(task1, task2, task3);
module.exports = {
  seriesTask,
  parallelTask,
};

串行会等上一个任务执行完成,再执行下一个,任务的完成时间为所有任务的总和,并行就会将所有任务一起执行。

读取文件、监听

在项目的 src 文件夹下定义 index.js 文件,通过 gulp 暴露的 src()dest() 方法用于处理计算机上存放的文件。

const { src, dest } = require('gulp');
const task = () => {
  return src('src/*.js').pipe(dest('dist'));
};
module.exports = {
  task,
};

此时src文件夹下的 index.js 文件被读取到 dist 文件夹下

此时的 js 代码没有做 es6 - es5 的转化,也没有压缩,想要达到这些效果,需要使用 gulp 的插件,分别是 gulp-babelgulp-terser,定义方式和 webpack 中 babel 和 terser是一致的,不太了解 babelterser 的朋友可以点链接查看。

需注意,使用 gulp-babel 需要安装 babel 的核心库 @babel/core 以及指定编译规则用到的预设 @babel/preset-env 或者其它插件。

const { src, dest } = require('gulp');
const babel = require('gulp-babel');
const terser = require('gulp-terser');

const task = () => {
  return src('src/*.js')
    .pipe(babel({ presets: ['@babel/preset-env'] }))
    .pipe(terser({ mangle: { toplevel: true } }))
    .pipe(dest('dist'));
};

module.exports = {
  task,
};

通过 pipe() 处理完资源返回一个“流”文件交给下一个插件处理,此时 dist 文夹下的 index.js 资源就进行了代码转换和压缩。

webpack 在编译的时候提供了 --watch 属性,当源码资源发生变化时,自动重新编译,gulp 也提供这样的功能,使用 watch() 来实现。

const { watch } = require('gulp');
watch('src/*.js', task);

使用 watch 后,编译不会结束,每当监听的文件修改并保存时,重新编译。

总结

  • gulp 主要以“流”的方式来处理资源,没有模块化,不适合大型项目。
  • gulp 每次处理即开启异步任务,可以并行、串行、监听资源的更新。
  • 使用插件,gulp 也能实现代码转化、压缩等功能。

以上就是 gulp 的介绍, 更多有关 前端工程化 的内容可以参考我其它的博文,持续更新中~


一颗冰淇淋
170 声望18 粉丝

开心学前端 : )