成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Vue 3.0 前瞻,體驗(yàn) Vue Function API

DrizzleX / 916人閱讀

摘要:與的一致有三個(gè)可選值,分別為在渲染后,即后才調(diào)用回調(diào)函數(shù),在渲染前,即前調(diào)用回調(diào)函數(shù),同步觸發(fā)。

最近 Vue 官方公布了 Vue 3.0 最重要的RFC:Function-based component API,并發(fā)布了兼容 Vue 2.0 版本的 plugin:vue-function-api,可用于提前體驗(yàn) Vue 3.0 版本的 Function-based component API。筆者出于學(xué)習(xí)的目的,提前在項(xiàng)目中嘗試了vue-function-api。

筆者計(jì)劃寫兩篇文章,本文為筆者計(jì)劃的第一篇,主要為筆者在體驗(yàn) Vue Function API 的學(xué)習(xí)心得。第二篇計(jì)劃寫閱讀vue-function-api的核心部分代碼原理,包括setupobservable、lifecycle。

本文閱讀時(shí)間約為15~20分鐘。

概述

Vue 2.x 及以前的高階組件的組織形式或多或少都會(huì)面臨一些問題,特別是在需要處理重復(fù)邏輯的項(xiàng)目中,一旦開發(fā)者組織項(xiàng)目結(jié)構(gòu)組織得不好,組件代碼極有可能被人詬病為“膠水代碼”。而在 Vue 2.x 及之前的版本,解決此類問題的辦法大致是下面的方案:

mixin

函數(shù)式組件

slots

筆者維護(hù)的項(xiàng)目也需要處理大量復(fù)用邏輯,在這之前,筆者一直嘗試使用mixin的方式來實(shí)現(xiàn)組件的復(fù)用。有些問題也一直會(huì)對(duì)開發(fā)者和維護(hù)者造成困惑,如一個(gè)組件同時(shí)mixin多個(gè)組件,很難分清對(duì)應(yīng)的屬性或方法寫在哪個(gè)mixin里。其次,mixin的命名空間沖突也可能造成問題。難以保證不同的mixin不用到同一個(gè)屬性名。為此,官方團(tuán)隊(duì)提出函數(shù)式寫法的意見征求稿,也就是RFC:Function-based component API。使用函數(shù)式的寫法,可以做到更靈活地復(fù)用組件,開發(fā)者在組織高階組件時(shí),不必在組件組織上考慮復(fù)用,可以更好地把精力集中在功能本身的開發(fā)上。

注:本文只是筆者使用vue-function-api提前體驗(yàn) Vue Function API ,而這個(gè) API 只是 Vue 3.0 的 RFC,而并非與最終 Vue 3.x API 一致。發(fā)布后可能有不一致的地方。
在 Vue 2.x 中使用

要想提前在Vue 2.x中體驗(yàn) Vue Function API ,需要引入vue-function-api,基本引入方式如下:

import Vue from "vue";
import { plugin as VueFunctionApiPlugin } from "vue-function-api";

Vue.use(VueFunctionApiPlugin);
基本組件示例

先來看一個(gè)基本的例子:



詳解 setup

setup函數(shù)是Vue Function API 構(gòu)建的函數(shù)式寫法的主邏輯,當(dāng)組件被創(chuàng)建時(shí),就會(huì)被調(diào)用,函數(shù)接受兩個(gè)參數(shù),分別是父級(jí)組件傳入的props和當(dāng)前組件的上下文context??聪旅孢@個(gè)例子,可以知道在context中可以獲取到下列屬性值:

const MyComponent = {
    props: {
        name: String
    },
    setup(props, context) {
        console.log(props.name);
        // context.attrs
        // context.slots
        // context.refs
        // context.emit
        // context.parent
        // context.root
    }
}
value & state

value函數(shù)創(chuàng)建一個(gè)包裝對(duì)象,它包含一個(gè)響應(yīng)式屬性value

那么為何要使用value呢,因?yàn)樵?b>JavaScript中,基本類型并沒有引用,為了保證屬性是響應(yīng)式的,只能借助包裝對(duì)象來實(shí)現(xiàn),這樣做的好處是組件狀態(tài)會(huì)以引用的方式保存下來,從而可以被在setup中調(diào)用的不同的模塊的函數(shù)以參數(shù)的形式傳遞,既能復(fù)用邏輯,又能方便地實(shí)現(xiàn)響應(yīng)式。

直接獲取包裝對(duì)象的值必須使用.value,但是,如果包裝對(duì)象作為另一個(gè)響應(yīng)式對(duì)象的屬性,Vue內(nèi)部會(huì)通過proxy來自動(dòng)展開包裝對(duì)象。同時(shí),在模板渲染的上下文中,也會(huì)被自動(dòng)展開。

import { state, value } from "vue-function-api";
const MyComponent = {
    setup() {
        const count = value(0);
        const obj = state({
            count,
        });
        console.log(obj.count) // 作為另一個(gè)響應(yīng)式對(duì)象的屬性,會(huì)被自動(dòng)展開

        obj.count++ // 作為另一個(gè)響應(yīng)式對(duì)象的屬性,會(huì)被自動(dòng)展開
        count.value++ // 直接獲取響應(yīng)式對(duì)象,必須使用.value

        return {
            count,
        };
    },
    template: ``,
};

