深入浅析Vue.js计算属性和侦听器


Posted in Javascript onMay 05, 2018

一、 概述

 计算属性

  模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的。在模板中放入太多的逻辑会让模板过重且难以维护。例如:

<div id="example">
 {{ message.split('').reverse().join('') }}
</div>

   在这个地方,模板不再是简单的声明式逻辑。你必须看一段时间才能意识到,这里是想要显示变量 message 的翻转字符串。当你想要在模板中多次引用此处的翻转字符串时,就会更加难以处理。

  所以,对于任何复杂逻辑,你都应当使用计算属性。

 基础例子

<div id="example">
 <p>Original message: "{{ message }}"</p>
 <p>Computed reversed message: "{{ reversedMessage }}"</p>
</div>

var vm = new Vue({
 el: '#example',
 data: {
  message: 'Hello'
 },
 computed: {
  // 计算属性的 getter
  reversedMessage: function () {
   // `this` 指向 vm 实例
   return this.message.split('').reverse().join('')
  }
 }
})

    结果:

深入浅析Vue.js计算属性和侦听器

       这里我们声明了一个计算属性 reversedMessage。我们提供的函数将用作属性 vm.reversedMessage 的 getter 函数

console.log(vm.reversedMessage) // => 'olleH'
vm.message = 'Goodbye'
console.log(vm.reversedMessage) // => 'eybdooG'

      你可以像绑定普通属性一样在模板中绑定计算属性。Vue 知道 vm.reversedMessage 依赖于 vm.message,因此当 vm.message 发生改变时,所有依赖 vm.reversedMessage 的绑定也会更新。而且最妙的是我们已经以声明的方式创建了这种依赖关系:计算属性的 getter 函数是没有副作用 (side effect) 的,这使它更易于测试和理解。

二、计算属性缓存 vs 方法

        你可能已经注意到我们可以通过在表达式中调用方法来达到同样的效果

<p>Reversed message: "{{ reversedMessage() }}"</p>

// 在组件中
methods: {
 reversedMessage: function () {
  return this.message.split('').reverse().join('')
 }
}

        我们可以将同一函数定义为一个方法而不是一个计算属性。两种方式的最终结果确实是完全相同的。然而,不同的是计算属性是基于它们的依赖进行缓存的。计算属性只有在它的相关依赖发生改变时才会重新求值。这就意味着只要 message 还没有发生改变,多次访问 reversedMessage 计算属性会立即返回之前的计算结果,而不必再次执行函数。

    这也同样意味着下面的计算属性将不再更新,因为 Date.now() 不是响应式依赖

computed: {
 now: function () {
  return Date.now()
 }
}

      相比之下,每当触发重新渲染时,调用方法将总会再次执行函数。

我们为什么需要缓存?

      假设我们有一个性能开销比较大的的计算属性 A,它需要遍历一个巨大的数组并做大量的计算。然后我们可能有其他的计算属性依赖于 A 。如果没有缓存,我们将不可避免的多次执行 A 的 getter!如果你不希望有缓存,请用方法来替代。

三、计算属性 vs 侦听属性

       Vue 提供了一种更通用的方式来观察和响应 Vue 实例上的数据变动:侦听属性。当你有一些数据需要随着其它数据变动而变动时,你很容易滥用 watch——特别是如果你之前使用过 AngularJS。然而,通常更好的做法是使用计算属性而不是命令式的 watch 回调。

  细想一下这个例子

<div id="demo">{{ fullName }}</div>

var vm = new Vue({
 el: '#demo',
 data: {
  firstName: 'Foo',
  lastName: 'Bar',
  fullName: 'Foo Bar'
 },
 watch: {
  firstName: function (val) {
   this.fullName = val + ' ' + this.lastName
  },
  lastName: function (val) {
   this.fullName = this.firstName + ' ' + val
  }
 }
})

     上面代码是命令式且重复的。将它与计算属性的版本进行比较:

var vm = new Vue({
 el: '#demo',
 data: {
  firstName: 'Foo',
  lastName: 'Bar'
 },
 computed: {
  fullName: function () {
   return this.firstName + ' ' + this.lastName
  }
 }
})

     是不是好多了。

四、计算属性的 setter

      计算属性默认只有 getter ,不过在需要时你也可以提供一个 setter

// ...
computed: {
 fullName: {
  // getter
  get: function () {
   return this.firstName + ' ' + this.lastName
  },
  // setter
  set: function (newValue) {
   var names = newValue.split(' ')
   this.firstName = names[0]
   this.lastName = names[names.length - 1]
  }
 }
}
// ...

现在再运行 vm.fullName = 'John Doe' 时,setter 会被调用,vm.firstName 和 vm.lastName 也会相应地被更新。

五、侦听器

      虽然计算属性在大多数情况下更合适,但有时也需要一个自定义的侦听器。这就是为什么 Vue 通过 watch 选项提供了一个更通用的方法,来响应数据的变化。当需要在数据变化时执行异步或开销较大的操作时,这个方式是最有用的。

    例如:

<div id="watch-example">
 <p>
  Ask a yes/no question:
  <input v-model="question">
 </p>
 <p>{{ answer }}</p>
</div>

<!-- 因为 AJAX 库和通用工具的生态已经相当丰富,Vue 核心代码没有重复 -->
<!-- 提供这些功能以保持精简。这也可以让你自由选择自己更熟悉的工具。 -->
<script src="https://cdn.jsdelivr.net/npm/axios@0.12.0/dist/axios.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/lodash@4.13.1/lodash.min.js"></script>
<script>
var watchExampleVM = new Vue({
 el: '#watch-example',
 data: {
  question: '',
  answer: 'I cannot give you an answer until you ask a question!'
 },
 watch: {
  // 如果 `question` 发生改变,这个函数就会运行
  question: function (newQuestion, oldQuestion) {
   this.answer = 'Waiting for you to stop typing...'
   this.getAnswer()
  }
 },
 methods: {
  // `_.debounce` 是一个通过 Lodash 限制操作频率的函数。
  // 在这个例子中,我们希望限制访问 yesno.wtf/api 的频率
  // AJAX 请求直到用户输入完毕才会发出。想要了解更多关于
  // `_.debounce` 函数 (及其近亲 `_.throttle`) 的知识,
  // 请参考:https://lodash.com/docs#debounce
  getAnswer: _.debounce(
   function () {
    if (this.question.indexOf('?') === -1) {
     this.answer = 'Questions usually contain a question mark. ;-)'
     return
    }
    this.answer = 'Thinking...'
    var vm = this
    axios.get('https://yesno.wtf/api')
     .then(function (response) {
      vm.answer = _.capitalize(response.data.answer)
     })
     .catch(function (error) {
      vm.answer = 'Error! Could not reach the API. ' + error
     })
   },
   // 这是我们为判定用户停止输入等待的毫秒数
   500
  )
 }
})
</script>

    结果:当没有输入?号,那么显示如下:

深入浅析Vue.js计算属性和侦听器

    当有?时候会输出“yes”或者“no”

 具体案例效果地址:侦听器

深入浅析Vue.js计算属性和侦听器

 在这个示例中,使用 watch 选项允许我们执行异步操作 (访问一个 API),限制我们执行该操作的频率,并在我们得到最终结果前,设置中间状态。这些都是计算属性无法做到的。

总结

以上所述是小编给大家介绍的Vue.js计算属性和侦听器,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!

