這應該是全網最詳細的Vue3.5版本解讀

来源:https://www.cnblogs.com/heavenYJJ/p/18397413
-Advertisement-
Play Games

Vue3.5正式版在這兩天發佈了,網上已經有了不少關於Vue3.5版本的解讀文章。但是歐陽發現這些文章對3.5中新增的功能介紹都不是很全,所以導致不少同學有個錯覺,覺得Vue3.5版本不過如此 ...


前言

Vue3.5正式版在這兩天發佈了,網上已經有了不少關於Vue3.5版本的解讀文章。但是歐陽發現這些文章對3.5中新增的功能介紹都不是很全,所以導致不少同學有個錯覺,覺得Vue3.5版本不過如此,選擇跳過這個版本等下個大版本再去更新。所以歐陽寫了這篇超級詳細的Vue3.5版本解讀文章,小伙伴們可以看看在3.5版本中有沒有增加一些你期待的功能。

關註公眾號:【前端歐陽】,給自己一個進階vue的機會

版本號

這次的版本號是天元突破紅蓮螺岩,這是07年出的一個二次元動漫,歐陽是沒看過的。在此之前我一直以為這次的版本號會叫黑神話:悟空,可能悟空不夠二次元吧。

響應式

響應式相關的內容主要分為:重構響應式、響應式props支持解構、新增onEffectCleanup函數、新增base watch函數、新增onWatcherCleanup函數、新增pauseresume方法。

重構響應式

這次響應式的重構是屬於Vue內部優化,對於普通開發者來說是無感的。重構後記憶體占用減少了56%,優化手段主要是通過版本計數雙向鏈表數據結構,靈感來源於Preact signals。後續歐陽會出一系列關於響應式相關的源碼文章,大家可以關註一波歐陽。

響應式props支持解構

在3.5中響應式props支持解構終於正式穩定了,在沒有這個功能之前我們想要在js中訪問prop必須要這樣寫:props.name,否則name將會丟失響應式。

有了響應式props解構後,在js中我們就可以直接解構出name來使用,比如下麵這樣的代碼:

<script setup lang="ts">
const { name } = defineProps({
  name: String,
});

console.log(name);
</script>

defineProps搭配解構一起使用後,在編譯時就可以將name處理成props.name。編譯後簡化的代碼如下:

setup(__props) {
  console.log(__props.name);
  const __returned__ = {};
  return __returned__;
}

從上面的代碼可以看到console.log(name)經過編譯後變成了console.log(__props.name),這樣處理後name當然就不會丟失響應式了。

新增onEffectCleanup函數

在組件卸載之前或者下一次watchEffect回調執行之前會自動調用onEffectCleanup函數,有了這個函數後你就不需要在組件的beforeUnmount鉤子函數去統一清理一些timer了。比如下麵這個場景:

import { watchEffect, ref } from "vue";
import { onEffectCleanup } from "@vue/reactivity";

const flag = ref(true);
watchEffect(() => {
  if (flag.value) {
    const timer = setInterval(() => {
      // 做一些事情
      console.log("do something");
    }, 200);
    onEffectCleanup(() => {
      clearInterval(timer);
    });
  }
});

上面這個例子在watchEffect中會去註冊一個迴圈調用的定時器,如果不使用onEffectCleanup,那麼我們就需要在beforeUnmount鉤子函數中去清理定時器。

但是有了onEffectCleanup後,將clearInterval放在他的回調中就可以了。當組件卸載時會自動執行onEffectCleanup傳入的回調函數,也就是會執行clearInterval清除定時器。

還有一點值得註意的是onEffectCleanup函數目前沒有在vue包中暴露出來,如果你想使用可以像我這樣從@vue/reactivity包中導入onEffectCleanup函數。

新增base watch函數

我們之前使用的watch函數是和Vue組件以及生命周期一起實現的,他們是深度綁定的,所以watch函數代碼的位置在vue源碼中的runtime-core模塊中。