如果某一個(gè)狀態(tài)不需要在不同函數(shù)中被響應(yīng)式修改,可以通過state創(chuàng)建響應(yīng)式對(duì)象,這個(gè)state創(chuàng)建的響應(yīng)式對(duì)象并不是包裝對(duì)象,不需要使用.value來取值。

watch & computed

watchcomputed的基本概念與 Vue 2.x 的watchcomputed一致,watch可以用于追蹤狀態(tài)變化來執(zhí)行一些后續(xù)操作,computed用于計(jì)算屬性,用于依賴屬性發(fā)生變化進(jìn)行重新計(jì)算。

computed返回一個(gè)只讀的包裝對(duì)象,和普通包裝對(duì)象一樣可以被setup函數(shù)返回,這樣就可以在模板上下文中使用computed屬性??梢越邮軆蓚€(gè)參數(shù),第一個(gè)參數(shù)返回當(dāng)前的計(jì)算屬性值,當(dāng)傳遞第二個(gè)參數(shù)時(shí),computed是可寫的。

import { value, computed } from "vue-function-api";

const count = value(0);
const countPlusOne = computed(() => count.value + 1);

console.log(countPlusOne.value); // 1

count.value++;
console.log(countPlusOne.value); // 2

// 可寫的計(jì)算屬性值
const writableComputed = computed(
    // read
    () => count.value + 1,
    // write
    val => {
        count.value = val - 1;
    },
);

watch第一個(gè)參數(shù)和computed類似,返回被監(jiān)聽的包裝對(duì)象屬性值,不過另外需要傳遞兩個(gè)參數(shù):第二個(gè)參數(shù)是回調(diào)函數(shù),當(dāng)數(shù)據(jù)源發(fā)生變化時(shí)觸發(fā)回調(diào)函數(shù),第三個(gè)參數(shù)是options。其默認(rèn)行為與 Vue 2.x 有所不同:

lazy:是否會(huì)在組件創(chuàng)建時(shí)就調(diào)用一次回調(diào)函數(shù),與 Vue 2.x 相反,lazy默認(rèn)是false,默認(rèn)會(huì)在組件創(chuàng)建時(shí)調(diào)用一次。

deep:與 Vue 2.x 的 deep 一致

flush:有三個(gè)可選值,分別為 "post"(在渲染后,即nextTick后才調(diào)用回調(diào)函數(shù)),"pre"(在渲染前,即nextTick前調(diào)用回調(diào)函數(shù)),"sync"(同步觸發(fā))。默認(rèn)值為"post"。

// double 是一個(gè)計(jì)算包裝對(duì)象
const double = computed(() => count.value * 2);

watch(double, value => {
    console.log("double the count is: ", value);
}); // -> double the count is: 0

count.value++; // -> double the count is: 2

當(dāng)watch多個(gè)被包裝對(duì)象屬性時(shí),參數(shù)均可以通過數(shù)組的方式進(jìn)行傳遞,同時(shí),與 Vue 2.x 的vm.$watch一樣,watch返回取消監(jiān)聽的函數(shù):

const stop = watch(
    [valueA, () => valueB.value],
    ([a, b], [prevA, prevB]) => {
        console.log(`a is: ${a}`);
        console.log(`b is: $`);
    }
);

stop();
注意:在RFC:Function-based component API初稿中,有提到effect-cleanup,是用于清理一些特殊情況的副作用的,目前已經(jīng)在提案中被取消了。
生命周期

所有現(xiàn)有的生命周期都有對(duì)應(yīng)的鉤子函數(shù),通過onXXX的形式創(chuàng)建,但有一點(diǎn)不同的是,destoryed鉤子函數(shù)需要使用unmounted代替:

import { onMounted, onUpdated, onUnmounted } from "vue-function-api";

const MyComponent = {
    setup() {
        onMounted(() => {
            console.log("mounted!");
        });
        onUpdated(() => {
            console.log("updated!");
        });
        // destroyed 調(diào)整為 unmounted
        onUnmounted(() => {
            console.log("unmounted!");
        });
    },
};
一些思考

上面的詳解部分,主要抽取的是 Vue Function API 的常見部分,并非RFC:Function-based component API的全部,例如其中的依賴注入,TypeScript類型推導(dǎo)等優(yōu)勢(shì),在這里,由于篇幅有限,想要了解更多的朋友,可以點(diǎn)開RFC:Function-based component API查看。個(gè)人也在Function-based component API討論區(qū)看到了更多地一些意見:

由于底層設(shè)計(jì),在setup取不到組件實(shí)例this的問題,這個(gè)問題在筆者嘗試體驗(yàn)時(shí)也遇到了,期待正式發(fā)布的 Vue 3.x 能夠改進(jìn)這個(gè)問題。

對(duì)于基本類型的值必須使用包裝對(duì)象的問題:在 RFC 討論區(qū),為了同時(shí)保證TypeScript類型推導(dǎo)、復(fù)用性和保留Vue的數(shù)據(jù)監(jiān)聽,包裝屬性必須使用.value來取值是討論最激烈的

關(guān)于包裝對(duì)象valuestate方法命名不清晰可能導(dǎo)致開發(fā)者誤導(dǎo)等問題,已經(jīng)在Amendment proposal to Function-based Component API這個(gè)提議中展開了討論:

setup() {
    const state = reactive({
        count: 0,
    });

    const double = computed(() => state.count * 2);

    function increment() {
        state.count++;
    }

    return {
        ...toBindings(state), // retains reactivity on mutations made to `state`
        double,
        increment,
    };
}

引入reactive API 和 binding API,其中reactive API 類似于 state API , binding API 類似于 value API。

之前使用的方法名state在 Vue 2.x 中可能被用作組件狀態(tài)對(duì)象,導(dǎo)致變量命名空間的沖突問題,團(tuán)隊(duì)認(rèn)為將state API 更名為 reactive 更為優(yōu)雅。開發(fā)者能夠?qū)懗?b>const state = ... ,然后通過state.xxxx這種方式來獲取組件狀態(tài),這樣也相對(duì)而言自然一些。

value方法用于封裝基本類型時(shí),確實(shí)會(huì)出現(xiàn)不夠優(yōu)雅的.value的情況,開發(fā)者可能會(huì)在直接對(duì)包裝對(duì)象取值時(shí)忘記使用.value,修正方案提出的 reactive API,其含義是創(chuàng)建響應(yīng)式對(duì)象,初始化狀態(tài)state就使用reactive創(chuàng)建,可保留每項(xiàng)屬性的gettersetter,這么做既滿足類型推導(dǎo),也可以保留響應(yīng)式引用,從而可在不同模塊中共享狀態(tài)值的引用。

reactive可能導(dǎo)致下面的問題,需要引入binding API。 解決,如使用reactive創(chuàng)建的響應(yīng)式對(duì)象,對(duì)其使用拓展運(yùn)算符...時(shí),則會(huì)丟失對(duì)象的gettersetter,提供toBindings方法能夠保留狀態(tài)的響應(yīng)式。

下一篇文章中,筆者將閱讀vue-function-api的核心部分代碼原理,包括setup、observable、lifecycle等,從內(nèi)部探索 Vue Function API 可能帶給我們的改變。

當(dāng)然,目前 Vue Function API 還處在討論階段,Vue 3.0 還處在開發(fā)階段,還是期待下半年 Vue 3.0 的初版問世吧,希望能給我們帶來更多的驚喜。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/106643.html

相關(guān)文章

  • 面向未來編程(Future-Oriented Programming),建設(shè)未來 Vue 生態(tài)

    摘要:狀態(tài)管理社區(qū)當(dāng)前最流行的狀態(tài)管理方案分別是單向數(shù)據(jù)流的和基于的。目前的現(xiàn)狀主要是由于并沒有將其底層的曝光出來。是一個(gè)開源項(xiàng)目,完全兼容的,擁簇面向未來編程,致力于加速未來生態(tài)圈的建設(shè)。 概要 Vue 的作者尤雨溪最近公布了 3.0 版本最重要的 RFC Vue Function-based API。在 React 推出 Hooks 后不久,Vue 社區(qū)也迎來了自己的組件邏輯復(fù)用機(jī)制。 ...

    J4ck_Chan 評(píng)論0 收藏0
  • 預(yù)計(jì)今年發(fā)布的Vue3.0到底有什么不一樣的地方?

    摘要:模板語法的將保持不變?;诘挠^察者機(jī)制目前,的反應(yīng)系統(tǒng)是使用的和。為了繼續(xù)支持,將發(fā)布一個(gè)支持舊觀察者機(jī)制和新版本的構(gòu)建。 showImg(https://segmentfault.com/img/remote/1460000017862774?w=1898&h=796); 還有幾個(gè)月距離vue2的首次發(fā)布就滿3年了,而vue的作者尤雨溪也在去年年末發(fā)布了關(guān)于vue3.0的計(jì)劃,如果不...

    fnngj 評(píng)論0 收藏0
  • 精讀《Vue3.0 Function API

    摘要:拿到的都是而不是原始值,且這個(gè)值會(huì)動(dòng)態(tài)變化。精讀對(duì)于的與,筆者做一些對(duì)比。因此采取了作為優(yōu)化方案只有當(dāng)?shù)诙€(gè)依賴參數(shù)變化時(shí)才返回新引用。不需要使用等進(jìn)行性能優(yōu)化,所有性能優(yōu)化都是自動(dòng)的。前端精讀幫你篩選靠譜的內(nèi)容。 1. 引言 Vue 3.0 的發(fā)布引起了軒然大波,讓我們解讀下它的 function api RFC 詳細(xì)了解一下 Vue 團(tuán)隊(duì)是怎么想的吧! 首先官方回答了幾個(gè)最受關(guān)注的...

    voyagelab 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<