深入理解Vue 的条件渲染和列表渲染


Posted in Javascript onSeptember 01, 2017

这两天学习了Vue.js 感觉条件渲染和列表渲染知识点挺多的,而且很重要,所以,今天添加一点小笔记。

条件渲染

v-if

在 < template > 中配合 v-if 渲染一整组

在使用 v-if 控制元素的时候,我们需要将它添加到这个元素上去。然而如果要切换很多元素的时候,一个个的添加就太麻烦了。这时候就可以使用 < template > 将一组元素进行包裹,并在上面使用 v-if。最终的渲染结果不会包含 < template > 元素。

<template v-if="ok">
 <h1>Title</h1>
 <p>Paragraph 1</p>
 <p>Paragraph 2</p>
</template>

<script>
  data:{
    ok:true
  }
</script>

我们更改 ok 的值,就可以控制整组的元素了

v-else

你可以使用 v-else 指令来表示 v-if 的“else 块”:

<div v-if="ok">
  Now you see me
</div>
<div v-else>
  Now you don't
</div>

v-else 元素必须紧跟在 v-if 或者 v-else-if 元素的后面——否则它将不会被识别。

v-else-if

v-else-if,顾名思义,充当 v-if 的“else-if 块”。可以链式地使用多次:

<div>
  <p v-if="sc>=90">优秀</p>
  <p v-else-if="sc>=60">及格</p>
  <p v-else="sc<60">不及格</p>
</div>

类似于 v-else,v-else-if 必须紧跟在 v-if 或者 v-else-if 元素之后。

可复用元素

Vue 会尽可能高效地渲染元素,通常会复用已有元素而不是从头开始渲染。这么做,除了使 Vue 变得非常快之外,还有一些有用的好处。例如,如果你允许用户在不同的登录方式之间切换:

<div class="exp">
  <template v-if="loginType === 'username'">
    <label>Username</label>
    <input placeholder="Enter your username">
  </template>

  <template v-else>
    <label>Email</label>
    <input placeholder="Enter your email address">
  </template>
  <input type="button" @click="btn" value="切换"/>
</div>

<script>
  var exp=new Vue({
    el:".exp",
    data:{
      loginType:"username"
    },
    methods:{
      btn:function(){
        if(this.loginType==="username"){
          this.loginType="email"
        }else{
          this.loginType="username"
        }
      }
    }
  })
</script>

那么在上面的代码中切换 loginType 将不会清除用户已经输入的内容。因为两个模板使用了相同的元素,< input > 不会被替换掉——仅仅是替换了它的 placeholder。

复制上面的代码,在自己的浏览器中试一试。

有时候我们不希望浏览器保留我们输入的内容,所以 Vue 为你提供了一种方式来声明“这两个元素是完全独立的——不要复用它们”。只需添加一个具有唯一值的 key 属性即可:

<template v-if="loginType === 'username'">
  <label>Username</label>
  <input placeholder="Enter your username" key="username">
</template>

<template v-else>
  <label>Email</label>
  <input placeholder="Enter your email address" key="email">
</template>

v-show

另一个用于根据条件展示元素的选项是 v-show 指令。用法大致一样:

<h1 v-show="ok">Hello!</h1>
<script>
  data:{
    ok:false
  }
</script>

不同的是带有 v-show 的元素始终会被渲染并保留在 DOM 中。v-show 是简单地切换元素的 CSS 属性 display 。

渲染如下

<div style="display:none;"></div>

列表渲染

使用 v-for 把一个数组对应为一组元素

我们用 v-for 指令根据一组数组的选项列表进行渲染。 v-for 指令需要以 item in items 形式的特殊语法, items 是源数据数组并且 item 是数组元素迭代的别名。

<div class="exp">
  <ul>
    <li v-for="item in items">{{item.text}}</li>
  </ul>
</div>

<script>
  data:{
    items:[
      {text:"eat"},
      {text:"play"},
      {text:"game"}
    ]
  }
</script>

渲染结果

<div class="exp">
  <ul>
    <li>eat</li>
    <li>play</li>
    <li>game</li>
  </ul>
</div>

v-for 还支持一个可选的第二个参数为当前项的索引。

<div class="exp">
  <ul>
    <li v-for="item,index in items">{{index}}-{{item.text}}</li>
  </ul>
</div>
<script>
  var exp=new Vue({
    el:".exp",
    data:{
      items:[
        {text:'eat'},
        {text:'paly'},
        {text:'game'}
      ]  
    }
  })
</script>

结果

0-eat
1-paly
2-game

一个对象的 v-for

你也可以用 v-for 通过一个对象的属性来迭代。

<div class="exp">
  <ul>
    <li v-for="value in obj">{{value}}</li>
  </ul>
</div>
<script>
  var exp=new Vue({
    el:".exp",
    data:{
      obj:{
        firstname:"PureView",
        lastname:"一个安静的美男子",
        age:18
      }
    }
  })
</script>

结果

PureView
一个安静的美男子
18

你一共可以提供三个参数,第二个参数为键名,第三个为索引:

<li v-for="value,key,index in obj">{{index+1}}. {{key}}: {{value}}</li>

结果

1. firstname: PureView
2. lastname: 一个安静的美男子
3. age: 18

数组更新检测

变异方法

Vue 包含一组观察数组的变异方法,所以它们也将会触发视图更新。这些方法如下:

  • push()
  • pop()
  • shift()
  • unshift()
  • splice()
  • sort()
  • reverse()

例如

<div class="exp">
  <ul>
    <li v-for="item in items">{{item.text}}</li>
  </ul>
</div>

<script>
  var exp=new Vue({
    el:".exp",
    data:{
      items:[
        {text:"eat"},
        {text:"play"},
        {text:"game"}
      ]
  }
  })
  exp.items.push({text:'watch TV'})
</script>

重塑数组

变异方法(mutation method),顾名思义,会改变被这些方法调用的原始数组。相比之下,也有非变异(non-mutating method)方法,例如: filter(), concat() 和 slice() 。这些不会改变原始数组,但总是返回一个新数组。当使用非变异方法时,可以用新数组替换旧数组:

data:{
  items:[
      {text:"eat"},
      {text:"play"},
      {text:"game"},
      {text:"gaming"},
      {text:"wot"},
      {text:"wows"},
      {text:"wt"}
    ]
  }
}
exp.items.slice(0,5)

利用上一节的例子,返回的值不会改变原数据,在控制台打印我们就能看到了。

注意事项

由于 JavaScript 的限制, Vue 不能检测以下变动的数组:

  • 当你利用索引直接设置一个项时,例如: vm.items[indexOfItem] = newValue
  • 当你修改数组的长度时,例如: vm.items.length = newLength

为了解决第一类问题,以下两种方式都可以实现和 vm.items[indexOfItem] = newValue 相同的效果, 同时也将触发状态更新:

// Vue.set
Vue.set(exp.items, indexOfItem, newValue)
// Array.prototype.splice
exp.items.splice(indexOfItem, 1, newValue)

为了解决第二类问题,你可以使用 splice:

exp.items.splice(newLength)

对象更新检测

由于现代JavaScript的限制,Vue无法检测属性添加或删除。 例如:

var exp=new Vue({
  data:{
    a:1
  }
})
vm.b=2 //模板内无响应

Vue是不允许动态地向已创建的实例添加新的根级属性的。这时候 Vue 提供了一个方法用来对对象添加属性:

Vue.set(object,key,value)

举个例子

var exp=new Vue({
  el:".exp",
  data:{
    obj:{
      me:"pureview",
      pet1:"dog",
      pet2:"cat",
      hobby:"games"
    } 
  }
})

我们在控制台输入下面的代码,就可以看到模板内的数据进行了更新

