tapable

tapable是一種事件驅動型事件流機制,本身是一個獨立的庫纷铣。webpack 通過 tapable 將實現(xiàn)與流程解耦召烂,所有具體實現(xiàn)通過插件的形式存在。

一. 工作流程
1.實例化Hook注冊事件監(jiān)聽
2.通過Hook觸發(fā)事件監(jiān)聽
3.執(zhí)行懶編譯生成的可執(zhí)行代碼

二庵寞、 Hook
Hook的執(zhí)行機制可分為同步和異步兩種,其中異步又分為并行和串行兩種模式薛匪。

Hook的鉤子類型分為四種:
1. Hook:普通鉤子捐川,監(jiān)聽器之間互相獨立不干擾
2. BailHook:熔斷鉤子,某個監(jiān)聽返回非undefined時后續(xù)不執(zhí)行
3. WaterfallHook:瀑布鉤子蛋辈,上一個監(jiān)聽的返回值可傳遞至下一個
4. LoopHook:循環(huán)鉤子属拾,如果當前返回非undefined則一直執(zhí)行(從頭開始執(zhí)行)(webpack中不常見)

三、同步鉤子

  1. SyncHook
    每個事件獨立執(zhí)行冷溶,互相不影響
const { SyncHook } = require('tapable')
let hook = new SyncHook(['name', 'age'])

hook.tap('fn1', function (name, age) {
  console.log('fn1--->', name, age)
})
hook.tap('fn2', function (name, age) {
  console.log('fn2--->', name, age)
})

hook.call('jerry', 18)

// fn1---> jerry 18
// fn2---> jerry 18
  1. SyncBailHook
    某個監(jiān)聽返回非undefined時后續(xù)不執(zhí)行
const { SyncBailHook } = require('tapable')
let hook = new SyncBailHook(['name', 'age'])

hook.tap('fn1', function (name, age) {
  console.log('fn1--->', name, age)
})
hook.tap('fn2', function (name, age) {
  console.log('fn2--->', name, age)
  return undefined
})
hook.tap('fn3', function (name, age) {
  console.log('fn3--->', name, age)
})

hook.call('jerry', 100)

// fn1---> jerry 100
// fn2---> jerry 100
// fn3---> jerry 100
  1. SyncWaterfallHook
    上一個監(jiān)聽的返回值可傳遞至下一個渐白,覆蓋原來的參數(shù)值
const { SyncWaterfallHook } = require('tapable')
let hook = new SyncWaterfallHook(['name', 'age'])

hook.tap('fn1', function (name, age) {
  console.log('fn1--->', name, age)
  return 'ret1'
})
hook.tap('fn2', function (name, age) {
  console.log('fn2--->', name, age)
  return 'ret2'
})
hook.tap('fn3', function (name, age) {
  console.log('fn3--->', name, age)
  return 'ret3'
})

hook.call('jerry', 33)

// fn1---> jerry 33
// fn2---> ret1 33
// fn3---> ret2 33
  1. SyncLoopHook
    執(zhí)行時,如果都沒有返回逞频,則執(zhí)行一遍后就退出
    如果有一個返回了非undefined纯衍,則從頭重新開始執(zhí)行
    當遇到返回了undefined則執(zhí)行完所有監(jiān)聽后退出
const { SyncLoopHook } = require('tapable')
let hook = new SyncLoopHook(['name', 'age'])

let count1 = 0
let count2 = 0

hook.tap('fn1', function (name, age) {
  console.log('fn1--->', name, age)
  if (++count1 === 1) {
    count1 = 0
    return undefined
  }
  return true
})
hook.tap('fn2', function (name, age) {
  console.log('fn2--->', name, age)
  if (++count2 === 2) {
    count2 = 0
    return undefined
  }
  return true
})
hook.tap('fn3', function (name, age) {
  console.log('fn3--->', name, age)
})

hook.call('foo', 100)

// 不是undefined時候從上到下繼續(xù)依次執(zhí)行,直到返回undefined
// fn1---> foo 100
// fn2---> foo 100
// fn1---> foo 100
// fn2---> foo 100
// fn3---> foo 100

四苗胀、異步鉤子
對于異步鉤子的使用襟诸,在添加事件監(jiān)聽時會存在三種方式: tap、tapAsync基协、 tapPromise

  1. AsyncParalleHook 異步并行
const { AsyncParallelHook } = require('tapable')

let hook = new AsyncParallelHook(['name'])

// 對于異步鉤子的使用歌亲,在添加事件監(jiān)聽時會存在三種方式: tap tapAsync tapPromise
// hook.tap('fn1', function (name) {
//   console.log('fn1--->', name)
// })

// hook.tap('fn2', function (name) {
//   console.log('fn2--->', name)
// })

// hook.callAsync('zoe', function () {
//   console.log('最后執(zhí)行了回調操作')
// })

/* console.time('time')
hook.tapAsync('fn1', function (name, callback) {
  setTimeout(() => {
    console.log('fn1--->', name)
    callback()
  }, 1000)
})

hook.tapAsync('fn2', function (name, callback) {
  setTimeout(() => {
    console.log('fn2--->', name)
    callback()
  }, 2000)
})

hook.callAsync('lg', function () {
  console.log('最后一個回調執(zhí)行了')
  console.timeEnd('time')
}) */

// 03 promise 
console.time('time')
hook.tapPromise('fn1', function (name) {
  return new Promise(function (resolve, reject) {
    setTimeout(() => {
      console.log('fn1--->', name)
      resolve()
    }, 1000)
  })
})