但是有的場景中我們只想使用vue的響應式功能,也就是vue源碼中的reactivity模塊,比如小程式vuemini。為此我們不得不將runtime-core模塊也導入到項目中,或者像vuemini一樣去手寫一個watch函數。

在3.5版本中重構了一個base watch函數,這個函數的實現和vue組件沒有一毛錢關係,所以他是在reactivity模塊中。詳情可以查看我之前的文章: Vue3.5新增的baseWatch讓watch函數和Vue組件徹底分手

還有一點就是這個base watch函數對於普通開發者來說沒有什麼影響,但是對於一些下游項目,比如vuemini來說是和受益的。

新增onWatcherCleanup函數

和前面的onEffectCleanup函數類似,在組件卸載之前或者下一次watch回調執行之前會自動調用onWatcherCleanup函數,同樣有了這個函數後你就不需要在組件的beforeUnmount鉤子函數去統一清理一些timer了。比如下麵這個場景:

import { watch, ref, onWatcherCleanup } from "vue";

watch(flag, () => {
  const timer = setInterval(() => {
    // 做一些事情
    console.log("do something");
  }, 200);
  onWatcherCleanup(() => {
    console.log("清理定時器");
    clearInterval(timer);
  });
});

onEffectCleanup函數不同的是我們可以從vue中import導入onWatcherCleanup函數。

新增pause和resume方法

有的場景中我們可能想在“一段時間中暫停一下”,不去執行watch或者watchEffect中的回調。等業務條件滿足後再去恢復執行watch或者watchEffect中的回調。在這種場景中pauseresume方法就能派上用場啦。

下麵這個是watchEffect的例子,代碼如下:

<template>
  <button @click="count++">count++</button>
  <button @click="runner2.pause()">暫停</button>
  <button @click="runner2.resume()">恢復</button>
</template>

<script setup lang="ts">
import { watchEffect } from "vue";

const count = ref(0);
const runner = watchEffect(() => {
  if (count.value > 0) {
    console.log(count.value);
  }
});
</script>

在上面的demo中,點擊count++按鈕後理論上每次都會執行一次watchEffect的回調。

但是當我們點擊了暫停按鈕後就會執行pause方法進行暫停,在暫停期間watchEffect的回調就不會執行了。

當我們再次點擊了恢復按鈕後就會執行resume方法進行恢復,此時watchEffect的回調就會重新執行。

console.log的結果如下圖:
console

從上圖中可以看到count列印到4後就沒接著列印了,因為我們執行了pause方法暫停了。當重新執行了resume方法恢復後可以看到count又重新開始列印了,此時從8開始列印了。

不光watchEffect可以執行pauseresume方法,watch一樣也可以執行pauseresume方法。代碼如下:

const runner = watch(count, () => {
  if (count.value > 0) {
    console.log(count.value);
  }
});

runner.pause()	// 暫停方法
runner.resume()	// 恢復方法

watch的deep選項支持傳入數字

在以前deep選項的值要麼是false,要麼是true,表明是否深度監聽一個對象。在3.5中deep選項支持傳入數字了,表明監控對象的深度。

比如下麵的這個demo:

const obj1 = ref({
  a: {
    b: 1,
    c: {
      d: 2,
      e: {
        f: 3,
      },
    },
  },
});

watch(
  obj1,
  () => {
    console.log("監聽到obj1變化");
  },
  {
    deep: 3,
  }
);

function changeDeep3Obj() {
  obj1.value.a.c.d = 20;
}

function changeDeep4Obj() {
  obj1.value.a.c.e.f = 30;
}

在上面的例子watchdeep選項值是3,表明監聽到對象的第3層。

changeDeep3Obj函數中就是修改對象的第3層的d屬性,所以能夠觸發watch的回調。

changeDeep4Obj函數是修改對象的第4層的f屬性,所以不能觸發watch的回調。

SSR服務端渲染

