真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

vue3源碼解讀之timeslicing的使用方法

今天給大家?guī)硪黄创a解析的文章,emm 是關(guān)于 vue3 的,vue3 源碼放出后,已經(jīng)有很多文章來分析它的源碼,我覺得很快又要爛大街了,哈哈

成都創(chuàng)新互聯(lián)專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務(wù),包含不限于成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站制作、靖安網(wǎng)絡(luò)推廣、微信小程序開發(fā)、靖安網(wǎng)絡(luò)營(yíng)銷、靖安企業(yè)策劃、靖安品牌公關(guān)、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運(yùn)營(yíng)等,從售前售中售后,我們都將竭誠(chéng)為您服務(wù),您的肯定,是我們最大的嘉獎(jiǎng);成都創(chuàng)新互聯(lián)為所有大學(xué)生創(chuàng)業(yè)者提供靖安建站搭建服務(wù),24小時(shí)服務(wù)熱線:028-86922220,官方網(wǎng)址:www.cdcxhl.com

不過今天我要解析的部分是已經(jīng)被廢除的 time slicing 部分,這部分源碼曾經(jīng)出現(xiàn)在 vue conf 2018 的視頻中,但是源碼已經(jīng)被移除掉了,之后可能也不會(huì)有人關(guān)注,所以應(yīng)該不會(huì)爛大街

打包

閱讀源碼之前,需要先進(jìn)行打包,打包出一份干凈可調(diào)試的文件很重要

vue3 使用的 rollup 進(jìn)行打包,我們需要先對(duì)它進(jìn)行改造

import cleanup from 'rollup-plugin-cleanup'
plugins: [
  cleanup() //增加了一個(gè) cleanup 插件
   
  tsPlugin,
  aliasPlugin,
  createReplacePlugin(isProductionBuild, isBunlderESMBuild, isCompat),
  ...plugins
],

增加 cleanup 插件主要目的是打包出無注釋的文件

以上,是我個(gè)人閱讀源碼的習(xí)慣,我覺得注釋和類型的作用就是礙眼的,所以先去掉再說

用例

我們?cè)谧x源碼之前,需要先實(shí)現(xiàn)一個(gè)正確用例,但是我讀的這個(gè)版本的源碼,還是 class 的,怎么辦?

這個(gè)時(shí)候我們可以根據(jù)測(cè)試用例來猜測(cè)并給出代碼

function block () {
 const start = performance.now()
 while (performance.now() - start < 2) {
 }
}

class Test extend Component {
 render (props) {
  block()
  return h('li', props.msg)
 }
}

class App extend Component {
 msg = ''
 render () {
  const list = []
  for (let i = 0; i < 200; i++) {
   list.push(h(Test, { key: i, msg: this.msg }))
  }
  return [
   h('input', {
    onInput: e => {
     this.msg = e.target.value
    }
   }),
   h('div',list)
  ]
 }
}

很好,現(xiàn)在我們有了一個(gè)爭(zhēng)取,簡(jiǎn)單的用例了,接下來就是一股腦調(diào)試

調(diào)試

由于我在 fre 中也實(shí)現(xiàn)了時(shí)間切片,所以我對(duì)它非常了解,我知道它的作用原理,所以我們直接搜索宏任務(wù),哈,果然有

window.addEventListener('message', event => {
  if (event.source !== window || event.data !== key) {
    return;
  }
  flushStartTimestamp = getNow();
  try {
    flush();
  }
  catch (e) {
    handleError(e);
  }
}, false);
function flushAfterMacroTask() {
  window.postMessage(key, `*`);
}

這段代碼非常容易理解,就是在宏任務(wù)隊(duì)列里執(zhí)行了 flush 函數(shù),繼續(xù)

然后關(guān)鍵就來了

function flush() {
  let job;
  while (true) {
    job = stageQueue.shift();
    if (job) {
      stageJob(job);
    }
    else {
      break;
    }
    {
      const now = getNow();
      if (now - flushStartTimestamp > frameBudget && job.expiration > now) {
        break; // 此處為關(guān)鍵,意思是超過16ms,或者任務(wù)過期,跳出循環(huán)
      }
    }
  }
  ... 以下代碼省略...

上面的循環(huán)很關(guān)鍵,它做的事情很簡(jiǎn)單的,從 stageQueue 里出棧一個(gè)任務(wù),然后執(zhí)行 stateJob

stateJob 做的事情很簡(jiǎn)單,就是往 commitQueue 里 push 這個(gè)任務(wù)

function stageJob(job) {
  if (job.ops.length === 0) {
    currentJob = job;
    job.cleanup = job();
    currentJob = null;
    commitQueue.push(job); //重點(diǎn)在這里
    job.status = 2;
  }
}

到目前為止,我們?cè)创a讀了一丟丟,但是已經(jīng)幾乎讀完了可以說

它的本質(zhì)就是,在宏任務(wù)中,stageQueue 作為低優(yōu)先級(jí)任務(wù)隊(duì)列,不斷的出棧,然后分批次(16ms 的閾值)入棧到 commitQueue 里

呼,其實(shí)如果不是寫文章,就可以到此為止了,但是寫文章為了湊字?jǐn)?shù)嘛,我們繼續(xù)

上面我們已經(jīng)知道了兩個(gè)隊(duì)列,stageQueue 和 commitQueue,但是并不知道他們里面都是什么東西

是什么東西被調(diào)度的呢?打印一下,你就知道:

console.log(stageQueue,commitQueue)

得出的結(jié)果是

function mountComponentInstance(){...}

看名字就知道是組件掛載函數(shù),當(dāng)然組件更新和卸載的函數(shù)也是同理