Vue.set(exp.obj,"todo","eating")

除了添加属性,我们也可以进行删除操作

Vue.delete(exp.obj,"pet2")

显示过滤/排序结果

有时,我们想要显示一个数组的过滤或排序副本,而不实际改变或重置原始数据。在这种情况下,可以创建返回过滤或排序数组的计算属性。

比如我们在一个数组中取其偶数

<div class="exp">
  <ul>
    <li v-for="n in numbers">{{n}}</li>
  </ul>
</div>

<script>
  var exp=new Vue({
    el:".exp",
    data:{
      num:[1,2,3,4,5,6,7,8,9,10]
    },
    computed:{
      numbers:function(){
        return this.num.filter(function(num){
          return num%2===0
        })
      }
    }
  })
</script>

模板显示结果:

2
4
6
8
10

在计算属性不适用的情况下 (例如,在嵌套 v-for 循环中) 你可以使用一个 method 方法:

<div class="exp">
  <ul>
    <li v-for="n in even(num)">{{n}}</li>
  </ul>
</div>

<script>
  var exp=new Vue({
    el:".exp",
    data:{
      num:[1,2,3,4,5,6,7,8,9,10]
    },
    methods:{
      even:function(num){
        return num.filter(function(num){
          return num%2===0
        })
      }
    }
  })
</script>

结果是一样的

一段取值范围的 v-for

v-for 也可以取整数。在这种情况下,它将重复多次模板。

<div>
 <span v-for="n in 10">{{ n }} </span>
</div>

结果

1 2 3 4 5 6 7 8 9 10

v-for 在 < template > 上

与模板v-if类似,您还可以使用带有 v-for 的< template >标签来呈现多个元素的块。

<ul>
 <template v-for="item in items">
  <li>{{ item.msg }}</li>
  <li class="divider"></li>
 </template>
</ul>

v-for 和 v-if

当 v-for 与 v-if 一起使用时,v-for 具有比 v-if 更高的优先级,这意味着 v-if 将分别重复运行于每个 v-for 循环中。当我们仅为某些项目渲染节点时,这可能很有用:

<li v-for="todo in todos" v-if="!todo.isComplete">
 {{ todo }}
</li>

而如果我们的目的是有条件地跳过循环的执行,那么可以将 v-if 置于外层元素 (或 < template >)上。如:

<ul v-if="todos.length">
 <li v-for="todo in todos">
  {{ todo }}
 </li>
</ul>
<p v-else>No todos left!</p>

组件的 v-for

在 Vue 的 2.2.0 以上的版本中,我们要在组件中使用 v-for 时,不许使用 key

<my-component v-for="(item,index) in itmes" v-bind:key="index"></my-component>

虽然在自定义组件里可以使用 v-for ,但是,他不能自动传递数据到组件里,因为组件有自己独立的作用域。为了传递迭代数据到组件里,我们要用 props :

<my-component v-for="(item,index) in items" v-bind:key="index" :lie="item.text"l></my-component>

<script>
  Vue.component('mycom', {
    template: "<p>{{this.lie}}</p>",
    props:["lie"]
  })
  var exp=new Vue({
    el:".exp",
    data:{
      items:[
        {text:'从前有座山'},
        {text:'山上有座庙'},
        {text:'庙里有个老和尚'},
        {text:'正在玩王者荣耀'}
      ]
    }
  })
</script>

结果

