摘要:是基于的自動(dòng)任務(wù)運(yùn)行器,她能自動(dòng)化的完成等文件的測(cè)試檢查合并壓縮格式化瀏覽器自動(dòng)刷新部署文件生成,并監(jiān)聽文件改動(dòng)后重復(fù)這些步驟。易于學(xué)習(xí)通過最少的,掌握毫不費(fèi)力,構(gòu)建工作盡在掌握如同一系列流管道。
Gulp是什么?
gulp是前端開發(fā)過程中一種基于流的代碼構(gòu)建工具,是自動(dòng)化項(xiàng)目的構(gòu)建利器;她不僅能對(duì)網(wǎng)站資源進(jìn)行優(yōu)化,而且能幫我們完成前端開發(fā)過程中的很多重復(fù)的任務(wù),使用它,可以大大提高我們的開發(fā)效率。
gulp是基于NodeJs的自動(dòng)任務(wù)運(yùn)行器,她能自動(dòng)化的完成JavaScript、coffee、sass、less、html/img、css等文件的測(cè)試、檢查、合并、壓縮、格式化、瀏覽器自動(dòng)刷新、部署文件生成,并監(jiān)聽文件改動(dòng)后重復(fù)這些步驟。在實(shí)現(xiàn)上,她借鑒了Unix操作系統(tǒng)的管道思想,前一級(jí)的輸出,直接變成后一級(jí)的輸入,使得在操作上非常簡單。
什么是流?流,簡單來說就是建立在面向?qū)ο蠡A(chǔ)上的一種抽象的數(shù)據(jù)處理工具。在流中,定義了一些處理數(shù)據(jù)的基本操作,如讀取數(shù)據(jù),寫入數(shù)據(jù)等,程序員是對(duì)流進(jìn)行所有操作的,而不用關(guān)心流的另一頭數(shù)據(jù)的真正流向。流不但可以處理文件,還可以處理動(dòng)態(tài)內(nèi)存、網(wǎng)絡(luò)數(shù)據(jù)等多種數(shù)據(jù)形式。
而gulp正是通過流和代碼優(yōu)于配置的策略來盡量簡化任務(wù)編寫的工作。當(dāng)使用流時(shí),gulp去除了中間文件,只將最后的輸出寫入磁盤,整個(gè)過程因此變得更快。
Gulp的特點(diǎn)易于使用:通過代碼優(yōu)于配置的策略,gulp 讓簡單的任務(wù)簡單,復(fù)雜的任務(wù)可管理。
構(gòu)建快捷:利用 Node.js 流的威力,你可以快速構(gòu)建項(xiàng)目并減少頻繁的 IO 操作。
易于學(xué)習(xí):通過最少的 API,掌握 gulp 毫不費(fèi)力,構(gòu)建工作盡在掌握:如同一系列流管道。
插件高質(zhì):gulp 嚴(yán)格的插件指南確保插件如你期望的那樣簡潔高質(zhì)得工作。
安裝首先全局安裝gulp:
npm install -g gulp
全局安裝gulp后,還需要在每個(gè)要使用gulp的項(xiàng)目中都多帶帶安裝一次。把目錄切換到你的項(xiàng)目文件夾中,然后在命令行中執(zhí)行:
npm install gulp
如果想在安裝的時(shí)候把gulp寫進(jìn)項(xiàng)目package.json文件的依賴中,則可以加上--save-dev:
npm install --save-dev gulp
這樣就完成了gulp的安裝,接下來就可以在項(xiàng)目中應(yīng)用gulp了。
gulp的使用一、建立gulpfile.js文件
gulp也需要一個(gè)文件作為它的主文件,在gulp中這個(gè)文件叫做gulpfile.js。新建一個(gè)文件名為gulpfile.js的文件,然后放到你的項(xiàng)目目錄中。之后要做的事情就是在gulpfile.js文件中定義我們的任務(wù)了。下面是一個(gè)最簡單的gulpfile.js文件內(nèi)容示例,它定義了一個(gè)默認(rèn)的任務(wù)。
var gulp = require("gulp"); gulp.task("default",function(){ console.log("hello world"); });
二、運(yùn)行g(shù)ulp任務(wù)
要運(yùn)行g(shù)ulp任務(wù),只需切換到存放gulpfile.js文件的目錄(windows平臺(tái)請(qǐng)使用cmd或者Power Shell等工具),然后在命令行中執(zhí)行gulp命令就行了,gulp后面可以加上要執(zhí)行的任務(wù)名,例如gulp task1,如果沒有指定任務(wù)名,則會(huì)執(zhí)行任務(wù)名為default的默認(rèn)任務(wù)。
工作方式在介紹gulp API之前,我們首先來說一下gulp.js工作方式。在gulp中,使用的是Nodejs中的stream(流),首先獲取到需要的stream,然后可以通過stream的pipe()方法把流導(dǎo)入到你想要的地方,比如gulp的插件中,經(jīng)過插件處理后的流又可以繼續(xù)導(dǎo)入到其他插件中,當(dāng)然也可以把流寫入到文件中。所以gulp是以stream為媒介的,它不需要頻繁的生成臨時(shí)文件,這也是我們應(yīng)用gulp的一個(gè)原因。
gulp的使用流程一般是:首先通過gulp.src()方法獲取到想要處理的文件流,然后把文件流通過pipe方法導(dǎo)入到gulp的插件中,最后把經(jīng)過插件處理后的流再通過pipe方法導(dǎo)入到gulp.dest()中,gulp.dest()方法則把流中的內(nèi)容寫入到文件中。例如:
var gulp = require("gulp"); gulp.src("script/jquery.js") // 獲取流的api .pipe(gulp.dest("dist/foo.js")); // 寫放文件的api
接下來我們將為大家講解gulp api,包括gulp.src(),gulp.task(),gulp.dest(),gulp.watch(),gulp.run()。
srcgulp.src()方法正是用來獲取流的,但要注意這個(gè)流里的內(nèi)容不是原始的文件流,而是一個(gè)虛擬文件對(duì)象流(Vinyl files),這個(gè)虛擬文件對(duì)象中存儲(chǔ)著原始文件的路徑、文件名、內(nèi)容等信息。其語法為:
gulp.src(globs[, options]);
globs參數(shù)是文件匹配模式(類似正則表達(dá)式),用來匹配文件路徑(包括文件名),當(dāng)然這里也可以直接指定某個(gè)具體的文件路徑。當(dāng)有多個(gè)匹配模式時(shí),該參數(shù)可以為一個(gè)數(shù)組;類型為String或Array。
當(dāng)有多種匹配模式時(shí)可以使用數(shù)組:
//使用數(shù)組的方式來匹配多種文件 gulp.src(["js/*.js","css/*.css","*.html"])
options為可選參數(shù)。以下為options的選項(xiàng)參數(shù):
options.buffer
類型: Boolean 默認(rèn)值: true
如果該項(xiàng)被設(shè)置為 false,那么將會(huì)以 stream 方式返回 file.contents 而不是文件 buffer 的形式。這在處理一些大文件的時(shí)候?qū)?huì)很有用。注意:插件可能并不會(huì)實(shí)現(xiàn)對(duì) stream 的支持。
options.read
類型: Boolean 默認(rèn)值: true
如果該項(xiàng)被設(shè)置為 false, 那么 file.contents 會(huì)返回空值(null),也就是并不會(huì)去讀取文件。
options.base
類型: String , 設(shè)置輸出路徑以某個(gè)路徑的某個(gè)組成部分為基礎(chǔ)向后拼接。
如, 請(qǐng)想像一下在一個(gè)路徑為 client/js/somedir 的目錄中,有一個(gè)文件叫 somefile.js :
gulp.src("client/js/**/*.js") // 匹配 "client/js/somedir/somefile.js" 現(xiàn)在 `base` 的值為 `client/js/` .pipe(minify()) .pipe(gulp.dest("build")); // 寫入 "build/somedir/somefile.js" 將`client/js/`替換為build gulp.src("client/js/**/*.js", { base: "client" }) // base 的值為 "client" .pipe(minify()) .pipe(gulp.dest("build")); // 寫入 "build/js/somedir/somefile.js" 將`client`替換為builddest
gulp.dest()方法是用來寫文件的,其語法為:
gulp.dest(path[,options])
path為寫入文件的路徑;
options為一個(gè)可選的參數(shù)對(duì)象,以下為選項(xiàng)參數(shù):
options.cwd
類型: String 默認(rèn)值: process.cwd()
輸出目錄的 cwd 參數(shù),只在所給的輸出目錄是相對(duì)路徑時(shí)候有效。
options.mode
類型: String 默認(rèn)值: 0777
八進(jìn)制權(quán)限字符,用以定義所有在輸出目錄中所創(chuàng)建的目錄的權(quán)限。
var gulp = require("gulp"); gulp.src("script/jquery.js") // 獲取流 .pipe(gulp.dest("dist/foo.js")); // 寫放文件
下面再說說生成的文件路徑與我們給gulp.dest()方法傳入的路徑參數(shù)之間的關(guān)系。
gulp.dest(path)生成的文件路徑是我們傳入的path參數(shù)后面再加上gulp.src()中有通配符開始出現(xiàn)的那部分路徑。例如:
var gulp = reruire("gulp"); //有通配符開始出現(xiàn)的那部分路徑為 **/*.js gulp.src("script/**/*.js") .pipe(gulp.dest("dist")); //最后生成的文件路徑為 dist/**/*.js //如果 **/*.js 匹配到的文件為 jquery/jquery.js ,則生成的文件路徑為 dist/jquery/jquery.js
用gulp.dest()把文件流寫入文件后,文件流仍然可以繼續(xù)使用。
watchgulp.watch()用來監(jiān)視文件的變化,當(dāng)文件發(fā)生變化后,我們可以利用它來執(zhí)行相應(yīng)的任務(wù),例如文件壓縮等。其語法為:
gulp.watch(glob[, opts], tasks);
glob 為要監(jiān)視的文件匹配模式,規(guī)則和用法與gulp.src()方法中的glob相同。 opts 為一個(gè)可選的配置對(duì)象,通常不需要用到。 tasks 為文件變化后要執(zhí)行的任務(wù),為一個(gè)數(shù)組。
gulp.task("uglify",function(){ //do something }); gulp.task("reload",function(){ //do something }); gulp.watch("js/**/*.js", ["uglify","reload"]);
gulp.watch()還有另外一種使用方式:
gulp.watch(glob[, opts, cb]);
glob和opts參數(shù)與第一種用法相同;
cb參數(shù)為一個(gè)函數(shù)。每當(dāng)監(jiān)視的文件發(fā)生變化時(shí),就會(huì)調(diào)用這個(gè)函數(shù),并且會(huì)給它傳入一個(gè)對(duì)象,該對(duì)象包含了文件變化的一些信息,type屬性為變化的類型,可以是added,changed,deleted;path屬性為發(fā)生變化的文件的路徑。
gulp.watch("js/**/*.js", function(event){ console.log(event.type); //變化類型 added為新增,deleted為刪除,changed為改變 console.log(event.path); //變化的文件的路徑 });task
gulp.task方法用來定義任務(wù),內(nèi)部使用的是Orchestrator(用于排序、執(zhí)行任務(wù)和最大并發(fā)依賴關(guān)系的模塊),其語法為:
gulp.task(name[, deps], fn)
name 為任務(wù)名;
deps 是當(dāng)前定義的任務(wù)需要依賴的其他任務(wù),為一個(gè)數(shù)組。當(dāng)前定義的任務(wù)會(huì)在所有依賴的任務(wù)執(zhí)行完畢后才開始執(zhí)行。如果沒有依賴,則可省略這個(gè)參數(shù);
fn 為任務(wù)函數(shù),我們把任務(wù)要執(zhí)行的代碼都寫在里面。該參數(shù)也是可選的。
當(dāng)你定義一個(gè)簡單的任務(wù)時(shí),需要傳入任務(wù)名字和執(zhí)行函數(shù)兩個(gè)屬性。
gulp.task("greet", function () { console.log("Hello world!"); });
執(zhí)行gulp greet的結(jié)果就是在控制臺(tái)上打印出“Hello world”。
你也可以定義一個(gè)在gulp開始運(yùn)行時(shí)候默認(rèn)執(zhí)行的任務(wù),并將這個(gè)任務(wù)命名為“default”:
gulp.task("default", function () { // Your default task });
前面已經(jīng)介紹了gulp.task的語法,但是當(dāng)有多個(gè)任務(wù)時(shí),需要知道怎么來控制任務(wù)的執(zhí)行順序。
可以通過任務(wù)依賴來實(shí)現(xiàn)。例如我想要執(zhí)行one,two,three這三個(gè)任務(wù),那我們就可以定義一個(gè)空的任務(wù),然后把那三個(gè)任務(wù)當(dāng)做這個(gè)空的任務(wù)的依賴就行了:
//只要執(zhí)行default任務(wù),就相當(dāng)于把one,two,three這三個(gè)任務(wù)執(zhí)行了 gulp.task("default",["one","two","three"]);
如果任務(wù)相互之間沒有依賴,任務(wù)就會(huì)按你書寫的順序來執(zhí)行,如果有依賴的話則會(huì)先執(zhí)行依賴的任務(wù)。但是如果某個(gè)任務(wù)所依賴的任務(wù)是異步的,就要注意了,gulp并不會(huì)等待那個(gè)所依賴的異步任務(wù)完成,而是會(huì)接著執(zhí)行后續(xù)的任務(wù)。例如:
gulp.task("one",function(){ //one是一個(gè)異步執(zhí)行的任務(wù) setTimeout(function(){ console.log("one is done") },5000); }); //two任務(wù)雖然依賴于one任務(wù),但并不會(huì)等到one任務(wù)中的異步操作完成后再執(zhí)行 gulp.task("two",["one"],function(){ console.log("two is done"); });
上面的例子中我們執(zhí)行two任務(wù)時(shí),會(huì)先執(zhí)行one任務(wù),但不會(huì)去等待one任務(wù)中的異步操作完成后再執(zhí)行two任務(wù),而是緊接著執(zhí)行two任務(wù)。所以two任務(wù)會(huì)在one任務(wù)中的異步操作完成之前就執(zhí)行了。
那如果我們想等待異步任務(wù)中的異步操作完成后再執(zhí)行后續(xù)的任務(wù),該怎么做呢?
有三種方法可以實(shí)現(xiàn):
第一:在異步操作完成后執(zhí)行一個(gè)回調(diào)函數(shù)來通知gulp這個(gè)異步任務(wù)已經(jīng)完成,這個(gè)回調(diào)函數(shù)就是任務(wù)函數(shù)的第一個(gè)參數(shù)。
gulp.task("one",function(cb){ //cb為任務(wù)函數(shù)提供的回調(diào),用來通知任務(wù)已經(jīng)完成 //one是一個(gè)異步執(zhí)行的任務(wù) exec(function(){ console.log("one is finish"); cb(); //執(zhí)行回調(diào),表示這個(gè)異步任務(wù)已經(jīng)完成 },5000); }); //這時(shí)two任務(wù)會(huì)在one任務(wù)中的異步操作完成后再執(zhí)行 gulp.task("two",["one"],function(){ console.log("two is finish"); });
第二:定義任務(wù)時(shí)返回一個(gè)流對(duì)象。適用于任務(wù)就是操作gulp.src獲取到的流的情況。
gulp.task("one",function(cb){ var stream = gulp.src("client/**/*.js") .pipe(exec()) //exec()中有某些異步操作 .pipe(gulp.dest("build")); return stream; }); gulp.task("two",["one"],function(){ console.log("two is done"); });
第三:返回一個(gè)promise對(duì)象,例如
var Q = require("q"); gulp.task("one", function() { var deferred = Q.defer(); // 執(zhí)行異步的操作 setTimeout(function() { deferred.resolve(); }, 1); return deferred.promise; }); gulp.task("two",["one"],function(){ console.log("two is done"); });run
gulp.run()表示要執(zhí)行的任務(wù)。可能會(huì)使用單個(gè)參數(shù)的形式傳遞多個(gè)任務(wù)。如下代碼:
gulp.task("end",function(){ gulp.run("task1","task3","task2"); });
注意:任務(wù)是盡可能多的并行執(zhí)行的,并且可能不會(huì)按照指定的順序運(yùn)行。
插件安裝在我們編寫gulp代碼時(shí)候,需要用到一些gulp的插件,可以通過npm install --save-dev 插件名稱 來安裝。如下代碼來安裝自動(dòng)加載插件:
npm install --save-dev gulp-load-plugins
要使用gulp的插件,首先得用require來把插件加載進(jìn)來。
var gulp=require("gulp"), plugins=require("gulp-load-plugins")(), uglify = require("gulp-uglify"), minifyHtml = require("gulp-minify-html"), rename = require("gulp-rename");
gulp的插件有很多種,我們將在本節(jié)內(nèi)容中為同學(xué)們講解幾個(gè)插件的用法。如還想了解更多插件,請(qǐng)查閱相關(guān)資料。
自動(dòng)加載gulp-load-plugins這個(gè)插件能自動(dòng)幫你加載package.json文件里的gulp插件。例如假設(shè)你的package.json文件里的依賴是這樣的:
{ "devDependencies": { "gulp": "~3.6.0", "gulp-rename": "~1.2.0", "gulp-ruby-sass": "~0.4.3", "gulp-load-plugins": "~0.5.1" } }
然后我們可以在gulpfile.js中使用gulp-load-plugins來幫我們加載插件:
var gulp = require("gulp"); //加載gulp-load-plugins插件,并馬上運(yùn)行它 var plugins = require("gulp-load-plugins")();
然后我們要使用gulp-rename和gulp-ruby-sass這兩個(gè)插件的時(shí)候,就可以使用plugins.rename和plugins.rubySass來代替了,也就是原始插件名去掉gulp-前綴,之后再轉(zhuǎn)換為駝峰命名。
重命名gulp-rename插件用來重命名文件流中的文件。用gulp.dest()方法寫入文件時(shí),文件名使用的是文件流中的文件名,如果要想改變文件名,那可以在之前用gulp-rename插件來改變文件流中的文件名。
var gulp = require("gulp"), rename = require("gulp-rename"), uglify = require("gulp-uglify"); gulp.task("rename", function () { gulp.src("src/1.js") .pipe(uglify()) //壓縮 .pipe(rename("1.min.js")) //會(huì)將1.js重命名為1.min.js .pipe(gulp.dest("js")); });js文件壓縮
gulp-uglify插件用來壓縮js文件。
var gulp = require("gulp"), uglify = require("gulp-uglify"); gulp.task("minify-js", function () { gulp.src("src/*.js") // 要壓縮的js文件 .pipe(uglify()) //使用uglify進(jìn)行壓縮 .pipe(gulp.dest("dist/js")); //壓縮后的路徑 });css文件壓縮
gulp-minify-css插件用來壓縮css文件。
var gulp = require("gulp"), minifyCss = require("gulp-minify-css"); gulp.task("minify-css", function () { gulp.src("src/*.css") // 要壓縮的css文件 .pipe(minifyCss()) //壓縮css .pipe(gulp.dest("dist/css")); });html文件壓縮
gulp-minify-html插件用來壓縮html文件。
var gulp = require("gulp"), minifyHtml = require("gulp-minify-html"); gulp.task("minify-html", function () { gulp.src("src/*.html") // 要壓縮的html文件 .pipe(minifyHtml()) //壓縮 .pipe(gulp.dest("dist/html")); });js代碼檢查
使用gulp-jshint插件,用來檢查js代碼。
var gulp = require("gulp"), jshint = require("gulp-jshint"); gulp.task("jsLint", function () { gulp.src("src/*.js") .pipe(jshint()) .pipe(jshint.reporter()); // 輸出檢查結(jié)果 });文件合并
使用gulp-concat插件,用來把多個(gè)文件合并為一個(gè)文件,我們可以用它來合并js或css文件等。
var gulp = require("gulp"), concat = require("gulp-concat"); gulp.task("concat", function () { gulp.src("src/*.js") //要合并的文件 .pipe(concat("all.js")) // 合并匹配到的js文件并命名為 "all.js" .pipe(gulp.dest("dist/js")); });圖片壓縮
可以使用gulp-imagemin插件來壓縮jpg、png、gif等圖片。
var gulp = require("gulp"); var imagemin = require("gulp-imagemin"); var pngquant = require("imagemin-pngquant"); //png圖片壓縮插件 gulp.task("default", function () { return gulp.src("src/images/*") .pipe(imagemin({ progressive: true, use: [pngquant()] //使用pngquant來壓縮png圖片 })) .pipe(gulp.dest("dist")); });自動(dòng)刷新
使用gulp-livereload插件,當(dāng)代碼變化時(shí),它可以幫我們自動(dòng)刷新頁面。
var gulp = require("gulp"), less = require("gulp-less"), livereload = require("gulp-livereload"); gulp.task("less", function() { gulp.src("less/*.less") .pipe(less()) .pipe(gulp.dest("css")) .pipe(livereload()); }); gulp.task("watch", function() { livereload.listen(); //要在這里調(diào)用listen()方法 gulp.watch("less/*.less", ["less"]); });
好了,到這里gulp的基本用法都已經(jīng)講解完了,最重要的還是在項(xiàng)目中多多使用才能盡快掌握。本文來源于筆者學(xué)習(xí)匯智網(wǎng)課堂的學(xué)習(xí)筆記。原課程作者:草莓。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/78780.html
摘要:前言與是目前圈子內(nèi)比較活躍的前端構(gòu)建工具。對(duì)于初學(xué)者來說,對(duì)這二者往往容易認(rèn)識(shí)不清,今天,就從事件的源頭,說清楚與。它可以將許多松散的模塊按照依賴和規(guī)則打包成符合生產(chǎn)環(huán)境部署的前端資源。打包后形成的文件出口。 前言:Webpack 與 gulp是目前圈子內(nèi)比較活躍的前端構(gòu)建工具。網(wǎng)上有很多二者比較的文章,面試中也會(huì)經(jīng)常遇到gulp,Webpack的區(qū)別這樣的問題。對(duì)于初學(xué)者來說,對(duì)這二...
摘要:今天跟大家分享的是關(guān)于目前自己的公司的項(xiàng)目優(yōu)化中所使用到的構(gòu)建工具。在進(jìn)行基礎(chǔ)鞏固的基礎(chǔ)上,使用構(gòu)建工具可以在開發(fā)的過程中進(jìn)行部分自動(dòng)化構(gòu)建,如對(duì)頁面的監(jiān)聽,對(duì)樣式的編譯,對(duì)代碼的壓縮與打包。在使用之前,要先對(duì)所需要的依賴進(jìn)行安裝。 Why? 今天跟大家分享的是關(guān)于目前自己的公司的項(xiàng)目優(yōu)化中所使用到的gulp構(gòu)建工具。目前公司對(duì)產(chǎn)品的優(yōu)化決定采用VueJs來進(jìn)行單頁面開發(fā),無疑這種新的...
以前我們敲靜態(tài)頁面都是寫好html,css, js,然后再去刷新瀏覽器,艾尼馬又不行,有重新寫過再刷新,一個(gè)頁面下來按chrl+r的次數(shù)可讓你的鍵盤多活好幾天,要不會(huì)刷新快捷按鈕那不得手殘了都。 后來,grunt,gulp等工具應(yīng)運(yùn)而生,當(dāng)然它們不止這個(gè)功能,但卻很好地減輕了我們的負(fù)擔(dān)。這篇文章介紹一些如何使用這些工具使構(gòu)建頁面變得簡單高效。這只是個(gè)人的目前在用的不成熟的方案,更專業(yè)的還請(qǐng)參考...
摘要:本節(jié)將學(xué)習(xí)是如何利用形成一套完整的前端工作流模式的。你也可以使用下面命令來強(qiáng)制安裝所有模塊,不管該模塊之前是否安裝過由于國內(nèi)墻的原因,使用安裝會(huì)非常緩慢,慢到想切,不過還好,我們可以使用淘寶提供的國內(nèi)鏡像進(jìn)行下載。 本節(jié)將學(xué)習(xí) Laravel 是如何利用 Sass, NPM, Gulp形成一套完整的前端工作流模式的。 一、句法強(qiáng)大的樣式表Sass Sass 是一種可用于編寫CSS的語言...
摘要:那時(shí)候所配置的任務(wù)監(jiān)聽匹配文件的變化自動(dòng)刷新瀏覽器自動(dòng)編譯自動(dòng)補(bǔ)全前綴多雪碧圖合并拼圖等等基于編譯圖片的任務(wù),已經(jīng)是完全滿足我們的需求了。直至到后來在雪碧圖的合并,多倍圖的輸出上,在上苦苦找尋不了比較完美的解決方案等等。 折騰 從 2015 到現(xiàn)在,短短的三年內(nèi),幾乎每年折騰一下工作流的 更新?lián)Q代 。從最早開始使用 Grunt 到 Gulp 再到 Webpack,再到 Rollup,...
閱讀 2600·2021-09-23 11:21
閱讀 1894·2021-09-22 15:15
閱讀 985·2021-09-10 11:27
閱讀 3452·2019-08-30 15:54
閱讀 666·2019-08-30 15:52
閱讀 1343·2019-08-30 15:44
閱讀 2358·2019-08-29 15:06
閱讀 2984·2019-08-28 18:21