vue-learning:16 - js - computed
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.total的setter方法捕捉到而已。
这也正是计算属性的特点一:依赖改变才会改变
上面的例子如果要实现对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就是对数据的显示的一种包装,这种包装的需要可能是出于数据格式的美化或数据调用上简化复用考虑。而computed的get方法提供一种包装方法,set也是一种包装方法。只是大部分情况下,一种get方法已足够我们解决需求。很少有get解决起来还不够完美的情况下再需要使用set配合的情况。
但也确实有需要get/set同时包装一个数据对象的情况,比如根据后端请求回的商品价格price是分的单位,但现在视图上渲染需要按一定格式如¥0.00元格式,并且在其它地方使用price参与运算需要元的单位。
一般常规的做法是,在每个调用price都手动除一次100,或者在请求成功的回调中对price先转化元单位,再赋值到data中响应。但也可以使用computed在set方法处理转为元再赋值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的更多相关文章
- Vue教程:计算属性computed与侦听器watch(三)
计算属性computed 模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的.在模板中放入太多的逻辑会让模板过重且难以维护.例如: <div id="example" ...
- 从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 十五 ║Vue基础:JS面向对象&字面量& this字
缘起 书接上文<从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 十四 ║ VUE 计划书 & 我的前后端开发简史>,昨天咱们说到了以我的经历说明的web开发经历的 ...
- Vue学习笔记:methods、computed、watch的区别
自:https://www.jb51.net/article/120073.htm 首先要说,methods,watch和computed都是以函数为基础的,但各自却都不同 而从作用机制和性质上看,m ...
- vue-learning:20 - js - 区别:filters / data / computed / watch / methods
区别:filters / data / computed / watch / methods 在配置对象options中,filters/data/computed/watch/methods的每一项 ...
- 从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 二十三║Vue实战:Vuex 其实很简单
前言 哈喽大家周五好,马上又是一个周末了,下周就是中秋了,下下周就是国庆啦,这里先祝福大家一个比一个假日嗨皮啦~~转眼我们的专题已经写了第 23 篇了,好几次都坚持不下去想要中断,不过每当看到群里的交 ...
- Vue中使用Cropper.js裁剪图片
Cropper.js是一款很好用的图片裁剪工具,可以对图片的尺寸.宽高比进行裁剪,满足诸如裁剪头像上传.商品图片编辑之类的需求. github: https://github.com/fengyuan ...
- vue初学:基础概念
一.vue使用步骤: 1.引包vue.js 2.html中写要操作的DOM节点 3.创建vue对象:new Vue({options}); 4.配置options:el:(要操作的对象,用选择器,同j ...
- vue中config/index.js:配置的详细理解
当我们需要和后台分离部署的时候,必须配置config/index.js: 用vue-cli 自动构建的目录里面 (环境变量及其基本变量的配置) var path = require('path') ...
- Vue教程:组件Component详解(六)
一.什么是组件? 组件 (Component) 是 Vue.js 最强大的功能之一.组件可以扩展 HTML 元素,封装可重用的代码.在较高层面上,组件是自定义元素,Vue.js 的编译器为它添加特殊功 ...
随机推荐
- day39-Spring 04-CGLIB的动态代理
JDK动态代理是有接口我给你创建一个类和你这个实现类是一样的, CGLIB不对实现接口的类生成代理,一个普通类也可以生成代理.CGLIB用继承的方式帮你生成代理对象.你父类有的方法我也有了,我想增强也 ...
- Gym - 101962B_Color Changing Sofa
题意:将一个沙发放到一个分成好几个色块(一个字母代表一种颜色)的房间里,要求沙发染成跟所在色块一样的颜色,沙发分成(0,1)两种,0可以染成一种颜色,1可以染成一种颜色(换句话说,沙发最多两种颜色), ...
- Mybatis表关联多对多
创建表 创建表对应的 JavaBean 对象 package com.tanlei.newer.model; import java.util.List; /** * @author:Mr.Tan * ...
- Java练习 SDUT-1689_斐波那契?
斐波那契? Time Limit: 1000 ms Memory Limit: 32768 KiB Problem Description 给出一个数列的递推公式,希望你能计算出该数列的第N个数.递推 ...
- redhat6.5安装yum
1.检查yum是否安装,默认情况下都是安装好的,总共4各包. rpm -qa |grep yum 卸载掉系统redhat自带的yum rpm -qa |grep yum |xargs rpm -e ...
- oracle函数 INITCAP(c1)
[功能]返回字符串并将字符串的第一个字母变为大写,其它字母小写; [参数]c1字符型表达式 [返回]字符型 [示例] SQL> select initcap('smith abc aBC') u ...
- HDU3844 Mining Your Own Business
HDU3844 Mining Your Own Business 问题描述John Digger是一个大型illudium phosdex矿的所有者.该矿山由一系列隧道组成,这些隧道在各个大型交叉口相 ...
- 模板—堆优化dijkstra
];]; void dijkstra(int s) { memset(dis,0x7f,sizeof(dis));ma(v); priority_queue<pair<int,int> ...
- uva 11754 Code Feat (中国剩余定理)
UVA 11754 一道中国剩余定理加上搜索的题目.分两种情况来考虑,当组合总数比较大的时候,就选择枚举的方式,组合总数的时候比较小时就选择搜索然后用中国剩余定理求出得数. 代码如下: #includ ...
- pytorch更新
Pytorch如何更新版本与卸载,使用pip,conda更新卸载Pytorch 2018年05月22日 07:33:52 醉雨轩Y 阅读数 19047 今天我们主要汇总如何使用使用ubuntu,C ...