Vue.js中讓人容易忽略的API詳解
nextTick
nextTick 是 Vue.js 提供的一個(gè)函數(shù),并非瀏覽器內(nèi)置。nextTick 函數(shù)接收一個(gè)回調(diào)函數(shù) cb,在下一個(gè) DOM 更新循環(huán)之后執(zhí)行。比如下面的示例:
<template>
<div>
<p v-if="show" ref="node">內(nèi)容</p>
<button @click="handleShow">顯示</button>
</div>
</template>
<script>
export default {
data () {
return {
show: false
}
},
methods: {
handleShow () {
this.show = true;
console.log(this.$refs.node); // undefined
this.$nextTick(() => {
console.log(this.$refs.node); // <p>內(nèi)容</p>
});
}
}
}
</script>
當(dāng) show 被置為 true 時(shí),這時(shí) p 節(jié)點(diǎn)還未被渲染,因此打印出的是 undefined,而在 nextTick 的回調(diào)里,p 已經(jīng)渲染好了,這時(shí)能正確打印出節(jié)點(diǎn)。
nextTick 的源碼在github.com/vuejs/vue/b…,可以看到,Vue.js 使用了 Promise、setTimeout 和 setImmediate 三種方法來實(shí)現(xiàn) nextTick,在不同環(huán)境會(huì)使用不同的方法。
v-model 語法糖
v-model 常用于表單元素上進(jìn)行數(shù)據(jù)的雙向綁定,比如 <input>。除了原生的元素,它還能在自定義組件中使用。
v-model 是一個(gè)語法糖,可以拆解為 props: value 和 events: input。就是說組件必須提供一個(gè)名為 value 的 prop,以及名為 input 的自定義事件,滿足這兩個(gè)條件,使用者就能在自定義組件上使用 v-model。比如下面的示例,實(shí)現(xiàn)了一個(gè)數(shù)字選擇器:
<template>
<div>
<button @click="increase(-1)">減 1</button>
<span style="color: red;padding: 6px">{{ currentValue }}</span>
<button @click="increase(1)">加 1</button>
</div>
</template>
<script>
export default {
name: 'InputNumber',
props: {
value: {
type: Number
}
},
data () {
return {
currentValue: this.value
}
},
watch: {
value (val) {
this.currentValue = val;
}
},
methods: {
increase (val) {
this.currentValue += val;
this.$emit('input', this.currentValue);
}
}
}
</script>
props 一般不能在組件內(nèi)修改,它是通過父級(jí)修改的,因此實(shí)現(xiàn) v-model 一般都會(huì)有一個(gè) currentValue 的內(nèi)部 data,初始時(shí)從 value 獲取一次值,當(dāng) value 修改時(shí),也通過 watch 監(jiān)聽到及時(shí)更新;組件不會(huì)修改 value 的值,而是修改 currentValue,同時(shí)將修改的值通過自定義事件 input 派發(fā)給父組件,父組件接收到后,由父組件修改 value。所以,上面的數(shù)字選擇器組件可以有下面兩種使用方式:
<template>
<InputNumber v-model="value" />
</template>
<script>
import InputNumber from '../components/input-number/input-number.vue';
export default {
components: { InputNumber },
data () {
return {
value: 1
}
}
}
</script>
或:
<template>
<InputNumber :value="value" @input="handleChange" />
</template>
<script>
import InputNumber from '../components/input-number/input-number.vue';
export default {
components: { InputNumber },
data () {
return {
value: 1
}
},
methods: {
handleChange (val) {
this.value = val;
}
}
}
</script>
如果你不想用 value 和 input 這兩個(gè)名字,從 Vue.js 2.2.0 版本開始,提供了一個(gè) model 的選項(xiàng),可以指定它們的名字,所以數(shù)字選擇器組件也可以這樣寫:
<template>
<div>
<button @click="increase(-1)">減 1</button>
<span style="color: red;padding: 6px">{{ currentValue }}</span>
<button @click="increase(1)">加 1</button>
</div>
</template>
<script>
export default {
name: 'InputNumber',
props: {
number: {
type: Number
}
},
model: {
prop: 'number',
event: 'change'
},
data () {
return {
currentValue: this.number
}
},
watch: {
value (val) {
this.currentValue = val;
}
},
methods: {
increase (val) {
this.currentValue += val;
this.$emit('number', this.currentValue);
}
}
}
</script>
在 model 選項(xiàng)里,就可以指定 prop 和 event 的名字了,而不一定非要用 value 和 input,因?yàn)檫@兩個(gè)名字在一些原生表單元素里,有其它用處。
.sync 修飾符
如果你使用過 Vue.js 1.x,一定對(duì) .sync 不陌生。在 1.x 里,可以使用 .sync 雙向綁定數(shù)據(jù),也就是父組件或子組件都能修改這個(gè)數(shù)據(jù),是雙向響應(yīng)的。在 Vue.js 2.x 里廢棄了這種用法,目的是盡可能將父子組件解耦,避免子組件無意中修改了父組件的狀態(tài)。
不過在 Vue.js 2.3.0 版本,又增加了 .sync 修飾符,但它的用法與 1.x 的不完全相同。2.x 的 .sync 不是真正的雙向綁定,而是一個(gè)語法糖,修改數(shù)據(jù)還是在父組件完成的,并非在子組件。
仍然是數(shù)字選擇器的示例,這次不用 v-model,而是用 .sync,可以這樣改寫:
<template>
<div>
<button @click="increase(-1)">減 1</button>
<span style="color: red;padding: 6px">{{ value }}</span>
<button @click="increase(1)">加 1</button>
</div>
</template>
<script>
export default {
name: 'InputNumber',
props: {
value: {
type: Number
}
},
methods: {
increase (val) {
this.$emit('update:value', this.value + val);
}
}
}
</script>
用例:
<template>
<InputNumber :value.sync="value" />
</template>
<script>
import InputNumber from '../components/input-number/input-number.vue';
export default {
components: { InputNumber },
data () {
return {
value: 1
}
}
}
</script>
看起來要比 v-model 的實(shí)現(xiàn)簡(jiǎn)單多,實(shí)現(xiàn)的效果是一樣的。v-model 在一個(gè)組件中只能有一個(gè),但 .sync 可以設(shè)置很多個(gè)。.sync 雖好,但也有限制,比如:
- 不能和表達(dá)式一起使用(如 v-bind:title.sync="doc.title + '!'" 是無效的);
- 不能用在字面量對(duì)象上(如 v-bind.sync="{ title: doc.title }" 是無法正常工作的)。
$set
在上一節(jié)已經(jīng)介紹過 $set,有兩種情況會(huì)用到它:
由于 JavaScript 的限制,Vue 不能檢測(cè)以下變動(dòng)的數(shù)組:
- 當(dāng)利用索引直接設(shè)置一個(gè)項(xiàng)時(shí),例如:this.items[index] = value;
- 當(dāng)修改數(shù)組的長(zhǎng)度時(shí),例如:vm.items.length = newLength。
由于 JavaScript 的限制,Vue 不能檢測(cè)對(duì)象屬性的添加或刪除。
舉例來看,就是:
// 數(shù)組
export default {
data () {
return {
items: ['a', 'b', 'c']
}
},
methods: {
handler () {
this.items[1] = 'x'; // 不是響應(yīng)性的
}
}
}
使用 $set:
// 數(shù)組
export default {
data () {
return {
items: ['a', 'b', 'c']
}
},
methods: {
handler () {
this.$set(this.items, 1, 'x'); // 是響應(yīng)性的
}
}
}
以對(duì)象為例:
// 對(duì)象
export default {
data () {
return {
item: {
a: 1
}
}
},
methods: {
handler () {
this.item.b = 2; // 不是響應(yīng)性的
}
}
}
使用 $set:
// 對(duì)象
export default {
data () {
return {
item: {
a: 1
}
}
},
methods: {
handler () {
this.$set(this.item, 'b', 2); // 是響應(yīng)性的
}
}
}
另外,數(shù)組的以下方法,都是可以觸發(fā)視圖更新的,也就是響應(yīng)性的:
push()、pop()、shift()、unshift()、splice()、sort()、reverse()。
還有一種小技巧,就是先 copy 一個(gè)數(shù)組,然后通過 index 修改后,再把原數(shù)組整個(gè)替換,比如:
handler () {
const data = [...this.items];
data[1] = 'x';
this.items = data;
}
計(jì)算屬性的 set
計(jì)算屬性(computed)很簡(jiǎn)單,而且也會(huì)大量使用,但大多數(shù)時(shí)候,我們只是用它默認(rèn)的 get 方法,也就是平時(shí)的常規(guī)寫法,通過 computed 獲取一個(gè)依賴其它狀態(tài)的數(shù)據(jù)。比如:
computed: {
fullName () {
return `${this.firstName} ${this.lastName}`;
}
}
這里的 fullName 事實(shí)上可以寫為一個(gè) Object,而非 Function,只是 Function 形式是我們默認(rèn)使用它的 get 方法,當(dāng)寫為 Object 時(shí),還能使用它的 set 方法:
computed: {
fullName: {
get () {
return `${this.firstName} ${this.lastName}`;
},
set (val) {
const names = val.split(' ');
this.firstName = names[0];
this.lastName = names[names.length - 1];
}
}
}
計(jì)算屬性大多時(shí)候只是讀取用,使用了 set 后,就可以寫入了,比如上面的示例,如果執(zhí)行 this.fullName = 'Aresn Liang',computed 的 set 就會(huì)調(diào)用,firstName 和 lastName 會(huì)被賦值為 Aresn 和 Liang。
總結(jié)
到此這篇關(guān)于Vue.js中讓人容易忽略的API的文章就介紹到這了,更多相關(guān)Vue.js中易忽略的API內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
vue如何集成raphael.js中國(guó)地圖的方法示例
最近的數(shù)據(jù)統(tǒng)計(jì)項(xiàng)目中要用到中國(guó)地圖,也就是在地圖上動(dòng)態(tài)的顯示某個(gè)時(shí)間段某個(gè)省份地區(qū)的統(tǒng)計(jì)數(shù)據(jù),我們不需要flash,僅僅依靠raphael.js以及SVG圖像就可以完成地圖的交互操作。本文就給大家介紹了關(guān)于利用vue集成raphael.js中國(guó)地圖的相關(guān)資料,需要的朋友可以參考下。2017-08-08
vue項(xiàng)目中使用vue-layer彈框插件的方法
這篇文章主要介紹了vue項(xiàng)目中使用vue-layer彈框插件的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
python虛擬環(huán)境 virtualenv的簡(jiǎn)單使用
virtualenv是一個(gè)創(chuàng)建隔絕的Python環(huán)境的工具。這篇文章主要介紹了python虛擬環(huán)境 virtualenv的簡(jiǎn)單使用,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-01-01
Vue中自動(dòng)生成路由配置文件覆蓋路由配置的思路詳解
這篇文章主要介紹了Vue中自動(dòng)生成路由配置文件覆蓋路由配置的思路詳解,大概思路是讀取@/views下所有index.vue如果當(dāng)前文件下有包含相同路徑則認(rèn)為是它的子路由,需要的朋友可以參考下2024-05-05
Pinia進(jìn)階setup函數(shù)式寫法封裝到企業(yè)項(xiàng)目
這篇文章主要為大家介紹了Pinia進(jìn)階setup函數(shù)式寫法封裝到企業(yè)項(xiàng)目實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
Vue2中使用axios的3種方法實(shí)例總結(jié)
axios從功能上來說就是主要用于我們前端向后端發(fā)送請(qǐng)求,是基于http客戶端的promise,面向?yàn)g覽器和nodejs,下面這篇文章主要給大家介紹了關(guān)于Vue2中使用axios的3種方法,需要的朋友可以參考下2022-09-09
在vue中高德地圖引入和軌跡的繪制的實(shí)現(xiàn)
這篇文章主要介紹了在vue中高德地圖引入和軌跡的繪制的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10

