一、前言

  在 Vue 中,我们可以很方便的将数据使用插值表达式( Mustache 语法)的方式渲染到页面元素中,但是插值表达式的设计初衷是用于简单运算,即我们不应该对差值做过多的操作。当我们需要对差值做进一步的处理时,这时,我们就应该使用到 Vue 中的计算属性来完成这一操作。同时,当差值数据变化时执行异步或开销较大的操作时,我们可以通过采用监听器的方式来达到我们的目的。

  学习系列目录地址:https://www.cnblogs.com/danvic712/p/9549100.html

  仓储地址:https://github.com/Lanesra712/VueTrial/blob/master/Chapter01-Rookie/watcher.html

二、干货合集

  1、计算属性

  计算属性,一般是用来描述一个属性值依赖于另一个的属性值,当我们使用插值表达式将计算属性绑定到页面元素上时,计算属性会在依赖的属性值的变化时自动的更新 DOM 元素。例如在下面的代码中,我们在 computed 中,定义了一个 reversedMessage 属性,它可以根据我们的 data 中的 message 属性的变化自动的获取到反转后的 message 的属性值。

<div id="app">
输入的值:<input type="text" v-model="message"><br />
反转的值:{{reversedMessage}}
</div> <script>
var vm = new Vue({
el: '#app',
data: {
message: ''
},
computed: {
reversedMessage: function () {
//这里的 this 指向 当前的 vm 实例
return this.message.split('').reverse().join('')
}
},
methods: {}
})
</script>

  

  可能你会发觉,这里的写法和我们定义方法时很相似,我们完全也可以在 methods 中定义一个方法来实现这个需求。原来,计算属性的本质就是一个方法,只不过,我们在使用计算属性的时候,是把计算属性的名称直接当做属性来使用,而并不会把计算属性当做一个方法去调用。

  那么,为什么我们还要去使用计算属性而不是去定义一个方法呢?原来,计算属性是基于它们的依赖进行缓存的。即只有在相关依赖发生改变时它们才会重新求值。例如在上面的例子中,只要 message 的属性值没有发生改变,无论任何使用我们使用到 reversedMessage 属性,都会立即返回之前的计算结果,而不必再次执行函数。

  反之,如果你使用方法的形式实现,当你使用到 reversedMessage 方法时,无论 message 属性是否发生了改变,方法都会重新执行一次,这无形中增加了系统的开销。当然,你也可以自己在方法中实现根据依赖进行缓存,嗯,如果你不嫌烦的话。。。

  在上面的案例中,对于 reversedMessage 这个计算属性来说,我们主要的目的是为了获取属性的值,即使用的是计算属性的 getter 方法。不过,如果你需要使用到计算属性的 setter 方法时,我们也是可以为计算属性提供一个 setter 方法的。

<div id="app">
输入的值:<input type="text" v-model="message"><br />
反转的值:{{reversedMessage}}
</div> <script>
var vm = new Vue({
el: '#app',
data: {
message: ''
},
computed: {
reversedMessage: {
get: function () {
return this.message.split('').reverse().join('')
},
set: function (value) {
this.message = value.split('').reverse().join('')
}
}
},
methods: {}
})
</script>

  在上面的代码中,我们为计算属性 reversedMessage 增加了一个 setter 方法:通过设置 reversedMessage 的值,同样进行反转操作,并最终将结果赋值给属性 message。

  2、监听属性

  在 vue 中,我们不光可以使用计算属性的方式来监听数据的变化,还可以使用 watch 监听器的方法来监测某个数据发生的变化。不同的是,计算属性仅仅是对于依赖数据的变化后进行的数据操作,而 watch 更加侧重于对于监测中的某个数据发生变化后所执行的一系列的业务逻辑操作。

  监听器以 key-value 的形式定义,key 是一个字符串,它是需要被监测的对象,而 value 则可以是字符串(方法的名称)、函数(可以获取到监听对象改变前的值以及更新后的值)或是一个对象(对象内可以包含回调函数的其它选项,例如是否初始化时执行监听 immediate,或是是否执行深度遍历 deep,即是否对对象内部的属性进行监听)。

  1)回调值为函数方法

  在下面的例子中,我们监听了 message 属性的变化,根据属性的变化后执行了回调方法,打印出了属性变化前后的值。

