computed

在指令章节讲过,插值{{ }}和指令都接受变量和表达式的写法,使用表达式可以进行简单的二元或三元运算。但如果要执行更加复杂的计算或频繁重复的计算,如果还是直接写在指令的表达式中会让代码过于臃肿,不好看不优雅。这个时候可以使用computed属性。

比如:

<!-- 假设后端返回的价格单位是分,显示格式要求¥0.00元 -->
<!-- bad -->
<div>总价:¥ {{ (price / 100).toFixed(2) }}元</div>
<!-- good -->
<div>总价:{{ total }}</div>
computed: {
total() {
return `¥ ${(this.price / 100).toFixed(2)}元`
}
},

计算属性就是data对象的一个扩展和增强版本。data中的值可以进行读写操作,同样计算属性的值也是可读可写的。

// 单纯读取时,函数写法,默认调用get方法
computed: {
total() {
return `¥ ${(this.price / 100).toFixed(2)}元`
}
}, // 可读可写时,对象写法
computed: {
total: {
get: function () {
return `¥ ${this.price.toFixed(2)}元`
},
set: function (newValue) {
this.price = newValue / 100
}
}
}

常规想法,在这里容易产生一个误区,当我对计算属性赋值,如this.total = 250,total值会被覆盖直接等于250。但是,实际对计算属性赋值,如果是对象形式,只是会触发执行set函数,具体视图如何显示,取决于我们在set函数体内的执行代码,有没有更新get中的依赖值。

看个例子

点击查看DEMO:computed by get and set

<div id="app">
<div>总价:{{ total }}</div>
<button @click="changePrice">price加1</button>
<button @click="changeTotal">直接赋值total</button>
<div>赋值列表:{{ memorySetterList }}</div>
</div>
new Vue({
el: "#app",
data: {
price: 100,
memorySetterList: [],
},
computed: {
total: {
get(){
return `¥ ${this.price.toFixed(2)}元`
},
set(setVal) {
this.memorySetterList.push(setVal)
}
}
},
methods: {
changeCount() {
this.price++
},
changeDouble() {
console.log("setBefore:this.total:",this.total)
this.total += 50
console.log("setAfter:this.total:",this.total)
}
}
})

结果显示,this.total的值变不会有改变,赋值列表每次都是同样的值,并每点一次加50的效果。只有当price值改变才会触发更新。所以直接对计算属性赋值this.price = someValue只是被this.totalsetter方法捕捉到而已。

这也正是计算属性的特点一:依赖改变才会改变

上面的例子如果要实现对total赋值的同时,视图中{{ total }}也显示改变,可以在计算属性的setter中对其依赖值price进行改变。

computed: {
total: {
get: function () {
return `¥ ${this.price.toFixed(2)}元`
},
set: function (newValue) {
this.price = newValue / 100
}
}
}

计算属性的另一个特点:值会被缓存

多处调用计算属性,都是相同的值。并且依赖改变,多处调用的值都是会同时更新。

<div id="app">
<div>count1:{{ count1 }}</div>
<div>addCount1ByComputed: {{ addCount1ByComputed}}</div>
<div>addCount2ByMethod:{{count2}}</div>
</div>
var vm = new Vue({
el: "#app",
data: {
count1: 1,
count2: 1,
},
computed: {
addCount1ByComputed() {
return this.count1 * 10
}
},
methods: {
addCount2ByMethod() {
this.count2++
},
},
mounted() {
console.log("========computed========")
for (let i=0; i<5; i++) {
console.log(this.addCount1ByComputed)
}
console.log("=======methods=======")
for (let i=0; i<5; i++) {
this.addCount2ByMethod()
console.log(this.count2)
}
},
})

计算属性的特点:

  • 只有当计算属性所依赖的值改变时,才会重新触发执行函数改变其值。
  • 计算属性会被缓存,如果其依赖值不改变,即使在多个地方多次调用,函数体代码也不会被执行,只是取用已经被缓存的值。

总结

我理解computed就是对数据的显示的一种包装,这种包装的需要可能是出于数据格式的美化或数据调用上简化复用考虑。而computedget方法提供一种包装方法,set也是一种包装方法。只是大部分情况下,一种get方法已足够我们解决需求。很少有get解决起来还不够完美的情况下再需要使用set配合的情况。

但也确实有需要get/set同时包装一个数据对象的情况,比如根据后端请求回的商品价格price是分的单位,但现在视图上渲染需要按一定格式如¥0.00元格式,并且在其它地方使用price参与运算需要元的单位。

一般常规的做法是,在每个调用price都手动除一次100,或者在请求成功的回调中对price先转化元单位,再赋值到data中响应。但也可以使用computedset方法处理转为元再赋值this.price,实现price相关包装逻辑都在一起。如下例:

<div>总价:{{ total }}</div>
new Vue({
el: "#app",
data: {
price: '',
},
computed: {
total: {
get: function () {
return `¥ ${this.price.toFixed(2)}元`
},
set: function (newValue) {
this.price = newValue / 100
}
}
}
}),
mounted() {
this.request_getGoodsPrice()
.then(
// 常规写法
// res => this.price = res.totalPrice / 100
// 使用set方法,将price的处理逻辑合在一起,都是total的计算属性方法中。
res => this.total = res.totalPrice
)
}

vue-learning:16 - js - computed的更多相关文章

  1. Vue教程:计算属性computed与侦听器watch(三)

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

  2. 从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 十五 ║Vue基础:JS面向对象&字面量& this字

    缘起 书接上文<从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 十四 ║ VUE 计划书 & 我的前后端开发简史>,昨天咱们说到了以我的经历说明的web开发经历的 ...

  3. Vue学习笔记:methods、computed、watch的区别

    自:https://www.jb51.net/article/120073.htm 首先要说,methods,watch和computed都是以函数为基础的,但各自却都不同 而从作用机制和性质上看,m ...

  4. vue-learning:20 - js - 区别:filters / data / computed / watch / methods

    区别:filters / data / computed / watch / methods 在配置对象options中,filters/data/computed/watch/methods的每一项 ...

  5. 从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 二十三║Vue实战:Vuex 其实很简单

    前言 哈喽大家周五好,马上又是一个周末了,下周就是中秋了,下下周就是国庆啦,这里先祝福大家一个比一个假日嗨皮啦~~转眼我们的专题已经写了第 23 篇了,好几次都坚持不下去想要中断,不过每当看到群里的交 ...

  6. Vue中使用Cropper.js裁剪图片

    Cropper.js是一款很好用的图片裁剪工具,可以对图片的尺寸.宽高比进行裁剪,满足诸如裁剪头像上传.商品图片编辑之类的需求. github: https://github.com/fengyuan ...

  7. vue初学:基础概念

    一.vue使用步骤: 1.引包vue.js 2.html中写要操作的DOM节点 3.创建vue对象:new Vue({options}); 4.配置options:el:(要操作的对象,用选择器,同j ...

  8. vue中config/index.js:配置的详细理解

    当我们需要和后台分离部署的时候,必须配置config/index.js: 用vue-cli 自动构建的目录里面  (环境变量及其基本变量的配置) var path = require('path') ...

  9. Vue教程:组件Component详解(六)

    一.什么是组件? 组件 (Component) 是 Vue.js 最强大的功能之一.组件可以扩展 HTML 元素,封装可重用的代码.在较高层面上,组件是自定义元素,Vue.js 的编译器为它添加特殊功 ...

随机推荐

  1. WPF/Silverlight深度解决方案:(九)HLSL自定义渲染特效之完美攻略(下)

    原文:WPF/Silverlight深度解决方案:(九)HLSL自定义渲染特效之完美攻略(下) 本想只用两节来完成关于HLSL自定义渲染相关知识的讲解,鉴于最近非常的多的朋友对此相当感兴趣,想知道最多 ...

  2. iOS 9适配

    iOS 9系统策略更新,请开发者注意升级 近期苹果公司iOS 9系统策略更新,限制了http协议的访问,此外应用需要在“Info.plist”中将要使用的URL Schemes列为白名单,才可正常检查 ...

  3. day39-Spring 03-JDK的动态代理

    package cn.itcast.spring3; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Meth ...

  4. 【JZOJ4859】【NOIP2016提高A组集训第7场11.4】连锁店

    题目描述 Dpstr开了个饮料连锁店,连锁店共有n家,出售的饮料种类相同.为了促销,Dpstr决定让每家连锁店开展赠送活动.具体来说,在第i家店,顾客可以用ai个饮料瓶兑换到bi瓶饮料和1个纪念币(注 ...

  5. Firefox浏览器怎么安装adobe flash player插件

    https://jingyan.baidu.com/article/0a52e3f435d171bf62ed7237.html 有些朋友在使用Firefox火狐浏览器,但是火狐浏览器安装以后是没有fl ...

  6. Hbase数据模型概念视图

  7. epoll与fork

    使用epoll时,如果在调用epoll_create之后,调用了fork创建子进程,那么父子进程虽然有各自epoll实例的副本,但是在内核中,它们引用的是同一个实例.子进程向自己的epoll实例添加. ...

  8. 【[Offer收割]编程练习赛9 B】水陆距离

    [题目链接]:http://hihocoder.com/problemset/problem/1478 [题意] [题解] 一开始把所有的水域的位置都加入到队列中去; 然后跑一个bfs. 第一次到达的 ...

  9. The Top 50 Proprietary Programs that Drive You Crazy — and Their Open Source Alternatives

    The Top 50 Proprietary Programs that Drive You Crazy — and Their Open Source Alternatives 01 / 22 / ...

  10. day5_python之环境变量设置

    1.设置环境变量 import os,sys print(os.path.abspath(__file__)) #当前py文件的绝对路径 print(os.path.dirname(os.path.a ...