概述

在Vue开发中,模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的。在模板中放入太多的逻辑会让模板过重且难以维护。当你想要在模板中多次引用相同表达式时,就会更加难以处理。所以,对于任何复杂逻辑,你都应当使用计算属性。本文主要讲解Vue中的计算属性和侦听器,仅供学习分享使用,如有不足之前,还请指正。

计算属性

计算属性步骤:

1. 在computed属性中增加reverseMsg方法,如下所示:

 <script type="text/javascript">
var vm = new Vue({
el: '#app',
data: {
msg: 'welcome to vue world!!!' },
computed: {
reverseMsg: function() {
// `this` 指向 vm 实例
return this.msg.split('').reverse().join('');
}
} });
</script>

2. 在Html中进行引用,你可以像绑定普通属性一样在模板中绑定计算属性。

Vue 知道 vm.reverseMsg 依赖于 vm.msg,因此当 vm.msg 发生改变时,所有依赖 vm.reverseMsg 的绑定也会更新。如下所示:

 <p>原始信息: {{ msg }}</p>
<p>计算属性反转信息: {{ reverseMsg }}</p>

采用表达式的方式 ,则是如下所示:

 <span>{{ msg.split('').reverse().join('') }}</span>

在这个地方,模板不再是简单的声明式逻辑。你必须看一段时间才能意识到,这里是想要显示变量 msg 的翻转字符串。此处对比一下,采用计算属性的方式,则更加简洁明了。

计算属性缓存 vs 方法

你可能已经注意到我们可以通过在表达式中调用方法来达到同样的效果,我们可以将同一函数定义为一个方法而不是一个计算属性。两种方式的最终结果确实是完全相同的。

如下所示,声明一个方法:

 <script type="text/javascript">
var vm = new Vue({
el: '#app',
data: {
msg: 'welcome to vue world!!!'
},
methods: {
reversedMsg: function() {
return this.msg.split('').reverse().join('');
}
} });
</script>

在Html中进行引用,如下所示:

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

差异:不同的是计算属性是基于它们的响应式依赖进行缓存的。 只在相关响应式依赖发生改变时它们才会重新求值。 这就意味着只要 msg 还没有发生改变,多次访问 reversedMsg计算属性会立即返回之前的计算结果,而不必再次执行函数。 即:计算属性只有当依赖属性发生改变时,才重新计算,而函数需要每次都重新计算。
也同样意味着下面的计算属性将不再更新,因为 Date.now() 不是响应式依赖:  相比之下,每当触发重新渲染时,调用方法将总会再次执行函数。

 <script type="text/javascript">
var vm = new Vue({
el: '#app',
data: {
msg: 'welcome to vue world!!!' },
computed: {
now: function() {
return Date.now().toString();
} });
</script>

如下所示,将不会随着时间的改变而触发。

 <p>{{now}}</p>

如果你不希望有缓存,请用方法来替代。

计算属性 vs 侦听属性

如下所示:有两个data属性,firstName和lastName,fullName依赖说前两个变化而变化。如果需要采用侦听属性,需要对firstName和lastName进行侦听。

 <script type="text/javascript">
var vm = new Vue({
el: '#app',
data: {
firstName: 'Foo',
lastName: 'Bar',
fullName: 'Foo Bar',
},
computed: {
fullName2: function() {
return this.firstName + ' ' + this.lastName;
}
},
watch: {
firstName: function(val) {
this.fullName = val + ' ' + this.lastName;
},
lastName: function(val) {
this.fullName = this.firstName + ' ' + val;
}
}
});
</script>

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

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

如上所示:fullName采用侦听属性,fullName2采用计算属性,对比一下,好得多了,不是吗?

计算属性的 setter

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

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

在UI中引用的方式是一样的。当fullName3的值发生改变时,将更新firstName和lastName。

 <p>{{fullName3}}</p>

侦听器

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

 <script type="text/javascript">
var vm = new Vue({
el: '#app',
data: {
question: '',
answer: 'I cannot give you an answer until you ask a question!' },
watch: {
// 如果 `question` 发生改变,这个函数就会运行
question: function(newQuestion, oldQuestion) {
if (newQuestion == '') {
this.answer = 'Waiting for you to stop typing...';
} else {
this.answer = '请回答';
}
}
}
});
</script>

在页面中绑定属性

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

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

在本示例中用的全部代码 ,如下所示:

 <!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>一起学Vue之计算属性和侦听器</title>
<!-- 开发环境版本,包含了有帮助的命令行警告 -->
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
<div id="app">
<span>{{msg}}</span>
<h2>计算属性</h2>
<!--
模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的。在模板中放入太多的逻辑会让模板过重且难以维护。例如:
-->
<br />
<span>
{{ msg.split('').reverse().join('') }}
</span>
<!--
在这个地方,模板不再是简单的声明式逻辑。你必须看一段时间才能意识到,这里是想要显示变量 msg 的翻转字符串。
当你想要在模板中多次引用此处的翻转字符串时,就会更加难以处理。
所以,对于任何复杂逻辑,你都应当使用计算属性
-->
<p>原始信息: {{ msg }}</p>
<p>计算属性反转信息: {{ reverseMsg }}</p>
<!--
这里我们声明了一个计算属性 reverseMsg。我们提供的函数将用作属性 vm.reverseMsg 的 getter 函数:
-->
<!--
你可以像绑定普通属性一样在模板中绑定计算属性。Vue 知道 vm.reverseMsg 依赖于 vm.msg,
因此当 vm.msg 发生改变时,所有依赖 vm.reverseMsg 的绑定也会更新。
-->
<h2>计算属性缓存 vs 方法</h2>
<p>Reversed message: "{{ reversedMsg() }}"</p>
<!--
你可能已经注意到我们可以通过在表达式中调用方法来达到同样的效果:
-->
<!--
我们可以将同一函数定义为一个方法而不是一个计算属性。两种方式的最终结果确实是完全相同的。
然而,不同的是计算属性是基于它们的响应式依赖进行缓存的。
只在相关响应式依赖发生改变时它们才会重新求值。
这就意味着只要 msg 还没有发生改变,多次访问 reversedMsg 计算属性会立即返回之前的计算结果,而不必再次执行函数。
即:计算属性只有当依赖属性发生改变时,才重新计算,而函数需要每次都重新计算。
-->
<!--
也同样意味着下面的计算属性将不再更新,因为 Date.now() 不是响应式依赖:
相比之下,每当触发重新渲染时,调用方法将总会再次执行函数。
如果你不希望有缓存,请用方法来替代
-->
<p>{{now}}</p>
<h2>计算属性 vs 侦听属性</h2>
<div id="demo">{{ fullName }}</div>
<!--
上面代码是命令式且重复的。将它与计算属性的版本进行比较:
-->
<div id="demo">{{ fullName2 }}</div>
<!--
好得多了,不是吗?
-->
<h2>计算属性的 setter</h2>
<!--
计算属性默认只有 getter ,不过在需要时你也可以提供一个 setter :
-->
<p>{{fullName3}}</p>
<h2>侦听器</h2>
<!--
虽然计算属性在大多数情况下更合适,但有时也需要一个自定义的侦听器。
这就是为什么 Vue 通过 watch 选项提供了一个更通用的方法,来响应数据的变化。
当需要在数据变化时执行异步或开销较大的操作时,这个方式是最有用的。
-->
<p>Ask a yes/no question:
<input v-model="question">
</p>
<p>{{ answer }}</p>
<!--
在这个示例中,使用 watch 选项允许我们执行异步操作 (访问一个 API),
限制我们执行该操作的频率,并在我们得到最终结果前,设置中间状态。
这些都是计算属性无法做到的。
-->
</div>
<script type="text/javascript">
var vm = new Vue({
el: '#app',
data: {
msg: 'welcome to vue world!!!',
firstName: 'Foo',
lastName: 'Bar',
fullName: 'Foo Bar',
question: '',
answer: 'I cannot give you an answer until you ask a question!' },
methods: {
reversedMsg: function() {
return this.msg.split('').reverse().join('')
}
},
computed: {
reverseMsg: function() {
// `this` 指向 vm 实例
return this.msg.split('').reverse().join('');
},
now: function() {
return Date.now().toString();
},
fullName2: function() {
return this.firstName + ' ' + this.lastName;
},
fullName3: {
//getter
get: function() {
return this.firstName + ' ' + this.lastName;
},
//setter
set: function(newValue) {
console.log('newValue=' + newValue);
var names = newValue.split(' ');
this.firstName = names[0];
this.lastName = names[names.length - 1];
}
}
},
watch: {
firstName: function(val) {
this.fullName = val + ' ' + this.lastName;
},
lastName: function(val) {
this.fullName = this.firstName + ' ' + val;
},
// 如果 `question` 发生改变,这个函数就会运行
question: function(newQuestion, oldQuestion) {
if (newQuestion == '') {
this.answer = 'Waiting for you to stop typing...';
} else {
this.answer = '请回答';
}
} } });
</script>
</body>
</html>

备注

暴虎冯河,死而无悔者,吾不与也;必也,临事而惧,好谋而成者也。

一起学Vue之计算属性和侦听器的更多相关文章

  1. vue基础——计算属性和侦听器

    计算属性——介绍 模板内的表达式非常便利,但是设计他们的初衷是用于简单计算的.在模板中放入太多的逻辑会让模板太过沉重切难以维护.如下: <div id="example"&g ...

  2. Vue的计算属性和侦听器

    1 计算属性:他是根据对象已有的属性计算出新的属性值.具有缓存的功能,如果原始属性不变,则用缓存.否则,重新计算. 前端 <form> <label>姓</label&g ...

  3. vue基础---计算属性和侦听器

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

  4. vue之计算属性和侦听器

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

  5. vue 之 计算属性和侦听器

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

  6. Vue.js之Vue计算属性、侦听器、样式绑定

    前言 上一篇介绍了Vue的基本概念,这一篇介绍一下Vue的基本使用. 一.搭建一个Vue程序 1.1 搭建Vue环境 搭建Vue的开发环境总共有三种方法: 引入CDN <script src=& ...

  7. vue计算属性和侦听器

    一.计算属性: main.js: var app = new Vue({ el: '#app', data: { math: 80, physics: 90, english: 30 }, compu ...

  8. Vue学习之vue中的计算属性和侦听器

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  9. Vue.js 生命周期、计算属性及侦听器

    一.创建一个Vue实例 每个Vue应用都是使用Vue函数创建一个Vue实例.所有的Vue组件都是一个Vue实例,并且接受相同的选项对象(一些根实例特有的选项除外). 数据和方法 当一个实例被创建后,它 ...

随机推荐

  1. 教你用Java web实现多条件过滤功能

    生活中,当你闲暇之余浏览资讯的时候,当你搜索资料但繁杂信息夹杂时候,你就会想,如何更为准确的定位需求信息.今天就为你带来: 分页查询 需求分析:在列表页面中,显示指定条数的数据,通过翻页按钮完成首页/ ...

  2. Theano教程

    让我们开始一个交互式会话(例如使用python或ipython)并导入Theano. from theano import * 你需要使用Theano的tensor子包中的几个符号.让我们以一个方便的 ...

  3. Java NIO 三大组件之 Buffer

    NIO大三组件 之Buffer 一.什么是Buffer Buffer是用于特定原始类型的数据的容器. 它的实质就是一组数组,用于存储不同类型的数据. 二.缓冲区的类型 缓冲区类型除了Boolean值类 ...

  4. P1087 FBI树

    题目描述 我们可以把由“00”和“11”组成的字符串分为三类:全“00”串称为BB串,全“11”串称为I串,既含“00”又含“11”的串则称为F串. FBIFBI树是一种二叉树,它的结点类型也包括FF ...

  5. 【RN - 基础】之FlexBox弹性布局

    前言 弹性盒模型(The Flexible Box Module),又叫FlexBox,意为“弹性布局”,旨在通过弹性的方式来对齐和分布容器中内容的空间,使其能适应不同的屏幕,为盒装模型提供最大的灵活 ...

  6. U盘安装centos 7 提示 “Warning: /dev/root does not exist

    背景介绍:公司需要使用台式机安装Centos 7.5 系统,来部署一个测试的数据库,在安装Centos 7.5 系统的时候,使用U启安装,但有问题. 提示信息如下 如图:安装centos 7时提示 & ...

  7. Linux海量数据高并发实时同步架构方案杂谈

    不论是Redhat还是CentOS系统,除去从CDN缓存或者数据库优化.动静分离等方面来说,在架构层面上,实 现海量数据高并发实时同步访问概括起来大概可以从以下几个方面去入手,当然NFS的存储也可以是 ...

  8. 使用 colgroup 和 col 实现响应式表格

    Table 在项目使用中十分频繁,特别是在后台管理系统中,table 无疑是数据展示的第一公民,在早些年的网页中,table 也是网页布局的第一选择,然后使用好 table 并不容易,其它有很多子元素 ...

  9. Spring Boot和Feign中使用Java 8时间日期API(LocalDate等)的序列化问题【转】

    Spring Boot和Feign中使用Java 8时间日期API(LocalDate等)的序列化问题 http://blog.didispace.com/Spring-Boot-And-Feign- ...

  10. shell一键部署nginx+tomcat

    1.首先拉取环境  tomcat需要用到jdk环境 提前准备好nginx源码包,放于指定目录下 vim test.sh #!/bin/bash yum -y install gcc gcc-c++ z ...