国产成人精品亚洲777人妖,欧美日韩精品一区视频,最新亚洲国产,国产乱码精品一区二区亚洲

您的位置:首頁技術(shù)文章
文章詳情頁

淺析 Vue 3.0 的組裝式 API(一)

瀏覽:137日期:2022-12-02 14:37:32

(一)響應(yīng)式數(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.0export default { data() { return { now: new Date(), }; },};</script>

用 Vue3 的組裝 API 實現(xiàn)的話,則是這樣:

// Vue 3.0export default { setup() { return { now: new Date(), }; },};

2. 更新數(shù)據(jù)

奇怪,看起來好像沒啥區(qū)別,只是把 data 改成了 setup 嗎?

并不是,假如我們現(xiàn)在對這個 DEMO 做個小改動,讓它每秒鐘刷新一次時間,用 Vue2 大概是這樣實現(xiàn):

// Vue 2.0export default { data() { return { now: new Date(), }; }, mounted() { setInterval(() => this.now = new Date(), 1000); },};

而 Vue3 的等效實現(xiàn)則為:

// Vue 3.0import { 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 的代碼從標(biāo)準(zhǔn) JS 模塊的角度來看有多奇怪:

mounted 中修改的 this.now 數(shù)據(jù)是在哪創(chuàng)建的?我們在模塊 default 對象的成員里并沒有找到對應(yīng)字段,倒是在 data 內(nèi)返回的另一個對象中有這個字段; 而 data 中返回的 now 也不是真正的 this.now,而是 this.now 的初始值,在 data 中 setInterval 修改 now 并不能更新渲染出來的時間; 如果想復(fù)用這個數(shù)據(jù)和它的更新邏輯,你必須將這樣的結(jié)構(gòu)單獨寫一份,然后通過特殊的 mixin 函數(shù)混入到當(dāng)前組件的構(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)造時才被綁在了同一個對象上,導(dǎo)致了運行時才能發(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 是直接將響應(yīng)數(shù)據(jù)的創(chuàng)建決定權(quán)、生命周期的通知回調(diào),都通過 API 的形式交給了開發(fā)者,更直觀明了和可控。

4. API 說明

下面詳細說說常用的幾個響應(yīng)式數(shù)據(jù)相關(guān) API:ref, reactive 和 toRefs。

(1) ref

上面例子中使用到的 ref,可以將一個數(shù)據(jù)包裝成響應(yīng)式數(shù)據(jù)代理對象。

const count = ref(0);console.log(count.value); // => 0count.value++;console.log(count.value); // => 1

當(dāng)你修改代理對象的 count.value 屬性時,模板中使用到 count 的位置將響應(yīng)數(shù)據(jù)的變化,更新視圖中的數(shù)據(jù)狀態(tài)。

(2) reactive

對于對象的響應(yīng)式封裝,使用 ref 稍顯麻煩:

const state = ref({ count: 0,});console.log(state.value.count); // => 0state.value.count++;console.log(state.value.count); // => 1

這時可以改為使用 reactive,像操作普通對象的字段一樣修改 count 即可更新視圖:

const state = reactive({ count: 0,});console.log(state.count); // => 0state.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 并沒有響應(yīng)式代理。綁定到模板上后,數(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)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Vue
相關(guān)文章:
主站蜘蛛池模板: 玉屏| 龙川县| 东安县| 彝良县| 砚山县| 措美县| 新龙县| 丹巴县| 石林| 浙江省| 延安市| 水富县| 方正县| 上栗县| 黄龙县| 东明县| 揭阳市| 黄骅市| 保亭| 东宁县| 兴山县| 大港区| 乌兰浩特市| 曲麻莱县| 皮山县| 潼南县| 安达市| 洪雅县| 定西市| 剑河县| 弥渡县| 大足县| 华坪县| 潼南县| 芮城县| 双鸭山市| 通化县| 永靖县| 图木舒克市| 利辛县| 荆州市|