Maison > interface Web > Voir.js > Comparaison des priorités Vue : pourquoi v-for est supérieur à v-if

Comparaison des priorités Vue : pourquoi v-for est supérieur à v-if

青灯夜游
Libérer: 2023-03-14 19:45:02
avant
1705 Les gens l'ont consulté

Pourquoi v-for a une priorité plus élevée que v-if dans vue ? L'article suivant répondra à cette question en analysant le code source. J'espère qu'il sera utile à tout le monde !

Comparaison des priorités Vue : pourquoi v-for est supérieur à v-if

Parfois, dans certaines interviews, nous demandons souvent qui a la plus haute priorité : v-for ou v-if. Ici, nous répondrons à cette question en analysant la source. question codée. v-forv-if谁的优先级高,这里就通过分析源码去解答一下这个问题。

下面的内容是在 当我们谈及v-model,我们在讨论什么?的基础上分析的,所以阅读下面内容之前可先看这篇文章。

继续从编译出发

以下面的例子出发分析:

new Vue({
    el:'#app',
    template:`
        <ul>
            <li v-for="(item,index) in items" v-if="index!==0">
                {{item}}
            </li>
        </ul>
    `
})
Copier après la connexion

从上篇文章可以知道,编译有三个步骤

  • parse : 解析模板字符串生成 AST语法树
  • optimize : 优化语法树,主要时标记静态节点,提高更新页面的性能 【相关推荐:vuejs视频教程web前端开发
  • codegen : 生成js代码,主要是render函数和staticRenderFns函数

我们再次顺着这三个步骤对上述例子进行分析。

parse

parse过程中,会对模板使用大量的正则表达式去进行解析。开头的例子会被解析成以下AST节点:

// 其实ast有很多属性,我这里只展示涉及到分析的属性
ast = {
  &#39;type&#39;: 1,
  &#39;tag&#39;: &#39;ul&#39;,
  &#39;attrsList&#39;: [],
  attrsMap: {},
  &#39;children&#39;: [{
    &#39;type&#39;: 1,
    &#39;tag&#39;: &#39;li&#39;,
    &#39;attrsList&#39;: [],
    &#39;attrsMap&#39;: {
      &#39;v-for&#39;: &#39;(item,index) in data&#39;,
      &#39;v-if&#39;: &#39;index!==0&#39;
     },
     // v-if解析出来的属性
    &#39;if&#39;: &#39;index!==0&#39;,
    &#39;ifConditions&#39;: [{
      &#39;exp&#39;: &#39;index!==0&#39;,
      &#39;block&#39;: // 指向el自身
    }],
    // v-for解析出来的属性
    &#39;for&#39;: &#39;items&#39;,
    &#39;alias&#39;: &#39;item&#39;,
    &#39;iterator1&#39;: &#39;index&#39;,

    &#39;parent&#39;: // 指向其父节点
    &#39;children&#39;: [
      &#39;type&#39;: 2,
      &#39;expression&#39;: &#39;_s(item)&#39;
      &#39;text&#39;: &#39;{{item}}&#39;,
      &#39;tokens&#39;: [
        {&#39;@binding&#39;:&#39;item&#39;},
      ]
    ]
  }]
}
Copier après la connexion

对于v-for指令,除了记录在attrsMapattrsList,还会新增for(对应要遍历的对象或数组),aliasiterator1,iterator2对应v-for指令绑定内容中的第一,第二,第三个参数,开头的例子没有第三个参数,因此没有iterator2属性。

对于v-if指令,把v-if指令中绑定的内容取出放在if中,与此同时初始化ifConditions属性为数组,然后往里面存放对象:{exp,block},其中exp存放v-if指令中绑定的内容,block指向el

optimize 过程在此不做分析,因为本例子没有静态节点。

codegen

上一篇文章从const code = generate(ast, options)开始分析过其生成代码的过程,generate内部会调用genElement用来解析el,也就是AST语法树。我们来看一下genElement的源码:

export function genElement (el: ASTElement, state: CodegenState): string {
  if (el.parent) {
    el.pre = el.pre || el.parent.pre
  }

  if (el.staticRoot && !el.staticProcessed) {
    return genStatic(el, state)
  } else if (el.once && !el.onceProcessed) {
    return genOnce(el, state)
  // 其实从此处可以初步知道为什么v-for优先级比v-if高,
  // 因为解析ast树生成渲染函数代码时,会先解析ast树中涉及到v-for的属性
  // 然后再解析ast树中涉及到v-if的属性
  // 而且genFor在会把el.forProcessed置为true,防止重复解析v-for相关属性
  } else if (el.for && !el.forProcessed) {
    return genFor(el, state)
  } else if (el.if && !el.ifProcessed) {
    return genIf(el, state)

  } else if (el.tag === &#39;template&#39; && !el.slotTarget && !state.pre) {
    return genChildren(el, state) || &#39;void 0&#39;
  } else if (el.tag === &#39;slot&#39;) {
    return genSlot(el, state)
  } else {
    // component or element
    let code
    if (el.component) {
      code = genComponent(el.component, el, state)
    } else {
      let data
      if (!el.plain || (el.pre && state.maybeComponent(el))) {
        data = genData(el, state)
      }

      const children = el.inlineTemplate ? null : genChildren(el, state, true)
      code = `_c(&#39;${el.tag}&#39;${        data ? `,${data}` : &#39;&#39; // data      }${        children ? `,${children}` : &#39;&#39; // children      })`
    }
    // module transforms
    for (let i = 0; i < state.transforms.length; i++) {
      code = state.transforms[i](el, code)
    }
    return code
  }
}
Copier après la connexion

接下来依次看看genForgenIf的函数源码:

export function genFor (el, state , altGen, altHelper) {
  const exp = el.for
  const alias = el.alias
  const iterator1 = el.iterator1 ? `,${el.iterator1}` : &#39;&#39;
  const iterator2 = el.iterator2 ? `,${el.iterator2}` : &#39;&#39;

  el.forProcessed = true // avoid recursion
  return `${altHelper || &#39;_l&#39;}((${exp}),` + 
    `function(${alias}${iterator1}${iterator2}){` +
      `return ${(altGen || genElement)(el, state)}` + //递归调用genElement
    &#39;})&#39;
}
Copier après la connexion

在我们的例子里,当他处理liast树时,会先调用genElement,处理到for属性时,此时forProcessed为虚值,此时调用genFor处理li树中的v-for相关的属性。然后再调用genElement处理li树,此时因为forProcessedgenFor中已被标记为true。因此genFor不会被执行,继而执行genIf处理与v-if相关的属性。

export function genIf (el,state,altGen,altEmpty) {
  el.ifProcessed = true // avoid recursion
  // 调用genIfConditions主要处理el.ifConditions属性
  return genIfConditions(el.ifConditions.slice(), state, altGen, altEmpty)
}

function genIfConditions (conditions, state, altGen, altEmpty) {
  if (!conditions.length) {
    return altEmpty || &#39;_e()&#39; // _e用于生成空VNode
  }

  const condition = conditions.shift()
  if (condition.exp) { //condition.exp即v-if绑定值,例子中则为&#39;index!==0&#39;
    // 生成一段带三目运算符的js代码字符串
    return `(${condition.exp})?${       genTernaryExp(condition.block)    }:${      genIfConditions(conditions, state, altGen, altEmpty)    }`
  } else {
    return `${genTernaryExp(condition.block)}`
  }

  // v-if with v-once should generate code like (a)?_m(0):_m(1)
  function genTernaryExp (el) {
    return altGen
      ? altGen(el, state)
      : el.once
        ? genOnce(el, state)
        : genElement(el, state)
  }
}
Copier après la connexion

参考 前端进阶面试题详细解答

最后,经过codegen生成的js代码如下:

function render() {
  with(this) {
    return _c(&#39;ul&#39;, _l((items), function (item, index) {
      return (index !== 0) ? _c(&#39;li&#39;) : _e()
    }), 0)
  }
}
Copier après la connexion

其中:

  • _c: 调用 createElement 去创建 VNode

  • _l: renderList函数,主要用来渲染列表

  • _e: createEmptyVNode函数,主要用来创建空VNode

总结

