你知道,什么时候用Vue计算属性吗?

x33g5p2x  于2021-12-01 转载在 Vue.js  
字(3.5k)|赞(0)|评价(0)|浏览(441)

本文分享自华为云社区《深入理解计算属性,知道什么时候该用Vue计算属性吗?》,作者: 前端老实人 。

计算属性

有些时候,我们在模板中放入了过多的逻辑,从而导致模板过重,且难以维护。例如:

  1. <div id="app">
  2. {{ message.split('').reverse().join('') }}
  3. </div>

碰到这样的情况,我们必须看一段时间才能意识到,这里是想要显示变量message的翻转字符串,而且,一旦我们想要在模板中多次使用翻转字符串时,会更加麻烦。 所以,当我们处理复杂逻辑时,都应该使用计算属性。

基础用法

计算属性是Vue配置对象中的属性,使用方式如下:

  1. <div id="app">
  2. <!-- 计算属性的值可以像data数据一样,直接被使用 -->
  3. {{ someComputed }}
  4. </div>
  5. const vm = new Vue({
  6. el: '#app',
  7. computed: {
  8. // 返回的值,就是计算属性的值
  9. someComputed () {
  10. return 'some values'
  11. }
  12. }
  13. })

例如,我们想要获取到一串字符串的翻转字符串,我们可以利用计算属性来做:

  1. <div id="app">
  2. <p>原始字符串: "{{ msg }}"</p>
  3. <p>翻转字符处啊: "{{ reversedMsg }}"</p>
  4. </div>
  5. const vm = new Vue({
  6. el: '#app',
  7. data: {
  8. msg: 'Hello'
  9. },
  10. computed: {
  11. reversedMsg: function () {
  12. return this.msg.split('').reverse().join('');
  13. }
  14. }
  15. })

我们可以看到,reversedMsg的值取决于msg的值,所以,当我们更改msg的值是,reversedMsg的值也会随之更改。

计算属性 vs 方法

其实,我们上述的功能,利用方法也可以实现,如:

  1. <div id="app">
  2. <p>原始字符串: "{{ msg }}"</p>
  3. <p>翻转字符串: "{{ reversedMsg() }}"</p>
  4. </div>
  5. const vm = new Vue({
  6. el: '#app',
  7. data: {
  8. msg: 'Hello'
  9. },
  10. methods: {
  11. reversedMsg: function () {
  12. return this.msg.split('').reverse().join('');
  13. }
  14. }
  15. })

虽然在表达式中调用方法也可以实现同样的效果,但是使用计算属性和使用方法有着本质的区别。 当使用方法时,每一次页面重新渲染,对应的方法都会重新执行一次,如:

  1. <div id="app">
  2. <p>{{ name }}</p>
  3. <p>{{ reversedMsg() }}</p>
  4. </div>
  5. const vm = new Vue({
  6. el: '#app',
  7. data: {
  8. msg: 'Hello',
  9. name: 'shanshan'
  10. },
  11. methods: {
  12. reversedMsg: function () {
  13. console.log('方法执行啦');
  14. return this.msg.split('').reverse().join('');
  15. }
  16. }
  17. })
  18. vm.name = 'duyi';

在上面的例子中我们可以看到,一旦更改name的值,页面会重新渲染,此刻控制台中打印出方法执行啦这串字符串,代表着reversedMsg这个函数执行了,但是我们并不需要该方法执行,因为改动的数据和这个函数没有任何关系,如果这个函数内的逻辑很复杂,那么对于性能来讲,也是一种消耗。

但是利用计算属性做,就不会有这样的现象出现,如:

  1. const vm = new Vue({
  2. el: '#app',
  3. data: {
  4. msg: 'Hello',
  5. name: 'shanshan'
  6. },
  7. computed: {
  8. reversedMsg: function () {
  9. console.log('计算执行啦');
  10. return this.msg.split('').reverse().join('');
  11. }
  12. }
  13. })
  14. vm.name = 'duyi';

