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
)
}

最新文章

  1. H5版俄罗斯方块(5)---需求演进和产品迭代
  2. iOS文件操作
  3. C++ “读取位置 0x****** 时发生访问冲突”的可能原因
  4. Flex学习第一天(两个数相加)
  5. JQuery 性能优化
  6. android 弹出带输入框的对话框
  7. Android系统--输入系统(九)Reader线程_核心类及配置文件
  8. 使用VS Code开发.Net Core 2.0 MVC Web应用程序教程之二
  9. 从SVN到Git最强指南
  10. You Are the One DP
  11. USB协议基础知识笔记
  12. Objective-C的面向对象特性(一)
  13. Stack编程队列
  14. zookeeper使用详解(命令、客户端、源码)
  15. Broadcom BCM94352z/DW1560驱动新姿势
  16. scoketserver模块(TCP协议 与 udp 协议)
  17. jstl-----之&lt;set&gt;标签
  18. cocos2dx-lua控制台报错集合
  19. 两个简单的API限流实现方案
  20. Git文件状态

热门文章

  1. Linux进程管理(二、 进程创建)
  2. 网络流24题 负载平衡(DCOJ8013)
  3. MUI - H5实现ios长按图标后进入图标排序及删除功能的效果
  4. zabbix3.4.x添加短信报警
  5. Introduction to 3D Game Programming with DirectX 12 学习笔记之 --- 第三章:变换
  6. nodeJs学习-07 express、body-parser;链式操作next
  7. Gym - 101480K_K - Kernel Knights (DFS)
  8. iptables禁止QQ端口
  9. 将 vue.js 获取的 html 文本转化为纯文本
  10. css实现简单的页面自适应宽度