到現(xiàn)在,我們也知道了參與調(diào)度的是組件掛載更新的函數(shù),所以本質(zhì)上,vue 的時(shí)間切片的基本單位是組件,也就是說,如果你的組件掛載需要一個(gè)小時(shí),那你仍然要卡一小時(shí)

湊字?jǐn)?shù)

剩下的內(nèi)容純屬湊字?jǐn)?shù),就是除了核心調(diào)度之外的東西

比如 commitQueue 是操作 dom 的,那它咋個(gè)操作

function commitJob(job) {
  const { ops, postEffects } = job;
  for (let i = 0; i < ops.length; i++) {
    applyOp(ops[i]); // 重點(diǎn)在這里
  }
  if (postEffects) {
    postEffectsQueue.push(...postEffects);
  }
  resetJob(job);
  job.status = 0;
}

如上,拿到 ops,然后進(jìn)行操作,我們看一下 ops 是啥就行了

[
,
  • , function CreactElement(){}]

    湊合湊合,是個(gè)數(shù)組,包含了 dom 操作的方法和被操作的元素

    然后這個(gè)過程是同步完成的,也就是所謂的高優(yōu)先級(jí)任務(wù),必須等到徹底收集完畢,才可以循環(huán)執(zhí)行它

    做完這個(gè),postEffectQueue 主要是一些額外的副作用和清理工作,我實(shí)在湊字?jǐn)?shù)無能,就不打印了

    總結(jié)

    最后我們用最直白的話,總結(jié)一下:

    在宏任務(wù)隊(duì)列中,不斷的從 stageQueue 分批次(16ms)將組件的函數(shù)轉(zhuǎn)移到 commitQueue 里,轉(zhuǎn)移完了,同步操作 dom

    原理其實(shí)還是利用了宏任務(wù)隊(duì)列,其實(shí)現(xiàn)在 vue 的做法和 fre 也有一點(diǎn)點(diǎn)類似,fre 是在宏任務(wù)中,盡可能更多的去訪問 reconcile 大循環(huán)

    關(guān)于廢除

    如開頭提到的,time slicing 這部分內(nèi)容已經(jīng)在 master 分支被移除了,關(guān)于為什么廢除,我特地發(fā)了 issue,可以戳這里:(天啊,我和尤終于可以和平地進(jìn)行交談了)

    https://github.com/vuejs/rfcs/issues/89

    簡(jiǎn)單說,就是 time slicing 的收益不大,除了 issue 中提到的,它本身的場(chǎng)景就少的可憐

    也因?yàn)?vue 現(xiàn)在的實(shí)現(xiàn),由于調(diào)度的基本單位是組件,所以它仍然會(huì)因?yàn)榻M件內(nèi)部的邏輯而被阻斷

    比如我把用例中用于阻斷的 block 函數(shù)改為 1s,就已經(jīng)徹底卡死了

    思考

    從 issue 和源碼本身,我們可以思考一些問題,同時(shí)用來湊字?jǐn)?shù)

    時(shí)間切片是否必須?

    答案是否定的,尤的回復(fù)已經(jīng)足夠充分了:https://github.com/vuejs/rfcs/issues/89#issuecomment-546988615

    大致有兩點(diǎn):

    • 除了高幀率動(dòng)畫,其他的場(chǎng)景幾乎都可以使用防抖和節(jié)流去提高響應(yīng)性能
    • vue 現(xiàn)在的實(shí)現(xiàn),粒度太大,最終的效果十分有限,不值得

    那,fre 呢?

    fre 的異步渲染,是否也存在這個(gè)問題,不得不承認(rèn),fre 雖然粒度很小,對(duì)于組件內(nèi)部的阻斷可以搞定,但是元素本身也可以被阻斷

    而且第一個(gè)問題也是存在的,就是沒有太多適用場(chǎng)景

    但是 fre 源碼層面還是意義重大的,即便這玩意搞出來,發(fā)現(xiàn)它作用不大,副作用不小,但 fre 作為我個(gè)人的學(xué)習(xí)和研究的項(xiàng)目,它的價(jià)值從來就不是業(yè)務(wù)層面的

    只是我應(yīng)該停下來,異步渲染搞定了,只是向大家展示它的源碼實(shí)現(xiàn),未來不應(yīng)該跟隨 react 去搞一堆業(yè)務(wù) API,如 useTransition 等等

    關(guān)于源碼?

    vue3 發(fā)版當(dāng)天,源碼解讀就放出了,但是到目前為止,所有的源碼解讀統(tǒng)統(tǒng)都是蹭熱度的
    不久的將來,vue 的源碼又要爛大街了……
    這種現(xiàn)象引起反省,我們讀源碼到底是為了什么?為了面試嗎?為了更好的寫業(yè)務(wù)?
    對(duì)我而言,僅僅只是感興趣,我對(duì)這部分源碼感興趣,我就去讀,并且只讀感興趣的部分
    其實(shí)大家也看到了,我很少寫源碼解讀的文章,因?yàn)槲乙恢狈磳?duì)所謂的【通讀源碼】
    將閱讀源碼作為一項(xiàng)工作,同樣的小函數(shù),讀了一遍又一遍,重復(fù)勞動(dòng)
    這和糊 shi 有什么區(qū)別呢?

    以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。


    本文標(biāo)題:vue3源碼解讀之timeslicing的使用方法
    網(wǎng)址分享:http://weahome.cn/article/igiepp.html

    其他資訊

    在線咨詢

    微信咨詢

    電話咨詢

    028-86922220(工作日)

    18980820575(7×24)

    提交需求

    返回頂部