服務端渲染SSR主要有這幾個部分:新增useId函數、Lazy Hydration  懶載入水合、data-allow-mismatch

新增useId函數

有時我們需要生成一個隨機數塞到DOM元素上,比如下麵這個場景:

<template>
  <label :htmlFor="id">Do you like Vue3.5?</label>
  <input type="checkbox" name="vue3.5" :id="id" />
</template>

<script setup lang="ts">
const id = Math.random();
</script>

在這個場景中我們需要生成一個隨機數id,在普通的客戶端渲染中這個代碼是沒問題的。

但是如果這個代碼是在SSR服務端渲染中那麼就會報警告了,如下圖:
useId

上面報錯的意思是服務端和客戶端生成的id不一樣,因為服務端和客戶端都執行了一次Math.random()生成id。由於Math.random()每次執行的結果都不同,自然服務端和客戶端生成的id也不同。

useId函數的作用就是為瞭解決這個問題。

當然useId也可以用於客戶端渲染的一些場景,比如在列表中我們需要一個唯一鍵,但是服務端又沒有給我們,這時我們就可以使用useId給列表中的每一項生成一個唯一鍵。

Lazy Hydration  懶載入水合

非同步組件現在可以通過 defineAsyncComponent() API 的 hydrate 選項來控制何時進行水合。(歐陽覺得這個普通開發者用不上,所以就不細講了)

data-allow-mismatch

SSR中有的時候確實在服務端和客戶端生成的html不一致,比如在DOM上面渲染當前時間,代碼如下:

<template>
  <div>當前時間是:{{ new Date() }}</div>
</template>

這種情況是避免不了會出現前面useId例子中的那種警告,此時我們可以使用data-allow-mismatch屬性來幹掉警告,代碼如下:

<template>
  <div data-allow-mismatch>當前時間是:{{ new Date() }}</div>
</template>

Custom Element 自定義元素改進

這個歐陽也覺得平時大家都用不上,所以就不細講了。

Teleport組件新增defer延遲屬性

Teleport組件的作用是將children中的內容傳送到指定的位置去,比如下麵的代碼:

<div id="target"></div>
<Teleport to="#target">被傳送的內容</Teleport>

文案被傳送的內容最終會渲染在id="target"的div元素中。

在之前有個限制,就是不能將<div id="target">放在Teleport組件的後面。

這個也很容易理解DOM是從上向下開始渲染的,如果先渲染到Teleport組件。然後就會去找id的值為target的元素,如果找不到當然就不能成功的將Teleport組件的子節點傳送到target的位置。

在3.5中為瞭解決這個問題,在Teleport組件上新增了一個defer延遲屬性。

加了defer延遲屬性後就能將target寫在Teleport組件的後面,代碼如下:

<Teleport defer to="#target">被傳送的內容</Teleport>
<div id="target"></div>

defer延遲屬性的實現也很簡單,就是等這一輪渲染周期結束後再去渲染Teleport組件。所以就算是target寫在Teleport組件的後面,等到渲染Teleport組件的時候target也已經渲染到頁面上了。

useTemplateRef函數

vue3中想要訪問DOM和子組件可以使用ref進行模版引用,但是這個ref有一些讓人迷惑的地方。

比如定義的ref變數到底是一個響應式數據還是DOM元素?

還有template中ref屬性的值明明是一個字元串,比如ref="inputEl",怎麼就和script中同名的inputEl變數綁到一塊了呢?

3.5中的useTemplateRef函數就可以完美的解決了這些問題。

這是3.5之前使用ref訪問input輸入框的例子:

<input type="text" ref="inputEl" />

const inputEl = ref<HTMLInputElement>();

這個寫法很不符合編程直覺,不知道有多少同學和歐陽一樣最開始用vue3時會給ref屬性綁定一個響應式變數。比如這樣::ref="inputEl"

更加要命的是這樣寫還不會報錯,就是inputEl中的值一直是undefined

最後一番排查後才發現ref屬性應該是綁定的變數名稱:ref="inputEl"

