经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » HTML/CSS » HTML5 » 查看文章
Vue.js 源码分析(十八) 指令篇 v-for 指令详解
来源:cnblogs  作者:大沙漠  时间:2019/7/5 9:13:06  对本文有异议

我们可以用 v-for 指令基于一个数组or对象来渲染一个列表,有五种使用方法,如下:

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>Document</title>
  6. <script src="https://cdn.jsdelivr.net/npm/vue@2.5.16/dist/vue.js"></script>
  7. </head>
  8. <body>
  9. <script>
  10. Vue.config.productionTip=false;
  11. Vue.config.devtools=false;
  12. </script>
  13. <div id="app">
  14. <p v-for="item in items">{{item}}</p> <!--数组格式一,渲染结果:<p>11</p><p>12</p> -->
  15. <p v-for="(item,index) in items">{{index}}->{{item}}</p> <!--数组格式二,渲染结果:<p>0->11</p><p>1->12</p>-->
  16. <p v-for="item in infos">{{item}}</p> <!--对象格式一,渲染结果:<p>gege</p><p>12</p>-->
  17. <p v-for="(item,key) in infos">{{key}}:{{item}}</p> <!--对象格式二,渲染结果:<p>name:gege</p><p>age:12</p>-->
  18. <p v-for="(item,key,index) in infos">{{index}}:{{key}}:{{item}}</p> <!--对象格式三,渲染结果:<p>0:name:gege</p><p>1:age:12</p>-->
  19. </div>
  20. <script>
  21. var app = new Vue({
  22. data(){
  23. return {
  24. items:[11,12], //v-for可以是个对象
  25. infos:{name:'gege',age:12} //也可以是个数组
  26. }
  27. },
  28. el:'#app'
  29. })
  30. </script>
  31. </body>
  32. </html>

 挺简单的,后台只要提供一个接口,返回一个数组或对象,前端通过v-for就可以渲染了,我们以上面对象的第三个格式为例讲一下源码,如下:

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>Document</title>
  6. <script src="https://cdn.jsdelivr.net/npm/vue@2.5.16/dist/vue.js"></script>
  7. </head>
  8. <body>
  9. <script>
  10. Vue.config.productionTip=false;
  11. Vue.config.devtools=false;
  12. </script>
  13. <div id="app">
  14. <p v-for="(item,key,index) in infos">{{index}}:{{key}}:{{item}}</p>
  15. </div>
  16. <script>
  17. var app = new Vue({
  18. data(){return {infos:{name:'gege',age:12}}},
  19. el:'#app'
  20. })
  21. </script>
  22. </body>
  23. </html>

 

源码分析


 在解析模板的时候,Vue的processFor()->parseFor()函数会根据v-for内容的不同解析出这四个变量,保存到AST对象的属性上:

  1. function processFor (el) { //第9367行 处理for指令
  2. var exp;
  3. if ((exp = getAndRemoveAttr(el, 'v-for'))) { //如果获取了v-for属性
  4. var res = parseFor(exp); //调用parseFor函数解析该属性
  5. if (res) { //如果res存在
  6. extend(el, res); //则调用extend()添加AST对象上
  7. } else {
  8. warn$2(
  9. ("Invalid v-for expression: " + exp)
  10. );
  11. }
  12. }
  13. }

parseFor()用于解析v-for的值,返回一个对象,如下:

  1. function parseFor (exp) { //第9383行 解析v-for属性 exp:v-for的值 ;例如:"(item,key,index) in infos"
  2. var inMatch = exp.match(forAliasRE); //用正则匹配 forAliasRE定义在9403行等于:/([^]*?)\s+(?:in|of)\s+([^]*)/;
  3. if (!inMatch) { return } //如果不能匹配,则返回false
  4. var res = {};
  5. res.for = inMatch[2].trim(); //for的值,这里等于:infos
  6. var alias = inMatch[1].trim().replace(stripParensRE, ''); //去除两边的括号,此时alias等于:item,key,index
  7. var iteratorMatch = alias.match(forIteratorRE); //匹配别名和索引
  8. if (iteratorMatch) { //如果匹配到了,即是这个格式:v-for="(item,index) in data"
  9. res.alias = alias.replace(forIteratorRE, ''); //获取别名
  10. res.iterator1 = iteratorMatch[1].trim(); //获取索引
  11. if (iteratorMatch[2]) {
  12. res.iterator2 = iteratorMatch[2].trim();
  13. }
  14. } else {
  15. res.alias = alias;
  16. }
  17. return res //返回对象,比如:{alias: "item",for: "infos",iterator1: "key",iterator2: "index"}
  18. }

 执行到这里后例子里的v-for保存了四个属性与v-for相关,如下:

接下来在generate生成rendre函数的时候会调用genFor()生成对应的_l函数,如下:

  1. function genFor ( //渲染v-for指令
  2. el,
  3. state,
  4. altGen,
  5. altHelper
  6. ) {
  7. var exp = el.for; //获取for的值
  8. var alias = el.alias; //获取别名
  9. var iterator1 = el.iterator1 ? ("," + (el.iterator1)) : ''; //获取索引(v-for的值为对象时则为key)
  10. var iterator2 = el.iterator2 ? ("," + (el.iterator2)) : ''; ///获取索引(v-for的值为对象时))
  11.  
  12. if ("development" !== 'production' &&
  13. state.maybeComponent(el) &&
  14. el.tag !== 'slot' &&
  15. el.tag !== 'template' &&
  16. !el.key
  17. ) {
  18. state.warn(
  19. "<" + (el.tag) + " v-for=\"" + alias + " in " + exp + "\">: component lists rendered with " +
  20. "v-for should have explicit keys. " +
  21. "See https://vuejs.org/guide/list.html#key for more info.",
  22. true /* tip */
  23. );
  24. }
  25. el.forProcessed = true; // avoid recursion
  26. return (altHelper || '_l') + "((" + exp + ")," + //拼凑_l函数
  27. "function(" + alias + iterator1 + iterator2 + "){" +
  28. "return " + ((altGen || genElement)(el, state)) +
  29. '})'
  30. }

最后生成的render函数等于:

  1. with(this){return _c('div',{attrs:{"id":"app"}},_l((infos),function(item,key,index){return _c('p',[_v(_s(index)+":"+_s(key)+":"+_s(item))])}))}

其中与v-for相关的如下:

  1. _l((infos),function(item,key,index){return _c('p',[_v(_s(index)+":"+_s(key)+":"+_s(item))

_l的第一个参数为我们的v-for的目标,也就是infos,一会儿会遍历该对象的

渲染生成VNode时就会执行Vue内部的_l函数,也就是全局的renderList,如下:

  1. function renderList ( //第3691行 渲染v-for指令
  2. val,
  3. render
  4. ) {
  5. var ret, i, l, keys, key;
  6. if (Array.isArray(val) || typeof val === 'string') { //如果val是个数组
  7. ret = new Array(val.length); //将ret定义成val一样大小的数组
  8. for (i = 0, l = val.length; i < l; i++) { //遍历val数组
  9. ret[i] = render(val[i], i); //依次调用render函数,参数1为值 参数2为索引 返回VNode,并把结果VNode保存到ret里面
  10. }
  11. } else if (typeof val === 'number') {
  12. ret = new Array(val);
  13. for (i = 0; i < val; i++) {
  14. ret[i] = render(i + 1, i);
  15. }
  16. } else if (isObject(val)) {
  17. keys = Object.keys(val);
  18. ret = new Array(keys.length);
  19. for (i = 0, l = keys.length; i < l; i++) {
  20. key = keys[i];
  21. ret[i] = render(val[key], key, i);
  22. }
  23. }
  24. if (isDef(ret)) { //如果ret存在(成功调用了)
  25. (ret)._isVList = true; //则给该数组添加一个_isVList标记,值为true
  26. }
  27. return ret //最后返回ret
  28. }

最后一起打包成VNode数组并返回,作为其他元素的子节点(_c的第三个参数)。

原文链接:http://www.cnblogs.com/greatdesert/p/11128230.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号