为什么v-for的优先级比v-if的高?总结来说是编译有三个过程,parse->optimize->codegen。在codegen过程中,会先解析AST树中的与v-for相关的属性,再解析与v-if相关的属性。除此之外,也可以知道Vuev-forv-if

Le contenu suivant est analysé sur la base de Quand on parle de v-model, de quoi parle-t-on ?, vous pouvez donc lire cet article avant de lire le contenu suivant.

Continuez pour commencer la compilation

Démarrez l'analyse avec l'exemple suivant : rrreeeComme vous pouvez le savoir grâce à l'article précédent, la compilation comporte trois étapes
  • parse : analyse la chaîne du modèle pour générer l'arbre de syntaxe AST
  • optimize : optimise la syntaxe arbre, marquant principalement les nœuds statiques. Améliorer les performances des pages mises à jour [Recommandations associées : tutoriel vidéo vuejs, Développement front-end Web🎜]
  • codegen : Générez du code js, principalement la fonction de rendu et la fonction staticRenderFns
🎜Nous analyserons l'exemple ci-dessus en suivant à nouveau ces trois étapes. 🎜

parse

🎜parseDans le processus, un grand nombre d'expressions régulières seront utilisées pour analyser le modèle. L'exemple du début sera analysé dans les nœuds AST suivants : 🎜rrreee🎜Pour la directive v-for, en plus des enregistrements dans attrsMap et attrsList , et ajoutera également for (correspondant à l'objet ou au tableau à parcourir), alias, iterator1 code>, <code>iterator2 correspond aux premier, deuxième et troisième paramètres du contenu de liaison de l'instruction <code>v-for L'exemple du début n'a pas le troisième paramètre, donc. il n'y a pas de propriété iterator2. 🎜🎜Pour l'instruction v-if, retirez le contenu lié dans l'instruction v-if et mettez-le dans if, et à la en même temps initializeifConditions est un tableau, puis l'objet y est stocké : {exp,block}, où exp stocke v-if Le contenu lié dans l'instruction, block pointe vers el. 🎜🎜optimiser Le processus n'est pas analysé ici car il n'y a pas de nœuds statiques dans cet exemple. 🎜

codegen

🎜L'article précédent a été analysé à partir de const code = generate(ast, options) In le processus de génération de code, generate appelle en interne genElement pour analyser el, qui est l'arbre syntaxique AST. Jetons un coup d'œil au code source de genElement : 🎜rrreee🎜 Ensuite, jetez un œil au code source des fonctions de genFor et genIf : 🎜rrreee🎜Dans notre Dans l'exemple, lorsqu'il traitera l'arbre ast de li, il appellera d'abord genElement puis traitera le for , à ce moment forProcessed est une valeur virtuelle, et à ce moment appelez genFor pour traiter le v-for. > lié ​​dans la propriété de l'arborescence li. Appelez ensuite genElement pour traiter l'arborescence li, car forProcessed a été marqué comme dans <code>genFor true. Par conséquent, genFor ne sera pas exécuté, puis genIf sera exécuté pour traiter les attributs liés à v-if. 🎜rrreee🎜RéférenceRéponses détaillées aux questions avancées d'entretien frontal🎜🎜🎜Finalement, le code js généré par codegen est le suivant : 🎜rrreee🎜Parmi eux : 🎜
  • 🎜_c : Appelez createElement pour créer un VNode🎜
  • 🎜 _l : fonction renderList, principalement utilisée pour rendre des listes🎜
  • 🎜_e : fonction createEmptyVNode, principalement utilisé pour créer un VNode
vide 🎜

Résumé

🎜Pourquoi v-for a-t-il une priorité plus élevée que v -si? En résumé, il y a trois processus en compilation, parse->optimize->codegen. Dans le processus codegen, les attributs liés à v-for dans l'arborescence AST seront d'abord analysés, puis les attributs liés à v-if Attributs associés. De plus, vous pouvez également savoir comment Vue gère v-for et v-if. 🎜🎜 (Partage de vidéos d'apprentissage : 🎜Tutoriel d'introduction à vuejs🎜, 🎜Vidéo de programmation de base🎜)🎜

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Étiquettes associées:
source:juejin.cn
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal