Vue面試中observable原理詳解
一、Observable 是什么

Observable 翻譯過來我們可以理解成可觀察的
我們先來看一下其在Vue中的定義
Vue.observable,讓一個(gè)對(duì)象變成響應(yīng)式數(shù)據(jù)。Vue 內(nèi)部會(huì)用它來處理 data 函數(shù)返回的對(duì)象
返回的對(duì)象可以直接用于渲染函數(shù)和計(jì)算屬性內(nèi),并且會(huì)在發(fā)生變更時(shí)觸發(fā)相應(yīng)的更新。也可以作為最小化的跨組件狀態(tài)存儲(chǔ)器
Vue.observable({ count : 1})
其作用等同于
new vue({ count : 1})
在 Vue 2.x 中,被傳入的對(duì)象會(huì)直接被 Vue.observable 變更,它和被返回的對(duì)象是同一個(gè)對(duì)象
在 Vue 3.x 中,則會(huì)返回一個(gè)可響應(yīng)的代理,而對(duì)源對(duì)象直接進(jìn)行變更仍然是不可響應(yīng)的
二、使用場(chǎng)景
在非父子組件通信時(shí),可以使用通常的bus或者使用vuex,但是實(shí)現(xiàn)的功能不是太復(fù)雜,而使用上面兩個(gè)又有點(diǎn)繁瑣。這時(shí),observable就是一個(gè)很好的選擇
創(chuàng)建一個(gè)js文件
// 引入vue
import Vue from 'vue
// 創(chuàng)建state對(duì)象,使用observable讓state對(duì)象可響應(yīng)
export let state = Vue.observable({
name: '張三',
'age': 38
})
// 創(chuàng)建對(duì)應(yīng)的方法
export let mutations = {
changeName(name) {
state.name = name
},
setAge(age) {
state.age = age
}
}
在.vue文件中直接使用即可
<template>
<div>
姓名:{{ name }}
年齡:{{ age }}
<button @click="changeName('李四')">改變姓名</button>
<button @click="setAge(18)">改變年齡</button>
</div>
</template>
import { state, mutations } from '@/store
export default {
// 在計(jì)算屬性中拿到值
computed: {
name() {
return state.name
},
age() {
return state.age
}
},
// 調(diào)用mutations里面的方法,更新數(shù)據(jù)
methods: {
changeName: mutations.changeName,
setAge: mutations.setAge
}
}
三、原理分析
源碼位置:src\core\observer\index.js
export function observe (value: any, asRootData: ?boolean): Observer | void {
if (!isObject(value) || value instanceof VNode) {
return
}
let ob: Observer | void
// 判斷是否存在__ob__響應(yīng)式屬性
if (hasOwn(value, '__ob__') && value.__ob__ instanceof Observer) {
ob = value.__ob__
} else if (
shouldObserve &&
!isServerRendering() &&
(Array.isArray(value) || isPlainObject(value)) &&
Object.isExtensible(value) &&
!value._isVue
) {
// 實(shí)例化Observer響應(yīng)式對(duì)象
ob = new Observer(value)
}
if (asRootData && ob) {
ob.vmCount++
}
return ob
}
Observer類
export class Observer {
value: any;
dep: Dep;
vmCount: number; // number of vms that have this object as root $data
constructor (value: any) {
this.value = value
this.dep = new Dep()
this.vmCount = 0
def(value, '__ob__', this)
if (Array.isArray(value)) {
if (hasProto) {
protoAugment(value, arrayMethods)
} else {
copyAugment(value, arrayMethods, arrayKeys)
}
this.observeArray(value)
} else {
// 實(shí)例化對(duì)象是一個(gè)對(duì)象,進(jìn)入walk方法
this.walk(value)
}
}
walk函數(shù)
walk (obj: Object) {
const keys = Object.keys(obj)
// 遍歷key,通過defineReactive創(chuàng)建響應(yīng)式對(duì)象
for (let i = 0; i < keys.length; i++) {
defineReactive(obj, keys[i])
}
}
defineReactive方法
export function defineReactive (
obj: Object,
key: string,
val: any,
customSetter?: ?Function,
shallow?: boolean
) {
const dep = new Dep()
const property = Object.getOwnPropertyDescriptor(obj, key)
if (property && property.configurable === false) {
return
}
// cater for pre-defined getter/setters
const getter = property && property.get
const setter = property && property.set
if ((!getter || setter) && arguments.length === 2) {
val = obj[key]
}
let childOb = !shallow && observe(val)
// 接下來調(diào)用Object.defineProperty()給對(duì)象定義響應(yīng)式屬性
Object.defineProperty(obj, key, {
enumerable: true,
configurable: true,
get: function reactiveGetter () {
const value = getter ? getter.call(obj) : val
if (Dep.target) {
dep.depend()
if (childOb) {
childOb.dep.depend()
if (Array.isArray(value)) {
dependArray(value)
}
}
}
return value
},
set: function reactiveSetter (newVal) {
const value = getter ? getter.call(obj) : val
/* eslint-disable no-self-compare */
if (newVal === value || (newVal !== newVal && value !== value)) {
return
}
/* eslint-enable no-self-compare */
if (process.env.NODE_ENV !== 'production' && customSetter) {
customSetter()
}
// #7981: for accessor properties without setter
if (getter && !setter) return
if (setter) {
setter.call(obj, newVal)
} else {
val = newVal
}
childOb = !shallow && observe(newVal)
// 對(duì)觀察者watchers進(jìn)行通知,state就成了全局響應(yīng)式對(duì)象
dep.notify()
}
})
}以上就是Vue面試中observable原理詳解的詳細(xì)內(nèi)容,更多關(guān)于Vue observable原理的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Vue動(dòng)態(tài)添加表單的實(shí)現(xiàn)方法
在Vue.js應(yīng)用中,動(dòng)態(tài)表單是一個(gè)常見的需求,尤其是當(dāng)表單字段的數(shù)量和類型需要根據(jù)用戶輸入或系統(tǒng)狀態(tài)動(dòng)態(tài)變化時(shí),本文將詳細(xì)介紹如何在Vue中實(shí)現(xiàn)動(dòng)態(tài)表單的創(chuàng)建,并通過多個(gè)示例展示具體的實(shí)現(xiàn)方法,需要的朋友可以參考下2024-09-09
Vue中常用rules校驗(yàn)規(guī)則(實(shí)例代碼)
這篇文章主要介紹了Vue中常用rules校驗(yàn)規(guī)則,本文通過實(shí)例代碼個(gè)大家介紹了一些校驗(yàn)方法,需要的朋友可以參考下2019-11-11
Vue中通過屬性綁定為元素綁定style行內(nèi)樣式的實(shí)例代碼
這篇文章主要介紹了Vue中通過屬性綁定為元素綁定style行內(nèi)樣式,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04
詳解如何在Vue3+TS的項(xiàng)目中使用NProgress進(jìn)度條
本文主要介紹了詳解如何在Vue3+TS的項(xiàng)目中使用NProgress進(jìn)度條,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06
使用Vue實(shí)現(xiàn)簡(jiǎn)單日歷效果
這篇文章主要為大家詳細(xì)介紹了使用Vue實(shí)現(xiàn)簡(jiǎn)單日歷效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08
vue獲取當(dāng)前點(diǎn)擊的元素并傳值的實(shí)例
下面小編就為大家分享一篇vue獲取當(dāng)前點(diǎn)擊的元素并傳值的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-03-03