hook.tapPromise('fn2', function (name) {
  return new Promise(function (resolve, reject) {
    setTimeout(() => {
      console.log('fn2--->', name)
      resolve()
    }, 2000)
  })
})

hook.promise('foo').then(() => {
  console.log('end執(zhí)行了')
  console.timeEnd('time')
})

// fn1---> foo
// fn2---> foo
// end執(zhí)行了
// time: 2.007s
  1. AsyncParallelBailHook 異步并行
const { AsyncParallelBailHook } = require('tapable')

let hook = new AsyncParallelBailHook(['name'])

console.time('time')
hook.tapAsync('fn1', function (name, callback) {
  setTimeout(() => {
    console.log('fn1--->', name)
    callback()
  }, 1000)
})

hook.tapAsync('fn2', function (name, callback) {
  setTimeout(() => {
    console.log('fn2--->', name)
    callback('err')
  }, 2000)
})

hook.tapAsync('fn3', function (name, callback) {
  setTimeout(() => {
    console.log('fn3--->', name)
    callback()
  }, 3000)
})

hook.callAsync('zce', function () {
  console.log('最后的回調執(zhí)行了')
  console.timeEnd('time')
})

// fn1---> zce
// fn2---> zce
// 最后的回調執(zhí)行了
// time: 2.004s
// fn3---> zce  執(zhí)行fn3是因為定時器到時間了,正常是不會執(zhí)行的
  1. AsyncSeriesHook 異步串行澜驮。 其他異步串行還有AsyncSeriesBailHook陷揪、AsyncSeriesLoopHook、AsyncSeriesWaterfallHook杂穷。
const { AsyncSeriesHook } = require('tapable')

let hook = new AsyncSeriesHook(['name'])

console.time('time')
hook.tapPromise('fn1', function (name) {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      console.log('fn1--->', name)
      resolve()
    }, 1000)
  })
})

hook.tapPromise('fn2', function (name) {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      console.log('fn2--->', name)
      resolve()
    }, 2000)
  })
})

hook.promise('foo').then(function () {
  console.log('~~~~')
  console.timeEnd('time')
})

// fn1---> foo
// fn2---> foo
// ~~~~
// time: 3.029s
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末悍缠,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子耐量,更是在濱河造成了極大的恐慌飞蚓,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,284評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件廊蜒,死亡現(xiàn)場離奇詭異趴拧,居然都是意外死亡溅漾,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,115評論 3 395
  • 文/潘曉璐 我一進店門著榴,熙熙樓的掌柜王于貴愁眉苦臉地迎上來樟凄,“玉大人,你說我怎么就攤上這事兄渺。” “怎么了汰现?”我有些...
    開封第一講書人閱讀 164,614評論 0 354
  • 文/不壞的土叔 我叫張陵挂谍,是天一觀的道長。 經(jīng)常有香客問我瞎饲,道長口叙,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,671評論 1 293
  • 正文 為了忘掉前任嗅战,我火速辦了婚禮妄田,結果婚禮上,老公的妹妹穿的比我還像新娘驮捍。我一直安慰自己疟呐,他們只是感情好,可當我...
    茶點故事閱讀 67,699評論 6 392
  • 文/花漫 我一把揭開白布东且。 她就那樣靜靜地躺著启具,像睡著了一般。 火紅的嫁衣襯著肌膚如雪珊泳。 梳的紋絲不亂的頭發(fā)上鲁冯,一...
    開封第一講書人閱讀 51,562評論 1 305
  • 那天,我揣著相機與錄音色查,去河邊找鬼薯演。 笑死,一個胖子當著我的面吹牛秧了,可吹牛的內容都是我干的跨扮。 我是一名探鬼主播,決...
    沈念sama閱讀 40,309評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼示惊,長吁一口氣:“原來是場噩夢啊……” “哼好港!你這毒婦竟也來了?” 一聲冷哼從身側響起米罚,我...
    開封第一講書人閱讀 39,223評論 0 276
  • 序言:老撾萬榮一對情侶失蹤钧汹,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后录择,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體拔莱,經(jīng)...
    沈念sama閱讀 45,668評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡碗降,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,859評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了塘秦。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片讼渊。...
    茶點故事閱讀 39,981評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖尊剔,靈堂內的尸體忽然破棺而出爪幻,到底是詐尸還是另有隱情,我是刑警寧澤须误,帶...
    沈念sama閱讀 35,705評論 5 347
  • 正文 年R本政府宣布挨稿,位于F島的核電站,受9級特大地震影響京痢,放射性物質發(fā)生泄漏奶甘。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,310評論 3 330
  • 文/蒙蒙 一祭椰、第九天 我趴在偏房一處隱蔽的房頂上張望臭家。 院中可真熱鬧,春花似錦方淤、人聲如沸钉赁。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,904評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽橄霉。三九已至,卻和暖如春邑蒋,著一層夾襖步出監(jiān)牢的瞬間姓蜂,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,023評論 1 270
  • 我被黑心中介騙來泰國打工医吊, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留钱慢,地道東北人。 一個月前我還...
    沈念sama閱讀 48,146評論 3 370
  • 正文 我出身青樓卿堂,卻偏偏與公主長得像束莫,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子草描,可洞房花燭夜當晚...
    茶點故事閱讀 44,933評論 2 355

推薦閱讀更多精彩內容