淺析 Vue 3.0 的組裝式 API(一)
(一)響應式數(shù)據(jù)
1. 簡單例子
從最簡單的數(shù)據(jù)綁定開始,在 Vue 2.0 中,我們這樣將一個數(shù)據(jù)綁定到模板的指定位置:
在組件創(chuàng)建參數(shù)的 data 構(gòu)造函數(shù)中返回一個用來綁定的數(shù)據(jù)對象,其中有個 now 字段,會被渲染到模板內(nèi)的 .app > p 內(nèi)。
<template>
<div class="app">
<h1>Hello world!</h1>
<p>Now is: {{now.toString()}}</p>
</div>
</template>
<script>
// Vue 2.0
export default {
data() {
return {
now: new Date(),
};
},
};
</script>
用 Vue3 的組裝 API 實現(xiàn)的話,則是這樣:
// Vue 3.0
export default {
setup() {
return {
now: new Date(),
};
},
};
2. 更新數(shù)據(jù)
奇怪,看起來好像沒啥區(qū)別,只是把 data 改成了 setup 嗎?
并不是,假如我們現(xiàn)在對這個 DEMO 做個小改動,讓它每秒鐘刷新一次時間,用 Vue2 大概是這樣實現(xiàn):
// Vue 2.0
export default {
data() {
return {
now: new Date(),
};
},
mounted() {
setInterval(() => this.now = new Date(), 1000);
},
};
而 Vue3 的等效實現(xiàn)則為:
// Vue 3.0
import { ref, onMounted } from 'vue';
export default {
setup() {
const now = ref(new Date());
onMounted(() => {
setInterval(() => now.value = new Date(), 1000);
});
return {
now,
};
},
};
3. 對比分析
寫了太多 Vue 的我們可能已經(jīng)忘了,Vue2 的代碼從標準 JS 模塊的角度來看有多奇怪:
mounted中修改的this.now數(shù)據(jù)是在哪創(chuàng)建的?我們在模塊default對象的成員里并沒有找到對應字段,倒是在data內(nèi)返回的另一個對象中有這個字段;- 而
data中返回的now也不是真正的this.now,而是this.now的初始值,在data中setInterval修改now并不能更新渲染出來的時間; - 如果想復用這個數(shù)據(jù)和它的更新邏輯,你必須將這樣的結(jié)構(gòu)單獨寫一份,然后通過特殊的
mixin函數(shù)混入到當前組件的構(gòu)造參數(shù)內(nèi)。
這一切,是因為整個模塊 default 對象其實是 vm 對象的構(gòu)造參數(shù)。其背后隱藏了對象的創(chuàng)建邏輯,在構(gòu)造對象時構(gòu)造參數(shù)中的一些不同層級的字段被綁定到了 vm 對象上。
不少新手可能都犯過一個錯誤,在 data 中返回的數(shù)據(jù)字段和 props、methods 或者 computed 中的字段命名撞車(尤其是使用名為 data 的字段),在編碼階段并不能被 IDE 直接發(fā)現(xiàn)。就是因為上面的原因,這些字段創(chuàng)建時隸屬于不同的位置,在之后構(gòu)造時才被綁在了同一個對象上,導致了運行時才能發(fā)現(xiàn)的沖突。
Vue3 中,改成提供 ref、reactive、toRef、onMounted 等函數(shù)的形式實現(xiàn),例子中:
- 在
setup中看到的now即是用于綁定的this.now; - 修改
now.value即可看到頁面狀態(tài)的更新; - 如果要封裝這份數(shù)據(jù)處理,只需要將
now和onMounted處理提取到同一個函數(shù)內(nèi),再將now返回即可,不再需要黑盒的mixin處理。
可以說 Vue3 是直接將響應數(shù)據(jù)的創(chuàng)建決定權(quán)、生命周期的通知回調(diào),都通過 API 的形式交給了開發(fā)者,更直觀明了和可控。
4. API 說明
下面詳細說說常用的幾個響應式數(shù)據(jù)相關(guān) API:ref, reactive 和 toRefs。
(1) ref
上面例子中使用到的 ref,可以將一個數(shù)據(jù)包裝成響應式數(shù)據(jù)代理對象。
const count = ref(0); console.log(count.value); // => 0 count.value++; console.log(count.value); // => 1
當你修改代理對象的 count.value 屬性時,模板中使用到 count 的位置將響應數(shù)據(jù)的變化,更新視圖中的數(shù)據(jù)狀態(tài)。
(2) reactive
對于對象的響應式封裝,使用 ref 稍顯麻煩:
const state = ref({
count: 0,
});
console.log(state.value.count); // => 0
state.value.count++;
console.log(state.value.count); // => 1
這時可以改為使用 reactive,像操作普通對象的字段一樣修改 count 即可更新視圖:
const state = reactive({
count: 0,
});
console.log(state.count); // => 0
state.count++;
console.log(state.count); // => 1
對代理對象 state 添加新的字段也可觸發(fā)視圖更新。
(3) toRefs
有時候,對象的名字過長,我們想直接在模板內(nèi)使用對象內(nèi)部字段,直接使用解構(gòu)是不行的:
import { reactive } from 'vue';
export default {
setup() {
const position = reactive({
x: 0,
y: 0,
});
return {
// 錯誤,解構(gòu)出來的 x, y 并沒有響應式代理。綁定到模板上后,數(shù)據(jù)變化無法觸發(fā)視圖更新
...position,
};
},
};
這個情況下,使用 toRefs 處理后再解構(gòu)賦值即可:
import { reactive, toRefs } from 'vue';
export default {
setup() {
const position = reactive({
x: 0,
y: 0,
});
return {
...toRefs(position),
};
},
};
但需要注意,toRefs 只處理調(diào)用時 position 的現(xiàn)有字段,如果在之后對 position 增加新字段,將無法觸發(fā)視圖更新。
以上就是淺析 Vue 3.0 的組裝式 API(一)的詳細內(nèi)容,更多關(guān)于Vue 組裝式 API的資料請關(guān)注腳本之家其它相關(guān)文章!
- vue設(shè)置全局訪問接口API地址操作
- vue項目接口管理,所有接口都在apis文件夾中統(tǒng)一管理操作
- VUE使用axios調(diào)用后臺API接口的方法
- Vue3新特性之在Composition API中使用CSS Modules
- Vue.js中Line第三方登錄api的實現(xiàn)代碼
- 使用Vue Composition API寫出清晰、可擴展的表單實現(xiàn)
- 精讀《Vue3.0 Function API》
- 淺談Vue3.0新版API之composition-api入坑指南
- vue 使用外部JS與調(diào)用原生API操作示例
- vue 更改連接后臺的api示例
- Vue項目中Api的組織和返回數(shù)據(jù)處理的操作
相關(guān)文章
vue3+element-plus props中的變量使用 v-model 報錯及解決方案
這篇文章主要介紹了vue3+element-plus props中的變量使用 v-model 報錯及解決方案,prop 是單向數(shù)據(jù)流,這里只能用:model-value,不能用v-model,本文給大家介紹的非常詳細,需要的朋友可以參考下2023-10-10
基于Vue3+TypeScript實現(xiàn)鼠標框選功能
這篇文章主要介紹了基于Vue3+TypeScript實現(xiàn)鼠標框選功能,文中通過代碼示例給大家講解的非常纖細,對大家的學習或工作有一定的幫助,需要的朋友可以參考下2024-07-07
vue+element+Java實現(xiàn)批量刪除功能
這篇文章主要介紹了vue+element+Java實現(xiàn)批量刪除功能,代碼簡單易懂,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下2019-04-04
vue獲取token實現(xiàn)token登錄的示例代碼
最近新做了個vue項目,正好項目中有登錄部分,本文就詳細的介紹一下登錄部分的實現(xiàn),文中通過示例代碼介紹的非常詳細,感興趣的小伙伴們可以參考一下2021-11-11

