在Vue中,watch可以追蹤數(shù)據(jù)變化并對數(shù)據(jù)變化做出響應(yīng)。使用watch可以在數(shù)據(jù)變化時(shí)進(jìn)行一些自定義操作,例如發(fā)起一個(gè)API請求等。在Vue中使用watch需要在組件中定義watch對象,watch對象包含需要監(jiān)聽的數(shù)據(jù)名稱以及對數(shù)據(jù)變化做出響應(yīng)的函數(shù)。
watch: { foo: function(newValue, oldValue) { // 對foo數(shù)據(jù)變化做出響應(yīng)的代碼 } }
Vue有兩種方式進(jìn)行心跳檢測:Object.defineProperty和Proxy。當(dāng)Vue監(jiān)聽一個(gè)對象時(shí),Vue會(huì)使用Object.defineProperty對對象的屬性進(jìn)行監(jiān)聽。當(dāng)對象中的屬性被更新時(shí),對象的setter會(huì)觸發(fā)Vue的響應(yīng)式機(jī)制,導(dǎo)致視圖重新渲染。ProxyAPI提供了監(jiān)聽整個(gè)對象的功能,使用ProxyAPI的Vue版本會(huì)啟用Proxy,Proxy可以對整個(gè)對象進(jìn)行監(jiān)聽,不需要像Object.defineProperty一樣對對象中的每個(gè)屬性進(jìn)行偵聽。
// Vue使用Proxy的示例代碼 const vm = new Vue({ data: { foo: 'bar' } }) vm.foo = 'baz'
watch除了能監(jiān)聽簡單的數(shù)據(jù)類型之外,還可以監(jiān)聽復(fù)雜的數(shù)據(jù)類型,例如對象和數(shù)組。當(dāng)需要監(jiān)聽數(shù)組中的某個(gè)元素變化時(shí),需要使用數(shù)組的下標(biāo)進(jìn)行監(jiān)聽。
// 監(jiān)聽數(shù)組元素的示例代碼 watch: { 'foo.1': function(newValue, oldValue) { // 對foo中的第二個(gè)元素進(jìn)行處理的代碼 } }
watch的另外一個(gè)重要用途是在異步操作中處理數(shù)據(jù)。例如,在進(jìn)行API請求時(shí)需要在請求成功后更新組件的某個(gè)數(shù)據(jù),可以使用watch來監(jiān)聽API請求返回的數(shù)據(jù),并在數(shù)據(jù)變化時(shí)進(jìn)行更新。
watch: { 'apiData': function(newValue, oldValue) { // 在API請求返回?cái)?shù)據(jù)時(shí)對組件進(jìn)行更新的代碼 } }
watch還有一個(gè)可選選項(xiàng),即immediate。當(dāng)immediate被設(shè)置為true時(shí),watch會(huì)在組件創(chuàng)建時(shí)執(zhí)行一次對應(yīng)的處理函數(shù)。也就是說,watch在組件創(chuàng)建時(shí)就會(huì)執(zhí)行一次,并且在Vue實(shí)例中的data發(fā)生變化時(shí),watch才會(huì)繼續(xù)執(zhí)行。
// immediate為true的示例代碼 watch: { foo: { immediate: true, handler: function(newValue, oldValue) { // 對foo進(jìn)行處理的代碼 } } }
在使用watch時(shí)需要小心,使用不當(dāng)可能會(huì)導(dǎo)致性能問題。由于watch會(huì)檢測到data中的每一個(gè)屬性的變化,因此在使用watch時(shí)需要詢問自己:我真的需要監(jiān)聽這個(gè)數(shù)據(jù)嗎?如果不需要在data變化時(shí)做出響應(yīng),那么最好不要使用watch。