前端开发近两年工程化大幅飙升。随着Nodejs大放异彩,静态文件处理不再需要其他语言辅助。主要的两大工具即为基于文件的grunt,基于流的gulp。简单来说,如果需要的只是文件处理,gulp绝对首选。如果是其他依赖于文件的任务管理,例如测试(karmamocha),推荐使用grunt

  gulp常用api:

  1. gulp.src(globs[,options])
  2.  
  3. gulp.dest(path[,options])
  4.  
  5. gulp.task(name[,deps], fn)
  6.  
  7. gulp.watch(glob [, opts], tasks) or gulp.watch(glob [, opts, cb])
  8.  
  9. gulp.start(["param"]);

一、gulp plugin开发依赖

  就插件开发难度而言,gulp远低于grunt。如果你只关注如何处理文件,而不关注细节,那么需要依赖Nodejs Transform stream的实现。可以使用官方推荐的through2,但推荐使用through-gulp。后者是基于前者,为gulp插件编写精简优化重写而来。千万不要使用through,这个包时间久远,长时间没有维护,而且部分mock实现的功能,到nodejs 0.10.x已经原生支持。如果只是想学习如何编写gulp插件,through-gulp更适合。

  through-gulp: https://github.com/bornkiller/through-gulp
  through2: https://github.com/rvagg/through2.git
  through: https://github.com/dominictarr/through

二、利用through-gulp开发gulp plugin

 依赖API

  1. var through = require('through-gulp');
  2. var stream = through(transformFunction, flushFunction);

 结构 

  1. // PLUGIN_NAME: sample
  2. var through = require('through-gulp');
  3.  
  4. function sample() {
  5. //通过through创建流stream
  6. var stream = through(function(file, encoding,callback) {
  7.  
  8. //进程文件判断
  9. if (file.isNull()) {
  10.  
  11. }
  12. if (file.isBuffer()) {
  13.  
  14. }
  15. if (file.isStream()) {
  16.  
  17. }
  18. // just pipe data next, or just do nothing to process file later in flushFunction
  19. // never forget callback to indicate that the file has been processed.
  20. this.push(file);
  21. callback();
  22. },function(callback) {
  23. // just pipe data next, just callback to indicate that the stream's over
  24. this.push(something);
  25. callback();
  26. });
  27.  
  28. //返回这个流文件
  29. return stream;
  30. };
  31.  
  32. // exporting the plugin
  33. module.exports = sample;

 这里

  1. through(function(file, encoding,callback){})发现file是一个对象,含有如下许多属性,但是我们常用的通常是file.path获取文件路径,file.contents获取文件内容

  1.  

 使用:

  1. var gulp = require('gulp');
  2. var sample = require('sample');
  3. gulp.task('sample', function() {
  4. return gulp.src(['source file'])
  5. .pipe(sample())
  6. .pipe(gulp.dest('file destiny'))
  7. });

  从以上我们可以看到,through-gulp插件写法,其实就是读取转换流,存储流,导出流的一个过程(一个文件一个文件的过去),如果我们不需要导出流进行链式写法,其实直接module.exports = sample就可以直接单向使用。

  下面来看一下简单的 gulp-pf-replace插件,理解原理:

  1. //Gulp默认使用buffer
  2.  
  3. var through = require("through-gulp"); //引入gulp插件模块
  4. var fs = require("fs");
  5. var http = require("http");
  6. var request = require("request");
  7. var path = require("path");
  8. var source = require('vinyl-source-stream'); //常规流转换为gulp支持的Vinyl文件格式
  9. var gutil = require('gulp-util');
  10. //gulp多功能的插件,可以替换扩展名,log颜色日志,模板
  11.  
  12. var chalk = require('chalk'); //设置颜色
  13. chalk.blue('Hello world!');
  14.  
  15. // 类型判断
  16. function isType(type){
  17. return function(o){
  18. return Object.prototype.toString.crall(o) === '[object ' + type + ']';
  19. }
  20. }
  21.  
  22. var isString = isType("String");
  23. var isObject = isType("Object");
  24. var isArray = isType("Array");
  25.  
  26. gutil.log('stuff happened', 'Really it did', gutil.colors.magenta('123'));
  27.  
  28. var i=0;
  29. //gulp插件原理就是一个流进入,流处理完出来
  30. function replace(modReplace) {
  31.  
  32. //通过through创建流stream
  33. var stream = through(function(file, encoding,callback) {
  34. //file为对象,含有path,clone,pipe,inspect,history,isNull,isDirectory 等,常用的是path
  35. //console.log(isObject(file));
  36.  
  37. //进程文件判断
  38. if (file.isNull()) {
  39. throw "NO Files,Please Check Files!"
  40. }
  41.  
  42. //buffer对象可以操作
  43. if (file.isBuffer()) {
  44. //拿到单个文件buffer
  45. var content = modReplace(file.contents.toString("utf-8"));
  46.  
  47. //console.log(contents);
  48. file.contents = new Buffer(content,"utf-8");
  49. //可以通过buffer.toString("utf-8")转换成字符串
  50. //contents = file.contents.toString("utf-8")
  51. }
  52.  
  53. //stream流是不能操作的,可以通过fs.readFileSync
  54. if (file.isStream()) {
  55. //同步读取
  56. var content = modReplace(fs.readFileSync(file.path).toString("utf-8"));
  57. file.contents = new Buffer(content,"utf-8");
  58. }
  59.  
  60. // just pipe data next, or just do nothing to process file later in flushFunction
  61. // never forget callback to indicate that the file has been processed.
  62. this.push(file);
  63. callback();
  64. i++;
  65. },function(callback) {
  66. gutil.log( gutil.colors.red(i) , gutil.colors.green("已经处理完毕!"));
  67. // just pipe data next, just callback to indicate that the stream's over
  68. // this.push(something);
  69. callback();
  70. });
  71.  
  72. //返回这个流文件
  73. return stream;
  74. };
  75.  
  76. // 导出插件
  77. module.exports = replace;

  使用:

  1. gulp.task("pfDefault",function(){
  2. return gulp.src("./tianzun/*.+(html|htm)",{buffer: true})
  3. .pipe(pfDefault(ypReplace))
  4. .pipe(gulp.dest("./out"))
           .on("finish",function(){
              console.log("处理完成")
           })
  5. });
  6.  
  7. //替换方法
  8. function ypReplace(data){
  9. return data.replace(/helloword/,"")
  10. }

  上面注解比较多,应该大多数人看得懂,这里我就不再做解释。

三、利用through2开发gulp plugin

  结构如下:

  1. // through2 是一个对 node 的 transform streams 简单封装
  2. var through = require('through2');
  3. var gutil = require('gulp-util');
  4. var PluginError = gutil.PluginError;
  5.  
  6. // 常量
  7. const PLUGIN_NAME = 'gulp-prefixer';
  8.  
  9. function prefixStream(prefixText) {
  10. var stream = through();
  11. stream.write(prefixText);
  12. return stream;
  13. }
  14.  
  15. // 插件级别函数 (处理文件)
  16. function gulpPrefixer(prefixText) {
  17.  
  18. if (!prefixText) {
  19. throw new PluginError(PLUGIN_NAME, 'Missing prefix text!');
  20. }
  21. prefixText = new Buffer(prefixText); // 预先分配
  22.  
  23. // 创建一个让每个文件通过的 stream 通道
  24. return through.obj(function(file, enc, cb) {
  25. if (file.isNull()) {
  26. // 返回空文件
  27. cb(null, file);
  28. }
  29. if (file.isBuffer()) {
  30. file.contents = Buffer.concat([prefixText, file.contents]);
  31. }
  32. if (file.isStream()) {
  33. file.contents = file.contents.pipe(prefixStream(prefixText));
  34. }
  35.  
  36. cb(null, file);
  37.  
  38. });
  39.  
  40. };
  41.  
  42. // 暴露(export)插件主函数
  43. module.exports = gulpPrefixer;

推荐阅读:

  Gulp思维——Gulp高级技巧  理解gulp底层处理是buffer、还是Vinyl文件格式流

  编写gulp指导

  through-gulp插件

  从零单排之gulp实战 理解gulp的相关原理