<div id="app">
输入的值:<input type="text" v-model="message">
</div> <script>
var vm = new Vue({
el: '#app',
data: {
message: ''
},
computed: {},
watch: {
//回调为未创建的方法
'message': function (newValue, oldValue) {
console.log(`新值:${newValue} --------- 旧值:${oldValue}`)
}
},
methods: {}
})
</script>

  同样的,我们可以通过方法名称指明回调为已经定义好的方法。

<div id="app">
输入的值:<input type="text" v-model="message">
</div> <script>
var vm = new Vue({
el: '#app',
data: {
message: ''
},
computed: {},
watch: {
//回调为已创建好的方法
'message': 'recording'
},
methods: {
recording: function (newValue, oldValue) {
console.log(`method记录:新值:${newValue} --------- 旧值:${oldValue}`)
}
}
})
</script>

  2)回调值为对象

  当我们监听的回调值为一个对象时,我们不仅可以设置回调函数,还可以设置一些回调的属性。例如,在下面的例子中,我们监听了 User 这个对象,同时执行了执行深度遍历,这时,当我们监听到 User.name 这个属性发生改变的时候,我们就可以执行我们的回调函数。注意,深度遍历默认为 false,当不启用深度遍历时,我们是无法监听到对象的内部属性的变化的。

<div id="app">
用户姓名:<input type="text" v-model="User.name">
</div> <script>
var vm = new Vue({
el: '#app',
data: {
message: '',
User: {
name: 'zhangsan',
gender: 'male'
}
},
computed: {},
watch: {
//回调为对象
'User': {
handler: function (newValue, oldValue) {
console.log(`对象记录:新值:${newValue.name} --------- 旧值:${oldValue.name}`)
},
deep: true
}
},
methods: {}
})
</script>

  可能你发现了,为什么 newValue 与 oldValue 都是一样的啊?原来,当我们监听的数据为对象或数组时,newValue 和 oldValue 是相等的,因为对象和数组都为引用类型,这两个的形参指向的也是同一个数据对象。同时,如果我们不启用深度遍历,我们将无法监听到对于 User 对象中 name 属性的变化。

<div id="app">
用户姓名:<input type="text" v-model="User.name">
</div> <script>
var vm = new Vue({
el: '#app',
data: {
message: '',
User: {
name: 'zhangsan',
gender: 'male'
}
},
computed: {},
watch: {
//回调为对象
'User': {
handler: function (newValue, oldValue) {
console.log(`对象记录:新值:${newValue.name} --------- 旧值:${oldValue.name}`)
},
deep: false
}
},
methods: {}
})
</script>

三、总结

  1、计算属性的结果会被缓存起来,只有依赖的属性发生变化时才会重新计算,必须返回一个数据,主要用来进行纯数据的操作。

  2、监听器主要用来监听某个数据的变化,从而去执行某些具体的回调业务逻辑,不仅仅局限于返回数据。

四、参考

  1、深入理解 Vue Computed 计算属性

  2、Vue 2.0学习笔记: Vue中的computed属性

  3、Vue系列之computed使用详解

