Gulp

gulp是基于NodeJs环境的一个构建工具,用来进行项目构建,处理如压缩js代码,转化less文件等任务,需要进行全局安装

npm install gulp -g

gulp实现不同的构建任务是通过其它插件来执行,gulp负责调用这些插件,可以简单理解为,gulp就是插卡游戏机,负责操作,如果想要不同的游戏那么需要换不同的卡带

图片1.png

需要注意,在使用gulp时需要将gulp下载到本地文件,同时下载gulp对应的插件,必须在本地新建文集gulpfile.js,名字是固定的,然后在文件内配置gulp

解析Less

  1. 需要在本地安装gulp,在当前文件夹执行命令

    npm install gulp
    

    注意安装完成后再本地目录会有一个node_modules文件来存放node的文件

  2. 在本地安装gulp-less插件

    npm install gulp-less
    
  3. 在gulpfile.js文件中进行配置

    var gulp=require("gulp"); //引入gulp文件,此时我们定义的gulp就转换为了一个对象,有gulp对应的方法
    
    var less=require("gulp-less"); //引入gulp-less文件,同上
    
    gulp.task("less2css",function(){ //通过.task来配置方法,less2css使我们定义的任务名
    
     gulp.src('./less/*.less')  //引入要转化的less文件,可以传入正则
         .pipe(less())  
         //pipe作用等同于管道符,将上一次的输入结果作为下一的输出,在这里将我们获取到的less文件传给less()函数处理
         .pipe(gulp.dest('./css'))
         //保存转换完成后的文件到指定的文件夹,如果该文件夹不存在那么会新创建一个
    })
    
  4. 在git中执行本次任务

    gulp less2css
    
  5. 完成后再当前目录会有一个css文件夹,内部就是我们转化后的css文件

相关插件

Less解析

gulp-less

压缩CSS

gulp-cssmin

压缩JS

gulp-uglify

gulp.task("uglify",function(){
    gulp.src("./js/*")
        .pipe(uglify())
        .pipe(gulp.dest("./release/js"))
})

添加前缀

gulp-autoprefixer

相同的一个文件对不同插件的使用是可以写在一个task命令下的

var gulp=require("gulp"); //引入gulp文件

var less=require("gulp-less"); //引入gulp-less文件,同上

var cssmin=require("gulp-cssmin"); //引入压缩css插件

gulp.task("less2css",function(){ 

    gulp.src('./less/*.less') 
        .pipe(less())  
        .pipe(cssmin())
        .pipe(gulp.dest('./css'))
        //先将代码转化为css格式然后将代码进行压缩后输出
})

gulp-autoprefixer是可以进行配置的

gulp.task("less2css",function(){ 
    gulp.src('./less/*.less') 
        .pipe(less())  
        .pipe(autoprefixer({
          browsers:['last 2 versions','Firefox >= 20'],//为什么版本的浏览器添加,可以选择,参数可以百度
          cascade:true,//是否美化,默认是true,效果如下
                       //-webkit-transform: rotate(45deg);
                       //        transform: rotate(45deg);
          remove:true  //是否删除不必要的前缀,默认为true
        }))
        .pipe(cssmin())
        .pipe(gulp.dest('./css'))
        //将代码转化为css格式然后为代码添加前缀后将代码进行压缩后输出
})

图片压缩

gulp-imagemin

gulp.task("image",function(){
    gulp.src(["./images/**/*","./img/*"],{base:"./"})
    //src中可以传入2个参数,第一个参数可以是单独的字符串,也可以是数组,如果是多个文件夹下都有图片需要进行压缩,那么直接将这些文件夹路径传入即可,当然也可以写正则
    //第二个参数可以写一个对象,对象中可以对src进行配置,主要了解base属性,该属性是用来声明在我们使用dest命令时如何存储经过操作的文件,如上,base赋值./代表gulp在存储的时候会去掉src中传入地址中的./,将剩余的部分拼接在dast路径上,如果我们在src中传入数组,那么会依次执行上述操作
        .pipe(imagemin())
        .pipe(gulp.dest("./release"))
})

代码合并

gulp-concat

gulp.task("concat",function(){
    gulp.src(["./css/index.css","./css/cart.css"]) //导入要合并的文件
    .pipe(concat("all.css")) //执行合并操作,all.css是我们为合并后的文件起的名字
    .pipe(cssmin()) //将合并后的文件进行压缩
    .pipe(gulp.dest("./release/css"))
})

压缩HTML

gulp-htmlmin

注意,该命令必须传入参数,否则没有效果

gulp.task("html",function(){
    gulp.src("./index.html")
        .pipe(htmlmin({
            removeComments: true,//清除HTML注释
            collapseWhitespace: true,//压缩HTML
            collapseBooleanAttributes: true,//省略布尔属性的值 <input checked="true"/> ==> <input />
            removeEmptyAttributes: true,//删除所有空格作属性值 <input id="" /> ==> <input />
            removeScriptTypeAttributes: true,//删除<script>的type="text/javascript"
            removeStyleLinkTypeAttributes: true,//删除<style>和<link>的type="text/css"
             minifyJS: true,//压缩页面JS
             minifyCSS: true//压缩页面CSS
        }))
        .pipe(gulp.dest("./release"))
})

添加版本号

gulp-rev

这里说一下浏览器的缓存机制,方便我们来加深理解,浏览器本身是会有缓存机制的,在第一次访问服务器的时候将文件下载到本地缓存起来,如果在下一次向服务器发送请求的时候发现文件的请求路径没有发生变化,那么就不从服务器请求,而是直接获取本地的缓存,这样就能够提升用户下一次打开页面时的速度,同时也会减小对服务器的压力,但是同时也会产生一个问题,如果我们在第一次提交的时候出现了BUG或者后期我们对JS或其他文件进行了更改,浏览器在加载页面时发现文件地址没有变化,那么它仍然会从缓存中获取文件,那么就会导致我们修改的部分不能反馈到用户的电脑上,所以我们在更改文件后要及时的改变文件的请求路径,来让浏览器重新下载我们的文件,在早期我们一直使用的是时间戳,也就是在url的地址后加一个当前的时间来让浏览器认为地址发生了改变继而来重新请求,时间戳一般设为一天

url:"....?new Date().getDate()"

但是这样也会有一个问题,如果我们一天之内出现了两个BUG,那么我们需要在一天之内两次修改文件,那么这个时间戳是不起作用的,但如果设置到秒,用户每一次请求都会重新加载,影响加载速度,所以最好的方法是我们能够合理的利用缓存,在我们修改了文件之后再重新加载,gulp为我们提供了解决的方案,就是添加版本号的插件,它会为我们文件添加一个字符,在我们内容改变的时候,字符也会改变,这就相当于改变了我们文件的地址路径,浏览器就会重新加载我们的文件,配合重命名插件一起使用

重命名

gulp-rename

gulp.task("uglify",function(){
    gulp.src("./js/*")
        .pipe(uglify())
        .pipe(rev()) //调用rev插件
        .pipe(gulp.dest("./release/js")) //将添加版本号后的文件进行存储
        .pipe(rev.manifest()) //用manifest命令来生成一个json文键储存每个文件版本号变化之后与之前的对应关系
        .pipe(rename("js-manifest.json")) //由于gulp-rev默认生成的json文件名都是相同的,会出现覆盖,所以需要                                          使用gulp-rename插件对每一个进行重命名
        .pipe(gulp.dest("./release/js/rev")) //将json文件存放发到该路径下
})

这里我们说一下gulp-rev的大概思路,在我们使用MD5加密的时候都知道MD5有一个特性,不管文件内容的大小,都会统一转换为一个32位的字符,如果内容不发生变化的时候,字符不会发生改变,但是如果内容发生改变,那么就会生成新的字符,由MD5这个特性,我们可以将文件内容通过MD5直接加密,然后将加密后的字符添加在我们文件名字的结尾,在内容改变时,名字也就不同

gulp-useref和gulp-if

这两个命令比较有意思,可以配合一起使用

使用gulp-useref前,需要在HTML页面中写入标记

    <!-- build:css ./all.css -->
    <!-- build是开头的固定写法,如果要合并的是css文件,那么:后写css,是js文件写js,,./all.css表示的是合并后存储的路径以及合并后的名称-->
    <link rel="stylesheet" href="./css/a.css">
    <link rel="stylesheet" href="./css/b.css">
    <!-- endbuild -->
    <!--endbuild是结尾的固定写法,这两个注释就是userf的标示,表示这两者之间的代码要合并-->

接着在gulpfile.js中写入命令