Javascript 相关文章推荐
js列举css中所有图标的实现代码
Jul 04 Javascript
通过AJAX的JS、JQuery两种方式解析XML示例介绍
Sep 23 Javascript
扩展jQuery对象时如何扩展成员变量具体怎么实现
Apr 25 Javascript
javascript的tab切换原理与效果实现方法
Jan 10 Javascript
jquery实现表格隔行换色效果
Nov 19 Javascript
jQuery leonaScroll 1.1 自定义滚动条插件(推荐)
Sep 17 Javascript
vue路由组件按需加载的几种方法小结
Jul 12 Javascript
微信小程序定位当前城市的方法
Jul 19 Javascript
vue监听对象及对象属性问题
Aug 20 Javascript
JavaScript装箱及拆箱boxing及unBoxing用法解析
Jun 15 Javascript
在vue中使用eslint,配合vscode的操作
Nov 09 Javascript
JavaScript中10个Reduce常用场景技巧
Jun 21 Javascript
详解js跨域请求的两种方式,支持post请求
May 05 #Javascript
vue 注册组件的使用详解
May 05 #Javascript
Vue三层嵌套路由的示例代码
May 05 #Javascript
动态加载JavaScript文件的3种方式
May 05 #Javascript
Node.js的Koa实现JWT用户认证方法
May 05 #Javascript
浅谈vue项目可以从哪些方面进行优化
May 05 #Javascript
Angular模版驱动表单的使用总结
May 05 #Javascript
You might like
php使用memcoder将视频转成mp4格式的方法
2015/03/12 PHP
PHP实现WebService的简单示例和实现步骤
2015/03/27 PHP
yum命令安装php7和相关扩展
2016/07/04 PHP
php使用flock阻塞写入文件和非阻塞写入文件的实例讲解
2017/07/10 PHP
PHP正则匹配到2个字符串之间的内容方法
2018/12/24 PHP
PHP设计模式之装饰器(装饰者)模式(Decorator)入门与应用详解
2019/12/13 PHP
javascript中的location用法简单介绍
2007/03/07 Javascript
Javascript attachEvent传递参数的办法
2009/12/14 Javascript
jquery默认校验规则整理
2014/03/24 Javascript
js图片闪动特效可以控制间隔时间如几分钟闪动一下
2014/08/12 Javascript
Javascript实现div的toggle效果实例分析
2015/06/09 Javascript
动态加载jQuery的两种方法实例分析
2015/07/17 Javascript
js实现字符串和数组之间相互转换操作
2016/01/12 Javascript
很棒的js选项卡切换效果
2016/07/15 Javascript
详解angularJs中自定义directive的数据交互
2017/01/13 Javascript
BootstrapValidator实现注册校验和登录错误提示效果
2017/03/10 Javascript
浅谈redux以及react-redux简单实现
2018/08/28 Javascript
机器学习之KNN算法原理及Python实现方法详解
2018/07/09 Python
Puppeteer使用示例详解
2019/06/20 Python
python批量修改图片尺寸,并保存指定路径的实现方法
2019/07/04 Python
Pytorch在dataloader类中设置shuffle的随机数种子方式
2020/01/14 Python
python 实现图片批量压缩的示例
2020/12/18 Python
美国批发零售网站:GearXS
2016/07/26 全球购物
Glamest意大利:女性在线奢侈品零售店
2019/04/28 全球购物
Perfume’s Club法国站:购买香水和化妆品
2019/05/02 全球购物
AJAX都有哪些有点和缺点
2012/11/03 面试题
Java中有几种类型的流?JDK为每种类型的流提供了一些抽象类以供继承,请说出他们分别是哪些类?
2012/05/30 面试题
大学学习生活感言
2014/01/18 职场文书
幼儿园元旦亲子活动方案
2014/02/17 职场文书
践行三严三实心得体会
2014/10/13 职场文书
12.4全国法制宣传日活动总结
2014/11/01 职场文书
驳回起诉裁定书
2015/05/19 职场文书
扩展多台相同的Web服务器
2021/04/01 Servers
vue引入Excel表格插件的方法
2021/04/28 Vue.js
Java多条件判断场景中规则执行器的设计
2021/06/26 Java/Android
一文弄懂MySQL索引创建原则
2022/02/28 MySQL