Vue3 ref構(gòu)建響應(yīng)式變量失效問題及解決
vue3 ref構(gòu)建響應(yīng)式變量失效
問題描述
在Vue3中使用ref聲明響應(yīng)式變量,同時用函數(shù)對值進行變化,但是無法響應(yīng)式改變值
<template>
? <p>{{userName}}</p>
? <button @click='change()'>change</button>
</template>
<script>
? //引入定義響應(yīng)式數(shù)據(jù)的函數(shù)
? import {reactive} from 'vue';
? import {ref} from "@vue/reactivity"; //!!!!!注意,這里有個坑,ref必須是引用自vue,而非@vue/reactivity
? export default {
? name: 'App',
? //為Vue3的新特性提供統(tǒng)一入口,代碼都會在這個函數(shù)中添加
? //在beforecreated之前進行,因此無法訪問this,亦即無法訪問data和method
? setup(){
? ? //定義響應(yīng)式數(shù)據(jù):數(shù)據(jù)變化,模板中渲染會自動刷新
? ? // const obj=reactive({
? ? // ? userName:'jack',
? ? // });
? ? //只定義一個變量,可以使用ref將變量定義為響應(yīng)式
? ? let userName=ref('jack')
? ? console.log(userName);
? ? const change=()=> {
? ? ? userName.value='rose' ? ? //注意修改的是ref對象的value屬性,但是在template中使用的時候不需要再加value
? ? ? console.log(userName);
? ? }
? ? return {userName,change}
? },
}
</script>解決方案
不知道為什么,當引用為
import {ref} from "@vue/reactivity"時,就會出現(xiàn)不響應(yīng)的情況,但是只需要改為
import {ref} from "vue"vue3響應(yīng)式API之ref和reactive
我們知道 ref 函數(shù)和 reactive 函數(shù)用于實現(xiàn)數(shù)據(jù)的響應(yīng)性。但是開發(fā)中如何選擇使用 ref 和 reactive 呢?來說說 ref 和 reactive 的區(qū)別。
回顧
在 Vue3 版本之前,響應(yīng)數(shù)據(jù)在 data 函數(shù)中定義
<template>
<h1>{{ title }}</h1>
</template>
<script>
export default {
data() {
return {
title: "Hello, Vue!"
};
}
};
</script>Vue2 會遍歷 data 中的所有屬性,使用 Object.defineProperty 把每個 property 全部轉(zhuǎn)為 getter/setter,getter 用來收集依賴,setter 用來執(zhí)行 notify,發(fā)布更新事件。
Vue2 對每個屬性創(chuàng)建一個 Dep 對象,作為訂閱發(fā)布模式的中間機構(gòu)來收集依賴。Vue 追蹤這些依賴,在其被訪問和修改時通知變更。
Vue3
Vue3 中引入了 ref,reactive 來創(chuàng)建響應(yīng)式數(shù)據(jù):
<template>
<h1>{{ title }}</h1>
<h2>{{ data.author }}</h2>
<button @click=""changeTitle>修改title</button>
</template>
<script>
import { ref, reactive, toRefs } from "vue";
export default {
setup() {
const title = ref("Hello, Vue 3!");
// 修改
function changeTitle(){
title.value == "Hello, Vue3!"
}
const data = reactive({
author: "青年碼農(nóng)",
age: "18"
});
return { title, data, changeTitle };
}
};
</script>從上面的代碼我們大概可以看出區(qū)別。ref 的作用就是將一個原始數(shù)據(jù)類型轉(zhuǎn)換成一個響應(yīng)式數(shù)據(jù),原始數(shù)據(jù)類型共有 7 個,分別是:String、Number、BigInt、Boolean、Symbol、Undefined、Null。但是有個奇淫技巧,那就是 ref 也可以是對象。后面我們再說。reactive 的作用就是將一個對象轉(zhuǎn)換成一個響應(yīng)式對象。
- ref:
ref 的作用就是將一個原始數(shù)據(jù)類型轉(zhuǎn)換成一個帶有響應(yīng)式特性的數(shù)據(jù)類型。
const title = ref("Hello, Vue 3!");ref 接收參數(shù)并將其包裹在一個帶有 value 屬性的對象中返回,然后可以使用該屬性訪問或更改響應(yīng)式變量的值,比如上面的代碼我們使用 count.value 去修改值,如下:
title.value?=?"Hello,?Vue3!"
上面提到了 ref 也是可以接受對象類型
const data = ref({
author: "青年碼農(nóng)",
age: "18"
});這種也是可以的,賦值的時候就會有點別扭了。
data.value.author?=?"nmgwap";
ref 響應(yīng)式原理是依賴于 Object.defineProperty(),因此如果是對象,還是建議用 reactive。

- reactive:
reactive 返回對象的響應(yīng)式副本, 它將解包所有深層的 refs,同時維持 ref 的響應(yīng)性。一般我們用來實現(xiàn)對象或者數(shù)組的響應(yīng)性。
const data = reactive({
author: "青年碼農(nóng)",
age: "18"
});修改和普通對象沒區(qū)別,視圖會實時更新
data.author?=?"nmgwap"
注意:
ref 是針對原始數(shù)據(jù)類型 和 reactive 是用于對象 這兩個 API 都是為了給 JavaScript 普通的數(shù)據(jù)類型賦予響應(yīng)式特性(reactivity)。
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
關(guān)于Element-ui中table默認選中toggleRowSelection問題
這篇文章主要介紹了關(guān)于Element-ui中table默認選中toggleRowSelection問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-08-08
vue項目打包清除console.log的四種方法總結(jié)
大家在項目開發(fā)的時候,需要看看一些后端接口返回的結(jié)果,會多次使用console.log項目開發(fā)完成打包的時候,發(fā)現(xiàn)控制臺一堆的console.log,非常頭疼,下面這篇文章主要給大家介紹了關(guān)于vue項目打包清除console.log的四種方法,需要的朋友可以參考下2023-04-04
vue項目報錯Uncaught runtime errors的解決方案
使用vue-cli的vue項目,出現(xiàn)編譯錯誤或警告時,在瀏覽器中顯示全屏覆蓋,提示報錯Uncaught runtime errors,本文給大家介紹了vue項目報錯Uncaught runtime errors的解決方案,需要的朋友可以參考下2024-01-01
antd upload上傳組件如何獲取服務(wù)端返回數(shù)據(jù)
這篇文章主要介紹了antd upload上傳組件如何獲取服務(wù)端返回數(shù)據(jù)問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-02-02
vue 父組件獲取子組件里面的data數(shù)據(jù)(實現(xiàn)步驟)
在Vue中,父組件可以通過`ref`引用子組件,并通過`$refs`屬性來訪問子組件的數(shù)據(jù),下面分步驟給大家介紹vue 父組件獲取子組件里面的data數(shù)據(jù),感興趣的朋友一起看看吧2024-06-06
Vue使用emit傳參,父組件接收不到數(shù)據(jù)的問題及解決
這篇文章主要介紹了Vue使用emit傳參,父組件接收不到數(shù)據(jù)的問題及解決,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-09-09