gulp.task('useref', function () {
    gulp.src('./index.html') //src中写入的是做了标记的HTML页面的地址
        .pipe(useref())
        .pipe(gulpif('*.js', uglify())) //gulp-if的作用是过滤文件,比如这里,它会把以js结尾的文件过滤出来,然                                        后调用uglify命令,将这部分代码进行压缩
        .pipe(gulp.dest('./release/view'));
});

gulp-userf还有删除文件的功能,例如我们在最后项目上线的时候,是不需要页面中的less.js的,所以我们需要将其删除

<!-- build:remove -->
<script src="./public/libs/less.js"></script>
<!-- endbuild -->

内容替换

gulp-rev- collector

用来将经过rev设定版本号以后页面中引入文件的名字替换原来的文件名

gulp.task('revCollertor', ['css', 'image', 'useref'], function () { //将之前的image等命令作为依赖注入到rev命令中,在rev命令执行之前先执行这些命令
    gulp.src(['./release/rev/*.json', './release/*.html', './release/views/*.html'], {base: './release'}) //[]中第一个参数是rev下的json文件,里面存储文件的对应关系,后面的参数分别是我们需要做替换的文件
        .pipe(revCollector())
        .pipe(gulp.dest('./release'));//存储到release文件下
});

注意在我们将less文件转化为css格式之后,页面中我们写的less名字需要在合并时改为.css的格式,否则在内容替换的时候gulp不能正确对应并更换

其他任务

对于一些不需要我们构建的任务,例如后台的代码,我们只需要将它们整体传入到release文件夹下即可

gulp.task('other', function () {
    
    gulp.src(['./api/*', './public/*(fonts|libs)/*'], {base: './'})
        .pipe(gulp.dest('./release'));

});

整体执行

在我们项目构建完成后,我们可已将所有的命令作为依赖注入到一个命令中,方便以后我们再次构建,再一次构建的时候只需要执行这个命令,而不需要将所有的命令都重复执行

gulp.task("release",["revController","ourther"]); //之前我们已经将其他命令注入到revController中,所以不需要                                                  重复传递了

gulp接口

task

在task中可以传入3个参数,分别是任务名称,任务依赖,以及执行任务的callback

gulp.task("revCollertor",["image","css"],function(){
  // gulpr任务的执行是异步操作,有可能在我们rev下的json文件还没有生成的情况下替换任务就已经开始了,那么此时替换任务找不到文件对应的json文件,默认不会替换页面内容,那么页面中的文件名就不会被替换,所以需要我们将之前的任务作为依赖注入到revCollertor中,在revCollertor执行时保证其他任务先执行完毕
  //这里表明revCollertor任务需要在image和css任务完成之后再执行
})

这里还需要注意,如果我们要想将其他任务作为依赖,那么在其他任务执行时需要将它们返回,在gulp前添加一个return,如下:

gulp.task('useref', function () {
    return gulp.src('./index.html') 
               .pipe(useref())
               .pipe(gulpif('*.js', uglify())) 
               .pipe(gulp.dest('./release/view'));
});

src

路径

pipe

管道

dest

存储,参数为存储路径

watch

可以实时监听,例如我们可以使用该功能实时监听less变化,然后将less解析为css

gulp.task("watchLess",function(){
    gulp.src("./css/index.less") //监听css下的index.less文件的变化
        .pipe(less()) //调用gulp-less插件
        .pipe(gulp.dest("./css/index.css")); //存储为css文件,然后在页面中直接引入index.css即可
})

还能够实现浏览器实时展现css文件效果

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容

  • gulpjs是一个前端构建工具,与gruntjs相比,gulpjs无需写一大堆繁杂的配置参数,API也非常简单,学...
    依依玖玥阅读 3,150评论 7 55
  • 前言 本文默认你已经安装好node环境,并且熟悉node命令,和window cd命令。 gulp简介 基于nod...
    9I阅读 1,979评论 4 50
  • 1、gulp的安装 首先确保你已经正确安装了nodejs环境。然后以全局方式安装gulp: npm install...
    F_imok阅读 2,369评论 1 11
  • 对网站资源进行优化,并使用不同浏览器测试并不是网站设计过程中最有意思的部分,但是这个过程中的很多重复的任务能够使用...
    懵逼js阅读 1,061评论 0 8
  • 在现在的前端开发中,前后端分离、模块化开发、版本控制、文件合并与压缩、mock数据等等一些原本后端的思想开始...
    Charlot阅读 5,435评论 1 32