從vue源碼看props的用法
前言
平時寫vue的時候知道 props 有很多種用法,今天我們來看看vue內(nèi)部是怎么處理 props 中那么多的用法的。
vue提供的props的用法
1. 數(shù)組形式
props: ['name', 'value']
2. 對象形式
對象形式內(nèi)部也提供了三種寫法:
props: {
// 基礎(chǔ)的類型檢查
name: String,
// 多個可能的類型
value: [String, Number],
// 對象形式
id: {
type: Number,
required: true
}
}
props實現(xiàn)的原理
function normalizeProps (options: Object, vm: ?Component) {
const props = options.props
if (!props) return
const res = {}
let i, val, name
if (Array.isArray(props)) {
...
} else if (isPlainObject(props)) {
...
} else if (process.env.NODE_ENV !== 'production') {
...
}
options.props = res
}
normalizeProps 函數(shù)就是vue實際處理 props 的地方,從函數(shù)名的翻譯我們可以看出該函數(shù)的功能就是標準化 props 的值。該函數(shù)主要分成3部分:① 從 options 對象中獲取 props 的值并且定義一個res空對象;②幾個 if ... else ,分別根據(jù) props 值的不同類型來處理 res 對象;③ 用處理后的 res 對象覆蓋原來 options 對象的 props 屬性的值。
接下來看看那幾個 if ... else 的代碼:
if (Array.isArray(props)) {
i = props.length
while (i--) {
val = props[i]
if (typeof val === 'string') {
name = camelize(val)
res[name] = { type: null }
} else if (process.env.NODE_ENV !== 'production') {
warn('props must be strings when using array syntax.')
}
}
}
這個代碼實際就是處理props的值為數(shù)組的情況,例如: props: ['name', 'value'] 。使用while遍歷該數(shù)組,如果數(shù)組內(nèi)元素的類型不是字符串并且不是生產(chǎn)環(huán)境,那么就拋錯:‘props的值類型為數(shù)組時,數(shù)組里面的元素的類型就必須是字符串'。如果是字符串的情況下,使用 camelize 函數(shù)處理一下 val 的值,并且賦值給 name 變量。這里的 camelize 函數(shù)的實際作用就是將 '-' 轉(zhuǎn)換為駝峰。 camelize 函數(shù)具體的實現(xiàn)方式在后面分析。然后在 res 對象上面添加一個為 name 變量的屬性,該屬性的值為空對象 { type: null } 。
props: ['name', 'value'] 這種寫法經(jīng)過上面的處理后就會變成了下面這樣:
props: {
name: {
type: null
},
value: {
type: null
}
}
接下來看看下面這個 else if(isPlainObject(props)) ,這里的 isPlainObject 函數(shù)實際就是返回 props 的值是否為 object , isPlainObject 函數(shù)的具體實現(xiàn)我們也在后面分析。
else if (isPlainObject(props)) {
for (const key in props) {
val = props[key]
name = camelize(key)
res[name] = isPlainObject(val)
? val
: { type: val }
}
}
使用 for...in 遍歷props對象,和上面一樣使用 camelize 函數(shù)將 '-' 轉(zhuǎn)換為駝峰。這里有個三目運算:
res[name] = isPlainObject(val) ? val : { type: val }
判斷了一下 val 如果是 object ,那么在res對象上面添加一個為name變量的屬性,并且將該屬性的值設(shè)置為val。這個其實就是處理下面這種props的寫法:
props: {
// 對象形式
id: {
type: Number,
required: true
}
}
如果 val 不是 object ,那么也在res對象上面添加一個為name變量的屬性,并且將該屬性的值設(shè)置為{ type: val }。這個其實就是處理下面這種props的寫法:
props: {
// 基礎(chǔ)的類型檢查
name: String,
// 多個可能的類型
value: [String, Number],
}
經(jīng)過處理后props會變成了下面這樣:
props: {
name: {
type: String
},
value: {
type: [String, Number]
}
}
所以不管我們使用vue提供的 props 哪種寫法,最終vue都會幫我們轉(zhuǎn)換成下面這種類型:
props: {
name: {
...,
type: '類型'
}
}
接下來看看上面提到的util函數(shù) isPlainObject ,先把源碼貼出來。
const _toString = Object.prototype.toString
export function isPlainObject (obj: any): boolean {
return _toString.call(obj) === '[object Object]'
}
其實 Object.prototype.toString.call(obj) 的值為obj對象的類型。例如:
Object.prototype.toString.call({a: 1}) // [object Object]
Object.prototype.toString.call(new Date) // [object Date]
Object.prototype.toString.call([1]) // [object Array]
Object.prototype.toString.call(null) // [object Null]
接下來看看上面提到的util函數(shù) camelize ,還是先把源碼貼出來。
export function cached<F: Function> (fn: F): F {
const cache = Object.create(null)
return (function cachedFn (str: string) {
const hit = cache[str]
return hit || (cache[str] = fn(str))
}: any)
}
const camelizeRE = /-(\w)/g
export const camelize = cached((str: string): string => {
return str.replace(camelizeRE, (_, c) => c ? c.toUpperCase() : '')
})
這里定義了兩個函數(shù),分別是 cached 和 camelize ,其中 camelize 就是我們上面調(diào)用的, cached 是在 camelize 函數(shù)內(nèi)部調(diào)用的。
我們先來看看 camelize 函數(shù),其實 camelize 函數(shù)就是執(zhí)行 cached 后返回的一個函數(shù)。調(diào)用 cached 時傳入了一個箭頭函數(shù),箭頭函數(shù)內(nèi)部是調(diào)用了正則的 replace 方法,將傳入的 str 變量中匹配 /-(\w)/g 的變成大寫字母,并且返回 replace 后的值。(也就是將 - 轉(zhuǎn)換成駝峰)。
再來看看 cached 函數(shù),該函數(shù)傳入的變量其實就是 camelize 那里的箭頭函數(shù),首先定義了一個 cache 空對象,然后直接返回了 cachedFn 函數(shù)。我們在外部調(diào)用 camelize(key) 時,其實就是執(zhí)行了這里的了 cachedFn 函數(shù), str 的值就是傳入的 key 的值。很明顯這里是一個閉包,可以在外部調(diào)用 camelize 函數(shù)的時候可以修改或者讀取這里定義的 cache 對象的值。獲取 cache 對象中 key 為 str 變量值的屬性值賦值給 hit 變量。如果有hit變量的值,那么就直接返回hit的值,如果沒有就執(zhí)行 camelize 傳入的箭頭函數(shù),并且將箭頭函數(shù)的返回值賦值給 catche 對象的 str 屬性。如果下次調(diào)用 camelize 函數(shù)時傳入了相同的 str ,那么就不會執(zhí)行箭頭函數(shù),直接返回閉包中的 cache 對象的 str 屬性的值。這里是性能優(yōu)化的一種手段。
例如:第一次調(diào)用 camelize('name') 后, cache 對象的值就變成了{name: 'name'}。然后在其他地方再次調(diào)用 camelize('name') 時再次執(zhí)行 cachedFn 函數(shù),此時 hit 變量的值為'name'。直接返回 hit 變量的值,不會執(zhí)行傳入的箭頭函數(shù)。
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Vue使用codemirror實現(xiàn)一個可插入自定義標簽的textarea
這篇文章主要為大家詳細介紹了Vue如何使用codemirror實現(xiàn)一個可插入自定義標簽的textarea,感興趣的小伙伴可以跟隨小編一起學習一下2024-02-02
Vue報錯Component?name"Home"should?always?be?mult
這篇文章主要介紹了Vue報錯Component?name"Home"should?always?be?multi問題及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-09-09
關(guān)于uniapp的高級表單組件mosowe-form
這篇文章主要介紹了關(guān)于uniapp的高級表單組件mosowe-form,由于一些表單標簽改來改去比較繁瑣,重復性很多,且樣式布局啥的幾乎萬變不離其中,為了偷懶,開發(fā)了mosowe-form及mosowe-table兩款高級組件,需要的朋友可以參考下2023-04-04
vue2自定義組件通過rollup配置發(fā)布到npm的詳細步驟
這篇文章主要介紹了vue2自定義組件通過rollup配置發(fā)布到npm,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-03-03
Vue表單驗證?trigger:'blur'OR?trigger:'change&ap
利用?elementUI?實現(xiàn)表單元素校驗時,出現(xiàn)下拉框內(nèi)容選中后校驗不消失的異常校驗情形,這篇文章主要介紹了Vue表單驗證?trigger:‘blur‘?OR?trigger:‘change‘?區(qū)別,需要的朋友可以參考下2023-09-09