Vue.js-05:第五章 - 计算属性与监听器的更多相关文章

  1. vue.js基础知识篇(3):计算属性、表单控件绑定

    第四章:计算属性 为了避免过多的逻辑造成模板的臃肿不堪,可使用计算属性来简化逻辑. 1.什么是计算属性 <!DOCTYPE html><html lang="en" ...

  2. Vue.js基本规则提炼总结及计算属性学习

    Vue.js基本须知: 1)以“{{}}”格式 “Mustache” 语法(双大括号)来绑定表达式输出文本值; 2)以“{{{}}}”格式绑定原始的html,绑定的表达式内为字符串格式的html内容, ...

  3. VUE -- Vue.js每天必学之计算属性computed与$watch

    在模板中绑定表达式是非常便利的,但是它们实际上只用于简单的操作.模板是为了描述视图的结构.在模板中放入太多的逻辑会让模板过重且难以维护.这就是为什么 Vue.js 将绑定表达式限制为一个表达式.如果需 ...

  4. Vue.js-----轻量高效的MVVM框架(五、计算属性)

    #基础例子 <div id="dr01"> <h4>#基础例子</h4> <div> num01={{num01}}, num02= ...

  5. 读JS高级——第五章-引用类型 _记录

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  6. vue 计算属性和监听器

    一.计算属性 模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的.在模板中放入太多的逻辑会让模板过重且难以维护.例如: <div> {{ message.split('').rev ...

  7. Vue.js 学习笔记 第3章 计算属性

    本篇目录: 3.1 什么是计算属性 3.2 计算属性用法 3.3 计算属性缓存 模板内容的表达式常用语简单的运算,当其过长或逻辑复杂时,会难以维护,本章的计算属性就是用于解决该问题的. 3.1 什么是 ...

  8. Vue基础第三章 - 计算属性

    1.计算属性介绍 在第二章中我们介绍了在Vue的{{}}中可以使用一些简单的表达式进行计算,但是当表达式过长或者逻辑过于复杂就会变得不易理解和维护,比如第二章的示例{{ text.split(',') ...

  9. 非node环境下的vue.js 实现简单的购物车计算功能 样式请无视

    都说vue的双向数据绑定好用,自己用了下,感觉做购物车没想象中好用.自己的实现如下: <!DOCTYPE html> <html lang="en"> &l ...

随机推荐

  1. Sublime Text 3 Build 3065 License key 注册码 秘钥

    -– BEGIN LICENSE -– Andrew Weber Single User License EA7E-855605 813A03DD 5E4AD9E6 6C0EEB94 BC99798F ...

  2. java的classpath路径中加点号 ‘.’ 的作用

    "."表示当前目录,就是编译或者执行程序时你所在的目录下的.class文件:而JAvA_HOME表示JDK安装路径 该路径在eclipse中是以vmarg的形式传入的,可以在任务管 ...

  3. Reactor和Proactor模式

    在高性能的I/O设计中,有两个比较著名的模式Reactor和Proactor模式,其中Reactor模式用于同步I/O,而Proactor运用于异步I/O操作.同步和异步 同步和异步是针对应用程序和内 ...

  4. acl.go

    package acl   import ( "github.com/armon/go-radix" )   var ( // allowAll is a singleton po ...

  5. Python3 ——斐波那契数列(经典)

    刚刚学习了 斐波那契数列,整理一下思路,写个博文给未来的学弟学妹参考一下,希望能够帮助到他们 永远爱你们的 ----新宝宝 经历过简单的学习之后,写出一个比较简单的代码,斐波那契数列:具体程序如下: ...

  6. Git----GitHub上传本地文件到git

    1.首先在git上创建一个库,用来保存上传的本地文件 2.通过命令 git init 把这个目录变成git可以管理的仓库 git init 3.将远程git库克隆一份保存到本地 git clone x ...

  7. CTF丨2019互联网安全城市巡回赛·西安站,我们来了!

    万物互联时代,网信事业发展突飞猛进,互联网悄然渗透到国民生活的每一个角落,伴随而来的网络安全威胁和风险也日渐突出.网络诈骗.钓鱼软件.勒索病毒等安全问题层出不穷,信息泄露等网络安全事件也频繁上演,给用 ...

  8. 【.NETCore开源】开弓没有回头箭

    2019.2.11 开工大吉!经过了半个月的休假,今天回归岗位重新拾起工作,却发现熟悉的代码生疏了.年前的计划回忆不起来了,俗称"节后综合症". 忆半月圈子 过年放假的前几天有多篇 ...

  9. [深度应用]·实战掌握Dlib人脸识别开发教程

    [深度应用]·实战掌握Dlib人脸识别开发教程 个人网站--> http://www.yansongsong.cn/ 项目GitHub地址--> https://github.com/xi ...

  10. 锁开销优化以及 CAS 简单说明

    锁开销优化以及 CAS 简单说明 锁 互斥锁是用来保护一个临界区,即保护一个访问共用资源的程序片段,而这些共用资源又无法同时被多个线程访问的特性.当有线程进入临界区段时,其他线程或是进程必须等待. 在 ...