使用useTemplateRef函數後就好多了,代碼如下:

<input type="text" ref="inputRef" />

const inputEl = useTemplateRef<HTMLInputElement>("inputRef");

使用useTemplateRef函數後會返回一個ref變數,useTemplateRef函數傳的參數是字元串"inputRef"

在template中ref屬性的值也是字元串"inputRef",所以useTemplateRef函數的返回值就指向了DOM元素input輸入框。這個比3.5之前的體驗要好很多了,詳情可以查看我之前的文章: 牛逼!Vue3.5的useTemplateRef讓ref操作DOM更加絲滑

總結

對於開發者來說Vue3.5版本中還是新增了許多有趣的功能的,比如:onEffectCleanup函數、onWatcherCleanup函數、pauseresume方法、watchdeep選項支持傳入數字、useId函數、Teleport組件新增defer延遲屬性、useTemplateRef函數。

這些功能在一些特殊場景中還是很有用的,歐陽的個人看法還是得將Vue升到3.5。

關註公眾號:【前端歐陽】,給自己一個進階vue的機會

另外歐陽寫了一本開源電子書vue3編譯原理揭秘,看完這本書可以讓你對vue編譯的認知有質的提升。這本書初、中級前端能看懂,完全免費,只求一個star。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 函數柯里化 含義: 可以大概理解為: 將fn(a,b,c)轉換為fn(a)(b)(c) 原函數: function sum(a,b){ return a+b } console.log(sum(1,2)) 柯里化後: function sum(a) { return function (b) { r ...
  • title: 如何在 Nuxt 3 中有效使用 TypeScript date: 2024/9/9 updated: 2024/9/9 author: cmdragon excerpt: 摘要:本文詳細介紹瞭如何在Nuxt 3項目中有效使用TypeScript,包括創建新項目、安裝TypeScrip ...
  • JavaScript速查表 本手冊絕大部分內容是從Airbnb JavaScript Style Guide精簡整理,將開發者們都明確的操作去掉,目的為了就是更快的速查。 此處為源地址。 譯制:HaleNing 目錄 基礎知識 類型 引用 對象 數組 解構 字元串 變數 屬性 測試 公共約束 註釋 ...
  • title: 使用 nuxi preview 命令預覽 Nuxt 應用 date: 2024/9/8 updated: 2024/9/8 author: cmdragon excerpt: 摘要:本文介紹瞭如何使用nuxi preview命令預覽Nuxt.js應用,包括安裝和準備環境、啟動預覽伺服器 ...
  • title: 使用 nuxi prepare 命令準備 Nuxt 項目 date: 2024/9/7 updated: 2024/9/7 author: cmdragon excerpt: 摘要:本文介紹nuxi prepare命令在Nuxt.js項目中的使用,該命令用於創建.nuxt目錄並生成類型 ...
  • 受 LabelImg 啟發的基於 web 的圖像標註工具,基於 Vue 框架 喲,網友們好,年更鴿子終於想起了他的博客園密碼。如標題所述,今天給大家帶來的是一個基於 vue2 的圖像標註工具。至於它誕生的契機呢,應該是我導 pass 掉了我的提議(讓甲方使用 LabelImg 進行數據標註),說是要 ...
  • title: 使用 nuxi init 創建全新 Nuxt 項目 date: 2024/9/6 updated: 2024/9/6 author: cmdragon excerpt: 摘要:本文介紹瞭如何使用nuxi init命令創建全新的Nuxt.js項目,包括安裝所需環境、命令使用方法、指定模板 ...
  • element-ui 的組件庫中沒有圖片下拉選擇組件,基於 el-select 組件做的改動並不能完全滿足需求,因此決定重寫一個。 從頭到尾做下來收穫很多,我決定把實現過程中遇到的問題記錄一下。 效果圖 線上試用地址 設計要點 接下來將上面代碼中的關鍵部分拆分介紹 1. 回顯選中的圖片和 label ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...