从前有座山
山上有座庙
庙里有个老和尚
正在玩王者荣耀

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
基于jquery的从一个页面跳转到另一个页面的指定位置的实现代码(带平滑移动的效果)
May 24 Javascript
jQuery实现的一个自定义Placeholder属性插件
Aug 11 Javascript
jQuery中first()方法用法实例
Jan 06 Javascript
js计算德州扑克牌面值的方法
Mar 04 Javascript
js滑动提示效果代码分享
Mar 10 Javascript
Node.js的基本知识简单汇总
Sep 19 Javascript
基于jquery实现的银行卡号每隔4位自动插入空格的实现代码
Nov 22 Javascript
jQuery插件FusionCharts绘制的3D环饼图效果示例【附demo源码】
Apr 02 jQuery
Webpack实现按需打包Lodash的几种方法详解
May 08 Javascript
值得收藏的八个常用的js正则表达式
Oct 19 Javascript
js神秘的电报密码 哈弗曼编码实现
Sep 10 Javascript
JS实现的雪花飘落特效示例
Dec 03 Javascript
js禁止Backspace键使浏览器后退的实现方法
Sep 01 #Javascript
JavaScript中数组常见操作技巧
Sep 01 #Javascript
js禁止浏览器页面后退功能的实例(推荐)
Sep 01 #Javascript
使用JS和canvas实现gif动图的停止和播放代码
Sep 01 #Javascript
关于Promise 异步编程的实例讲解
Sep 01 #Javascript
[js高手之路]单例模式实现模态框的示例
Sep 01 #Javascript
JQuery EasyUI 结合ztrIee的后台页面开发实例
Sep 01 #jQuery
You might like
MVC模式的PHP实现
2006/10/09 PHP
php htmlspecialchars加强版
2010/02/16 PHP
解析PHP跳出循环的方法以及continue、break、exit的区别介绍
2013/07/01 PHP
php用正则表达式匹配中文实例详解
2013/11/06 PHP
php上传图片获取路径及给表单字段赋值的方法
2016/01/23 PHP
PHP正则+Snoopy抓取框架实现的抓取淘宝店信誉功能实例
2017/05/17 PHP
laravel框架邮箱认证实现方法详解
2019/11/22 PHP
jQuery选择头像并实时显示的代码
2010/06/27 Javascript
jQuery实现tab标签自动切换的方法
2015/02/28 Javascript
cocos2dx骨骼动画Armature源码剖析(三)
2015/09/08 Javascript
莱鸟介绍window.print()方法
2016/01/06 Javascript
浅析JavaScript动画模拟拖拽原理
2016/12/09 Javascript
Bootstrap表单控件使用方法详解
2017/01/11 Javascript
JS笛卡尔积算法与多重数组笛卡尔积实现方法示例
2017/12/01 Javascript
vue中将html字符串转换成html后遇到的问题小结
2018/12/10 Javascript
jQuery+PHP+Ajax实现动态数字统计展示功能
2019/12/25 jQuery
Vue 实现登录界面验证码功能
2020/01/03 Javascript
JS中循环遍历数组的四种方式总结
2021/01/23 Javascript
python检测是文件还是目录的方法
2015/07/03 Python
Python 常用string函数详解
2016/05/30 Python
Django objects.all()、objects.get()与objects.filter()之间的区别介绍
2017/06/12 Python
django定期执行任务(实例讲解)
2017/11/03 Python
python中的插值 scipy-interp的实现代码
2018/07/23 Python
利用python画出折线图
2018/07/26 Python
Python3 shutil(高级文件操作模块)实例用法总结
2020/02/19 Python
使用Python对Dicom文件进行读取与写入的实现
2020/04/20 Python
Python如何爬取b站热门视频并导入Excel
2020/08/10 Python
python中@property的作用和getter setter的解释
2020/12/22 Python
CSS3的 fit-content实现水平居中
2017/09/07 HTML / CSS
The North Face北面法国官网:美国著名户外品牌
2019/11/01 全球购物
市场营销专业毕业生自荐信
2013/11/02 职场文书
小学英语教学反思案例
2014/02/04 职场文书
车贷收入证明范本
2014/09/14 职场文书
医院营销工作计划
2015/01/16 职场文书
2015年音乐教学工作总结
2015/07/22 职场文书
前端JavaScript大管家 package.json
2021/11/02 Javascript