溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Vue中的methods、watch、computed的區(qū)別

發(fā)布時(shí)間:2020-10-20 17:16:06 來源:腳本之家 閱讀:171 作者:william 欄目:web開發(fā)

看到這個(gè)標(biāo)題就知道這篇文章接下來要講的內(nèi)容,我們在使用vue的時(shí)候methods、watch、computed這三個(gè)特性一定經(jīng)常使用,因?yàn)樗鼈兪欠浅5挠杏?,但是沒有徹底的理解它們的區(qū)別和各自的使用場景,也很難用好它們,希望接下來的介紹為你答疑解惑。

computed

我們先來看計(jì)算屬性:computed,光看名字也知道是用來干什么的,計(jì)算屬性當(dāng)然是用來計(jì)算的,但是是怎么計(jì)算的呢?計(jì)算屬性有兩個(gè)顯著的特點(diǎn):

  • 計(jì)算屬性計(jì)算時(shí)所依賴的屬性一定是響應(yīng)式依賴,否則計(jì)算屬性不會(huì)執(zhí)行
  • 計(jì)算屬性是基于依賴進(jìn)行緩存的,就是說在依賴沒有更新的情況,調(diào)用計(jì)算屬性并不會(huì)重新計(jì)算,可以減少開銷

我們來看一個(gè)相關(guān)的例子:

<div id="app">
 <div>{{ arr.join('') }}</div>
 <div>{{ arr1 }}</div>
 <div>{{ getDate }}</div>
 <div>{{ getDate1 }}</div>
</div>
var app = new Vue({
 el: '#app',
 data: {
  date: '',
  arr: ['a', 'b', 'c']
 },
 computed: {
  getDate () {
   return Date.now()
  },
  getDate1 () {
   return this.date
  },
  arr1 () {
   return this.arr.join('')
  }
 },
 created () {
  setInterval(() => {
   this.date = Date.now()
  }, 1000)
 }
})

看上面的例子,我們在寫vue的時(shí)候,經(jīng)常會(huì)碰到要對data的值進(jìn)行操作的情況,為了方便,總是會(huì)有人直接在模版中對data的值進(jìn)行計(jì)算操作,就像我上面例子中寫的在模版中將數(shù)組轉(zhuǎn)化為字符串,這樣寫有問題嗎?語法沒有問題,但是在模版中使用太多的計(jì)算,維護(hù)會(huì)是個(gè)問題,換個(gè)人來看代碼的時(shí)候會(huì)很痛苦,這種寫法就好像在html中插入js的邏輯運(yùn)算,可維護(hù)性極差。另外一個(gè)展示的就是computed的響應(yīng)式依賴的問題,當(dāng)我們調(diào)用getDate的時(shí)候返回的Date.now()返回的是一個(gè)非響應(yīng)式的依賴因此getDate返回的值不會(huì)變。

應(yīng)用場景

看了computed的特點(diǎn)之后,那么它的應(yīng)用場景是什么呢?個(gè)人看法,但不限于以下場景:

  • 復(fù)雜的渲染數(shù)據(jù)計(jì)算,用computed計(jì)算屬性可以減少一定計(jì)算開銷,增加可維護(hù)性
  • 從Vuex Store中收集相關(guān)信息,對Vuex中的數(shù)據(jù)做計(jì)算的時(shí)候的要特別注意computed的緩存屬性,在對Vuex中的對象值進(jìn)行屬性修改的時(shí)候,并不會(huì)觸發(fā)computed的中值的變化,這時(shí)需要Object.assign({},obj)對依賴對象進(jìn)行跟新返回一個(gè)新對象觸發(fā)依賴跟新
  • 表單校驗(yàn),這個(gè)應(yīng)用場景應(yīng)該是比較常見的,利用正則表達(dá)式對每個(gè)表單項(xiàng)的實(shí)時(shí)監(jiān)控,判斷表單是否可以提交

methods

methods大家應(yīng)該都會(huì)用,是vue中的方法屬性,所有的方法調(diào)用都會(huì)寫到這里面,大家用的最多也是在累似@click這樣事件調(diào)用中使用,但是很多人都忽視methods的另一個(gè)用法,就是在模版中使用methods,下面來看一個(gè)例子:

<div id="app">
 <div v-for="(item, idx) in arr">
  {{ matching(item) }}
 </div>
</div>
var app = new Vue({
 el: '#app',
 data: {
  arr: ['a', 'b', 'c'],
  obj: {a: 'hello', b: 'world'}
 },
 methods: {
  matching (key) {
   if (this.obj[key]) {
    return this.obj[key]
   } else {
    return 'not found'
   }
  }
 }
})

Vue中的methods、watch、computed的區(qū)別

上面的例子就是methods在模版中常常使用的一個(gè)場景,當(dāng)模版中的某個(gè)循環(huán)的值需要進(jìn)行一定邏輯運(yùn)算時(shí),這時(shí)候你就可以使用methods方法,將對應(yīng)的值傳入,然后計(jì)算出結(jié)果返回到模版顯示,這個(gè)時(shí)候用computed是沒法實(shí)現(xiàn)的,computed中你無法傳參,methods和computed除了這個(gè)不一樣之外,還有就是在methods中的計(jì)算是不會(huì)做緩存的,也就是你調(diào)用多少次就會(huì)計(jì)算多少次,相對computed的開銷要大一些。

watch

偵聽屬性是專門用來觀察和響應(yīng)vue實(shí)例上的數(shù)據(jù)變動(dòng),能使用watch屬性的場景基本上都可以使用computed屬性,而且computed屬性開銷小,性能高,因此能使用computed就盡量使用computed屬性,那么watch屬性是不是就沒用武之地了呢?當(dāng)執(zhí)行異步操作的時(shí)候你可能就必須用watch而不是computed了,下面看一個(gè)例子:

<div id="app">
 <div>{{ obj1.a }}</div>
</div>
var app = new Vue({
 el: '#app',
 data: {
  obj: {a: 'hello'},
  obj1: {a: 'hello'},
  test: 'aaa'
 },
 computed: {
  getObj () {
   setTimeout(() => {
    this.obj.a = this.test + 'word'
    return this.obj
   }, 1000)
  }
 },
 watch: {
  test () {
   setTimeout(() => {
    this.obj1.a = this.test + 'word'
   }, 1000)
  }
 },
 mounted () {
  this.test = 'hello'
 }
})

上述例子中,當(dāng)在模版中調(diào)用getObj.a時(shí),如果沒有setTimeout這異步操作,直接返回一個(gè)值是可以直接在模版中顯示的,但是由于加異步操作就會(huì)導(dǎo)致沒有返回值同時(shí)調(diào)用對象的屬性,就會(huì)報(bào)錯(cuò),而調(diào)用obj1.a卻不一樣,模版會(huì)先顯示hello,然后在觸發(fā)了watch屬性時(shí),setTimeout觸發(fā),一秒鐘之后模版會(huì)顯示helloword,這就watch中可以使用異步函數(shù),而computed不行的原因

總結(jié)

希望看了這篇文章能對你區(qū)分methods、computed、watch的用法能有所幫助。

這篇文章如果有錯(cuò)誤或不嚴(yán)謹(jǐn)?shù)牡胤剑瑲g迎批評指正,如果喜歡,歡迎點(diǎn)贊收藏

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI