深入理解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 相关文章推荐
地址栏上的一段语句,改变页面的风格。(教程)
Apr 02 Javascript
正负小数点后两位浮点数实现原理及代码
Sep 06 Javascript
ExtJS4 表格的嵌套 rowExpander应用
May 02 Javascript
jQueryMobile之Helloworld与页面切换的方法
Feb 04 Javascript
Jquery 效果使用详解
Nov 23 Javascript
解析Node.js异常处理中domain模块的使用方法
Feb 16 Javascript
原生JS和jQuery版实现文件上传功能
Apr 18 Javascript
设置点击文本框或图片弹出日历控件的实现代码
May 12 Javascript
原生JS实现移动端web轮播图详解(结合Tween算法造轮子)
Sep 10 Javascript
微信小程序 函数防抖 解决重复点击消耗性能问题实现代码
Sep 12 Javascript
vscode调试node.js的实现方法
Mar 22 Javascript
Vant 中的Toast设置全局的延迟时间操作
Nov 04 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
不用GD库生成当前时间的PNG格式图象的程序
2006/10/09 PHP
php checkbox 取值详细说明
2010/08/19 PHP
.htaccess文件保护实例讲解
2011/02/06 PHP
PHP的PSR规范中文版
2013/09/28 PHP
Yii2.0 RESTful API 基础配置教程详解
2018/12/26 PHP
PHP常用的类封装小结【4个工具类】
2019/06/28 PHP
Laravel + Elasticsearch 实现中文搜索的方法
2020/02/02 PHP
js 弹出菜单/窗口效果
2011/10/30 Javascript
微信小程序 开发之滑块视图容器(swiper)详解及实例代码
2017/02/22 Javascript
vue.js系列中的vue-fontawesome使用
2018/02/10 Javascript
layui select动态添加option的实例
2018/03/07 Javascript
详解Vue单元测试case写法
2018/05/24 Javascript
浅谈Webpack多页应用HMR卡住问题
2019/04/24 Javascript
python不带重复的全排列代码
2013/08/13 Python
Windows系统配置python脚本开机启动的3种方法分享
2015/03/10 Python
Python新手实现2048小游戏
2015/03/31 Python
python实现FTP服务器服务的方法
2017/04/11 Python
python如何定义带参数的装饰器
2018/03/20 Python
django celery redis使用具体实践
2019/04/08 Python
Django如何将URL映射到视图
2019/07/29 Python
linux 下python多线程递归复制文件夹及文件夹中的文件
2020/01/02 Python
TensorFlow实现自定义Op方式
2020/02/04 Python
一款html5 canvas实现的图片玻璃碎片特效
2014/09/11 HTML / CSS
西班牙用户之间买卖视频游戏的平台:Wakkap
2020/03/21 全球购物
如何手工释放资源
2013/12/15 面试题
JDBC操作数据库的基本流程是什么
2014/10/28 面试题
电话客服工作职责
2014/07/27 职场文书
领导干部对照检查材料
2014/08/24 职场文书
文明倡议书
2015/01/19 职场文书
食品质检员岗位职责
2015/04/08 职场文书
全国劳模先进事迹材料(2016精选版)
2016/02/25 职场文书
2016年教师党员承诺书范文
2016/03/24 职场文书
2019年励志签名:致拼搏路上的自己
2019/10/11 职场文书
Golang二维切片初始化的实现
2021/04/08 Golang
python实现图片批量压缩
2021/04/24 Python
sql时间段切分实现每隔x分钟出一份高速门架车流量
2022/02/28 SQL Server