此时可以看到,当给数据name重新赋值时,计算属性并没有执行。 所以,计算属性和方法的最本质的区别,是:计算属性是基于响应式依赖进行缓存的,计算属性的值一直存于缓存中,只要它依赖的data数据不改变,每次访问计算属性,都会立刻返回缓存的结果,而不是再次执行函数。而方法则是每次触发重新渲染,调用方法将总会再次执行函数。

那么,为什么需要缓存呢?

假如说,我们有一个计算属性A,它需要遍历一个巨大的数组并且做巨大的计算。然后我们需要使用到这个计算属性A,如果没有缓存,我们就会再次执行A的函数,这样性能开销就变得很大了。

深入计算属性

计算属性除了写成一个函数之外,还可以写成一个对象,对象内有两个属性,getter&setter,这两个属性皆为函数,写法如下:

  1. const vm = new Vue({
  2. el: '#app',
  3. computed: {
  4. fullName: {
  5. getter () {
  6. // 一些代码
  7. },
  8. setter () {
  9. // 一些代码
  10. }
  11. }
  12. }
  13. })

getter 读取

在前面,我们直接将计算属性写成了一个函数,这个函数即为getter函数。也就是说,计算属性默认只有getter。 getter的this,被自动绑定为Vue实例。

何时执行?

当我们去获取某一个计算属性时,就会执行get函数。

  1. const vm = new Vue({
  2. el: '#app',
  3. data: {
  4. msg: 'Hello'
  5. },
  6. computed: {
  7. reversedMsg: {
  8. getter () {
  9. return this.msg.split('').reverse().join('');
  10. }
  11. }
  12. }
  13. })

setter 设置

可选,set函数在给计算属性重新赋值时会执行。 参数:为被重新设置的值。 setter的this,被自动绑定为Vue实例。

  1. const vm = new Vue({
  2. el: '#app',
  3. data: {
  4. msg: 'Hello',
  5. firstStr: ''
  6. },
  7. computed: {
  8. reversedMsg: {
  9. getter () {
  10. return this.msg.split('').reverse().join('');
  11. },
  12. setter (newVal) {
  13. this.firstStr = newVal[0];
  14. }
  15. }
  16. }
  17. })

要注意,即使给计算属性赋了值,计算属性也不会改变,在重复一遍,只有当依赖的响应式属性变化了,计算属性才会重新计算。

练习_姓名筛选

  1. personArr: [
  2. {
  3. name: '',
  4. src: '.jpg',
  5. des: '颈椎不好',
  6. sex: 'm',
  7. id: '056482'
  8. },
  9. {
  10. name: '',
  11. src: '.jpg',
  12. des: '我是谁',
  13. sex: 'f',
  14. id: '157894'
  15. },
  16. {
  17. name: '',
  18. src: '.jpg', des: '我长得很好看',
  19. sex: 'f',
  20. id: '2849245'
  21. },
  22. {
  23. name: '',
  24. src: '.jpeg',
  25. des: '你没有见过陌生的脸',
  26. sex: 'm',
  27. id: '348515'
  28. },
  29. {
  30. name: '',
  31. src: '.jpeg',
  32. des: '瓜皮刘',
  33. sex: 'm',
  34. id: '478454'
  35. }
  36. ],

练习_全选商品

  1. courseList: [
  2. {
  3. poster: '.jpg',
  4. title: '',
  5. price: 1299,
  6. cart: 1,
  7. id: 0
  8. },
  9. {
  10. poster: '.jpg',
  11. title: '',
  12. price: 1148,
  13. cart: 1,
  14. id: 1595402664708
  15. },
  16. {
  17. poster: '.jpg',
  18. title: '',
  19. price: 1,
  20. cart: 1,
  21. id: 1596305473062
  22. },
  23. {
  24. poster: '.jpg',
  25. title: '',
  26. price: 1,
  27. cart: 1,
  28. id: 1595413512182
  29. },
  30. {
  31. poster: '.jpg',
  32. title: '',
  33. price: 12798,
  34. cart: 1,
  35. id: 1596302161181
  36. },
  37. {
  38. poster: '.jpg',
  39. title: '',
  40. price: 1,
  41. cart: 1,
  42. id: 1596300025301,
  43. },
  44. ]

点击关注,第一时间了解华为云新鲜技术~

相关文章

最新文章

更多