教你写gulp plugin的更多相关文章

  1. Flutter实战:手把手教你写Flutter Plugin

    前言 如果你对移动端有所关注,那么你一定会听说过Flutter.得益于Google,Flutter一经推出便得受到了广泛关注.很多开发者跃跃欲试,国内部分大厂,诸如美团.闲鱼等团队已经开始了Flutt ...

  2. Android开发之手把手教你写ButterKnife框架(二)

    欢迎转载,转载请标明出处: http://blog.csdn.net/johnny901114/article/details/52664112 本文出自:[余志强的博客] 上一篇博客Android开 ...

  3. 半个小时教你写一个装(bi)逼(she)之地图搜租房

    半个小时教你写一个装(bi)逼(she)之地图搜租房 首先需要一个Python3环境,怎么准备我就不多说了,实在不会的出门右转看一下廖雪峰老师的博客. HTML部分 代码来自:高德API+Python ...

  4. 手把手教你写DI_0_DI是什么?

    DI是什么? Dependency Injection 常常简称为:DI. 它是实现控制反转(Inversion of Control – IoC)的一个模式. fowler 大大大神 "几 ...

  5. 手把手教你写Sublime中的Snippet

    手把手教你写Sublime中的Snippet Sublime Text号称最性感的编辑器, 并且越来越多人使用, 美观, 高效 关于如何使用Sublime text可以参考我的另一篇文章, 相信你会喜 ...

  6. 看过《大湿教我写.net通用权限框架(1)之菜单导航篇》之后发生的事(续)——主界面

    引言 在UML系列学习中的小插曲:看过<大湿教我写.net通用权限框架(1)之菜单导航篇>之后发生的事 在上篇中只拿登录界面练练手,不把主界面抠出来,实在难受,严重的强迫症啊.之前一直在总 ...

  7. 手把手教你写LKM rookit! 之 第一个lkm程序及模块隐藏(一)

    唉,一开始在纠结起个什么名字,感觉名字常常的很装逼,于是起了个这<手把手教你写LKM rookit> 我觉得: 你们觉得:...... 开始之前,我们先来理解一句话:一切的操作都是系统调用 ...

  8. 手把手教你写电商爬虫-第三课 实战尚妆网AJAX请求处理和内容提取

    版权声明:本文为博主原创文章,未经博主允许不得转载. 系列教程: 手把手教你写电商爬虫-第一课 找个软柿子捏捏 手把手教你写电商爬虫-第二课 实战尚妆网分页商品采集爬虫 看完两篇,相信大家已经从开始的 ...

  9. 手把手教你写电商爬虫-第四课 淘宝网商品爬虫自动JS渲染

    版权声明:本文为博主原创文章,未经博主允许不得转载. 系列教程: 手把手教你写电商爬虫-第一课 找个软柿子捏捏 手把手教你写电商爬虫-第二课 实战尚妆网分页商品采集爬虫 手把手教你写电商爬虫-第三课 ...

随机推荐

  1. 【Linux】理解分区

    http://blog.csdn.net/aaronychen/article/details/2270048 主分区逻辑分区设置 http://forum.ubuntu.org.cn/viewtop ...

  2. tooltips插件

    摘要: 继‘带箭头提示框’,本文将分享几款带箭头提示框. qtipqTip是一种先进的提示插件,基于jQuery框架.以用户友好,而且功能丰富,qTip为您提供不一般的功能,如圆角和语音气泡提示,并且 ...

  3. python 函数结果缓存一段时间的装饰器

    把函数结果缓存一段时间,比如读取一个mongodb,mongodb中的内容又在发生变化,如果从部署后,自始至终只去读一次那就感触不到变化了,如果每次调用一个函数就去读取那太频繁了耽误响应时间也加大了c ...

  4. python内存泄漏,python垃圾手动回收,1

    部署的舆情系统,内存变大,找原因. 一个小例子. def func(): local_list = list(range(10000000)) func() time.sleep(200) 能够观察到 ...

  5. [OpenCV] IplImage and Operation

    IplImage 一.资源 In this chapter, APIs will make U crazy. Good luck! Next, Review Linear Algebra. Ref:  ...

  6. c#中如何退出程序后自动重新启动程序

    //触发退出程序事件 private void button1_Click(object sender, EventArgs e)        {             Application.E ...

  7. SQLServer------远程调用失败

    1.情况 出现 2.解决方法 打开“控制面板” -> “卸载程序” -> 找到 “Microsoft SQL Server 2016) ExpressLocalDB”将其卸载 -> ...

  8. JAVA简单内存泄露分析及解决

    一.问题产生    项目采用Tomcat6.0为服务器,数据库为mysql5.1,数据库持久层为hibernate3.0,以springMVC3.0为框架,项目开发完成后,上线前夕进行稳定性拷机,测试 ...

  9. c#事件Unity与.Net对比

    今天在看Unity3d的书,发现上面的调用事件比较特殊,比如说按钮事件 该方法写在OnGUI方法中if(GUILayout.Button("按钮1")) { //执行事件的处理 } ...

  10. spring+hibernate 下载

    http://www.cnblogs.com/haogj/archive/2012/07/28/nhibernate.html-------原文 http://www.springframework. ...