Heim > Web-Frontend > View.js > Hauptteil

Teilen nützlicher Informationen, die Ihnen helfen, Vue.nextTick in Vue zu verstehen

青灯夜游
Freigeben: 2022-03-22 11:38:00
nach vorne
2265 Leute haben es durchsucht

Dieser Artikel wird mit Ihnen VueReine Informationen teilen und Ihnen Vue.nextTick vorstellen, die Sie nicht kennen. Ich hoffe, dass es für alle hilfreich ist!

Teilen nützlicher Informationen, die Ihnen helfen, Vue.nextTick in Vue zu verstehen

Freunde, die Vue mehr oder weniger verwendet haben, kennen $nextTick~ Bevor Sie nextTick offiziell erklären, sollten Sie meiner Meinung nach klar wissen, dass Vue bei der Aktualisierung asynchron ist DOM. Code>, da der nächste Erklärungsprozess zusammen mit Komponentenaktualisierungen erläutert wird ~ Kommen wir ohne weitere Umschweife direkt zum Thema (in diesem Artikel wird der Vue-Quellcode der Version v2.6.14$nextTick~ 在正式讲解nextTick之前,我想你应该清楚知道 Vue 在更新 DOM 时是异步执行的,因为接下来讲解过程会结合组件更新一起讲~ 事不宜迟,我们直进主题吧(本文以v2.6.14版本的Vue源码进行讲解)【相关推荐:vuejs视频教程

一、nextTick小测试

你真的了解nextTick吗?来,直接上题~

<template>
  <div id="app">
    <p ref="name">{{ name }}</p>
    <button @click="handleClick">修改name</button>
  </div>
</template>

<script>
  export default {
  name: &#39;App&#39;,
  data () {
    return {
      name: &#39;井柏然&#39;
    }
  },
  mounted() {
    console.log(&#39;mounted&#39;, this.$refs.name.innerText)
  },
  methods: {
    handleClick () {
      this.$nextTick(() => console.log(&#39;nextTick1&#39;, this.$refs.name.innerText))
      this.name = &#39;jngboran&#39;
      console.log(&#39;sync log&#39;, this.$refs.name.innerText)
      this.$nextTick(() => console.log(&#39;nextTick2&#39;, this.$refs.name.innerText))
    }
  }
}
</script>
Nach dem Login kopieren

请问上述代码中,当点击按钮“修改name”时,&#39;nextTick1&#39;&#39;sync log&#39;&#39;nextTick2&#39;对应的this.$refs.name.innerText分别会输出什么?注意,这里打印的是DOM的innerText~(文章结尾处会贴出答案)

如果此时的你有非常坚定的答案,那你可以不用继续往下看了~但如果你对自己的答案有所顾虑,那不如跟着我,接着往下看。相信你看完,不需要看到答案都能有个肯定的答案了~!


二、nextTick源码实现

源码位于core/util/next-tick中。可以将其分为4个部分来看,直接上代码

1. 全局变量

callbacks队列、pending状态

const callbacks = [] // 存放cb的队列
let pending = false // 是否马上遍历队列,执行cb的标志
Nach dem Login kopieren

2. flushCallbacks

遍历callbacks执行每个cb

function flushCallbacks () {
  pending = false // 注意这里,一旦执行,pending马上被重置为false
  const copies = callbacks.slice(0)
  callbacks.length = 0
  for (let i = 0; i < copies.length; i++) {
    copies[i]() // 执行每个cb
  }
}
Nach dem Login kopieren

3. nextTick的异步实现

根据执行环境的支持程度采用不同的异步实现策略

let timerFunc // nextTick异步实现fn

if (typeof Promise !== &#39;undefined&#39; && isNative(Promise)) {
  // Promise方案
  const p = Promise.resolve()
  timerFunc = () => {
    p.then(flushCallbacks) // 将flushCallbacks包装进Promise.then中
  }
  isUsingMicroTask = true
} else if (!isIE && typeof MutationObserver !== &#39;undefined&#39; && (
  isNative(MutationObserver) ||
  MutationObserver.toString() === &#39;[object MutationObserverConstructor]&#39;
)) {
  // MutationObserver方案
  let counter = 1
  const observer = new MutationObserver(flushCallbacks) // 将flushCallbacks作为观测变化的cb
  const textNode = document.createTextNode(String(counter)) // 创建文本节点
  // 观测文本节点变化
  observer.observe(textNode, {
    characterData: true
  })
  // timerFunc改变文本节点的data,以触发观测的回调flushCallbacks
  timerFunc = () => { 
    counter = (counter + 1) % 2
    textNode.data = String(counter)
  }
  isUsingMicroTask = true
} else if (typeof setImmediate !== &#39;undefined&#39; && isNative(setImmediate)) {
  // setImmediate方案
  timerFunc = () => {
    setImmediate(flushCallbacks)
  }
} else {
  // 最终降级方案setTimeout
  timerFunc = () => {
    setTimeout(flushCallbacks, 0)
  }
}
Nach dem Login kopieren
  • 这里用个真实案例加深对MutationObserver的理解。毕竟比起其他三种异步方案,这个应该是大家最陌生的
    const observer = new MutationObserver(() => console.log(&#39;观测到文本节点变化&#39;))
    const textNode = document.createTextNode(String(1))
    observer.observe(textNode, {
        characterData: true
    })
    
    console.log(&#39;script start&#39;)
    setTimeout(() => console.log(&#39;timeout1&#39;))
    textNode.data = String(2) // 这里对文本节点进行值的修改
    console.log(&#39;script end&#39;)
    Nach dem Login kopieren
  • 知道对应的输出会是怎么样的吗?
    • script startscript end会在第一轮宏任务中执行,这点没问题

    • setTimeout会被放入下一轮宏任务执行

    • MutationObserver是微任务,所以会在本轮宏任务后执行,所以先于setTimeout

  • 结果如下图:
    Teilen nützlicher Informationen, die Ihnen helfen, Vue.nextTick in Vue zu verstehen

4. nextTick方法实现

cbPromise方式

export function nextTick (cb?: Function, ctx?: Object) {
  let _resolve
  // 往全局的callbacks队列中添加cb
  callbacks.push(() => {
    if (cb) {
      try {
        cb.call(ctx)
      } catch (e) {
        handleError(e, ctx, &#39;nextTick&#39;)
      }
    } else if (_resolve) {
      // 这里是支持Promise的写法
      _resolve(ctx)
    }
  })
  if (!pending) {
    pending = true
    // 执行timerFunc,在下一个Tick中执行callbacks中的所有cb
    timerFunc()
  }
  // 对Promise的实现,这也是我们使用时可以写成nextTick.then的原因
  if (!cb && typeof Promise !== &#39;undefined&#39;) {
    return new Promise(resolve => {
      _resolve = resolve
    })
  }
}
Nach dem Login kopieren
  • 深入细节,理解pending有什么用,如何运作?

案例1,同一轮Tick中执行2次$nextTicktimerFunc只会被执行一次

this.$nextTick(() => console.log(&#39;nextTick1&#39;))
this.$nextTick(() => console.log(&#39;nextTick2&#39;))
Nach dem Login kopieren
  • 用图看看更直观?

Teilen nützlicher Informationen, die Ihnen helfen, Vue.nextTick in Vue zu verstehen


三、Vue组件的异步更新

这里如果有对Vue组件化派发更新不是十分了解的朋友,可以先戳这里,看图解Vue响应式原理了解下Vue组件化和派发更新的相关内容再回来看噢~

Vue的异步更新DOM其实也是使用nextTick来实现的,跟我们平时使用的$nextTick其实是同一个~

这里我们回顾一下,当我们改变一个属性值的时候会发生什么?

Teilen nützlicher Informationen, die Ihnen helfen, Vue.nextTick in Vue zu verstehen

根据上图派发更新过程,我们从watcher.update开时讲起,以渲染Watcher为例,进入到queueWatcher

1. queueWatcher做了什么?

// 用来存放Wathcer的队列。注意,不要跟nextTick的callbacks搞混了,都是队列,但用处不同~
const queue: Array<Watcher> = []

function queueWatcher (watcher: Watcher) {
  const id = watcher.id // 拿到Wathcer的id,这个id每个watcher都有且全局唯一
  if (has[id] == null) {
    // 避免添加重复wathcer,这也是异步渲染的优化做法
    has[id] = true
    if (!flushing) {
      queue.push(watcher)
    }
    if (!waiting) {
      waiting = true
      // 这里把flushSchedulerQueue推进nextTick的callbacks队列中
      nextTick(flushSchedulerQueue)
    }
  }
}
Nach dem Login kopieren

2. flushSchedulerQueue zur Erklärung verwendet) [ Verwandte Empfehlungen: vuejs Video-Tutorial

< h2 data-id=" heading-0">1. NextTick-Quiz
  • Verstehen Sie nextTick wirklich? Komm schon, geh direkt zur Frage~
  • function flushSchedulerQueue () {
      currentFlushTimestamp = getNow()
      flushing = true
      let watcher, id
      // 排序保证先父后子执行更新,保证userWatcher在渲染Watcher前
      queue.sort((a, b) => a.id - b.id)
      // 遍历所有的需要派发更新的Watcher执行更新
      for (index = 0; index < queue.length; index++) {
        watcher = queue[index]
        id = watcher.id
        has[id] = null
        // 真正执行派发更新,render -> update -> patch
        watcher.run()
      }
    }
    Nach dem Login kopieren
Im obigen Code, wenn auf die

-Schaltfläche „Name ändern“ geklickt wird, 'nextTick1', 'sync log'</ code>, Was wird durch den entsprechenden <code>this.$refs.name.innerText von 'nextTick2' ausgegeben? HinweisTeilen nützlicher Informationen, die Ihnen helfen, Vue.nextTick in Vue zu verstehen: Was hier gedruckt wird, ist der innere Text des DOM~ (die Antwort wird am Ende des Artikels veröffentlicht)


Wenn Sie zu diesem Zeitpunkt eine sehr eindeutige Antwort haben, ist dies nicht erforderlich Lesen Sie weiter ~ Aber wenn Sie Recht haben. Wenn Sie sich über Ihre Antwort Sorgen machen, folgen Sie mir doch und lesen Sie weiter. Ich glaube, dass Sie nach dem Lesen eine eindeutige Antwort haben werden, ohne die Antwort sehen zu müssen~!

2. NextTick-Quellcode-Implementierung

Der Quellcode befindet sich in core/util/next-tick. Es kann in 4 Teile unterteilt werden und direkt zum Code gehen🎜

Globale Variablen🎜🎜🎜Rückrufe-Warteschlange, ausstehend Status 🎜
this.$nextTick(() => console.log(&#39;nextTick1&#39;, this.$refs.name.innerText))
this.name = &#39;jngboran&#39;
console.log(&#39;sync log&#39;, this.$refs.name.innerText)
this.$nextTick(() => console.log(&#39;nextTick2&#39;, this.$refs.name.innerText))
Nach dem Login kopieren
Nach dem Login kopieren
🎜

2. flushCallbacks🎜🎜🎜Traverse Callbacks, um jeden cb auszuführen🎜rrreee🎜

3. Asynchrone Implementierung von nextTick🎜🎜🎜Verschiedene asynchrone Implementierungsstrategien entsprechend der Unterstützungsstufe der Ausführungsumgebung anwenden🎜rrreee🎜🎜Hier ist ein echter Fall Vertiefen Sie das Verständnis von MutationObserver. Im Vergleich zu den anderen drei asynchronen Lösungen dürfte diese für jeden die unbekannteste sein. Wissen Sie, wie die entsprechende Ausgabe aussehen wird?

Verwandte Etiketten:
vue
Quelle:juejin.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage