成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術文章
文章詳情頁

vue使用mixins優化組件

瀏覽:59日期:2022-09-30 11:27:53

vue 提供了 mixins 這個 API,可以讓我們將組件中的可復用功能抽取出來,放入 mixin 中,然后在組件中引入 mixin,可以讓組件顯得不再臃腫,提高了代碼的可復用性。

如何理解 mixins 呢 ?我們可以將 mixins 理解成一個數組,數組中有單或多個 mixin,mixin 的本質就是一個 JS 對象,它可以有 data、created、methods 等等 vue 實例中擁有的所有屬性,甚至可以在 mixins 中再次嵌套 mixins,It’s amazing !

舉個簡單的栗子:

<div id='app'> <h1>{{ message }}</h1></div>

const myMixin = { data() { return { message: ’this is mixin message’ } }, created() { console.log(’mixin created’) }}const vm = new Vue({ el: ’#app’, mixins: [myMixin], data() { return { message: ’this is vue instance message’ } }, created() { console.log(this.message) // => Root Vue Instance console.log(’vue instance created’) // => created myMixin // => created Root Vue Instance }})

mixins 與 Vue Instance 合并時,會將 created 等鉤子函數合并成數組,mixins 的鉤子優先調用,當 data、methods 對象鍵值沖突時,以組件優先。

PS: 如果對 mixins 的概念還不太清的小伙伴,可以去 vue 官方文檔 看一下 vue mixins 的基本概念和用法。

mixins 實現

那 mixins 是如何實現的呢 ?當 vue 在實例化的時候,會調用 mergeOptions 函數進行 options 的合并,函數申明在 vue/src/core/util/options.js 文件。

export function mergeOptions( parent: Object, child: Object, vm?: Component): Object { ... // 如果有 child.extends 遞歸調用 mergeOptions 實現屬性拷貝 const extendsFrom = child.extends if (extendsFrom) { parent = mergeOptions(parent, extendsFrom, vm) } // 如果有 child.mixins 遞歸調用 mergeOptions 實現屬性拷貝 if (child.mixins) { for (let i = 0, l = child.mixins.length; i < l; i++) { parent = mergeOptions(parent, child.mixins[i], vm) } } // 申明 options 空對象,用來保存屬性拷貝結果 const options = {} let key // 遍歷 parent 對象,調用 mergeField 進行屬性拷貝 for (key in parent) { mergeField(key) } // 遍歷 parent 對象,調用 mergeField 進行屬性拷貝 for (key in child) { if (!hasOwn(parent, key)) { mergeField(key) } } // 屬性拷貝實現方法 function mergeField(key) { // 穿透賦值,默認為 defaultStrat const strat = strats[key] || defaultStrat options[key] = strat(parent[key], child[key], vm, key) } return options}

為了保持代碼簡潔,已經將 mergeOptions 函數不重要的代碼刪除,剩余部分我們慢慢來看。

const extendsFrom = child.extendsif (extendsFrom) { parent = mergeOptions(parent, extendsFrom, vm)}

首先申明 extendsFrom 變量保存 child.extends,如果 extendsFrom 為真,遞歸調用 mergeOptions 進行屬性拷貝,并且將 merge 結果保存到 parent 變量。

if (child.mixins) { for (let i = 0, l = child.mixins.length; i < l; i++) { parent = mergeOptions(parent, child.mixins[i], vm) }}

如果 child.mixins 為真,循環 mixins 數組,遞歸調用 mergeOptions 實現屬性拷貝,仍舊將 merge 結果保存到 parent 變量。

接下來是關于 parent、child 的屬性賦值:

const options = {}let keyfor (key in parent) { mergeField(key)}for (key in child) { if (!hasOwn(parent, key)) { mergeField(key) }}

申明 options 空對象,用來保存屬性拷貝的結果,也作為遞歸調用 mergeOptions 的返回值。

這里首先會調用 for...in 對 parent 進行循環,在循環中不斷調用 mergeField 函數。

接著調用 for...in 對 child 進行循環,這里有點不太一樣,會調用 hasOwn 判斷 parent 上是否有這個 key,如果沒有再調用 mergeField 函數,這樣避免了重復調用。

那么這個 mergeField 函數到底是用來做什么的呢?

function mergeField(key) { // 穿透賦值,默認為 defaultStrat const strat = strats[key] || defaultStrat options[key] = strat(parent[key], child[key], vm, key)}

mergeField 函數接收一個 key,首先會申明 strat 變量,如果 strats[key] 為真,就將 strats[key] 賦值給 strat。

const strats = config.optionMergeStrategies...optionMergeStrategies: Object.create(null),...

strats 其實就是 Object.create(null),Object.create 用來創建一個新對象,strats 默認是調用 Object.create(null) 生成的空對象。

順便說一句,vue 也向外暴露了 Vue.config.optionMergeStrategies,可以實現自定義選項合并策略。

如果 strats[key] 為假,這里會用 || 做穿透賦值,將 defaultStrat 默認函數賦值給 strat。

const defaultStrat = function(parentVal: any, childVal: any): any { return childVal === undefined ? parentVal : childVal}

defaultStrat 函數返回一個三元表達式,如果 childVal 為 undefined,返回 parentVal,否則返回 childVal,這里主要以 childVal 優先,這也是為什么有 component > mixins > extends 這樣的優先級。

mergeField 函數最后會將調用 strat 的結果賦值給 options[key]。

mergeOptions 函數最后會 merge 所有 options、 mixins、 extends,并將 options 對象返回,然后再去實例化 vue。

鉤子函數的合并

我們來看看鉤子函數是怎么進行合并的。

function mergeHook( parentVal: ?Array<Function>, childVal: ?Function | ?Array<Function>): ?Array<Function> { return childVal ? parentVal ? parentVal.concat(childVal) : Array.isArray(childVal) ? childVal : [childVal] : parentVal}LIFECYCLE_HOOKS.forEach(hook => { strats[hook] = mergeHook})

循環 LIFECYCLE_HOOKS 數組,不斷調用 mergeHook 函數,將返回值賦值給 strats[hook]。

export const LIFECYCLE_HOOKS = [ ’beforeCreate’, ’created’, ’beforeMount’, ’mounted’, ’beforeUpdate’, ’updated’, ’beforeDestroy’, ’destroyed’, ’activated’, ’deactivated’, ’errorCaptured’]

LIFECYCLE_HOOKS 就是申明的 vue 所有的鉤子函數字符串。

mergeHook 函數會返回 3 層嵌套的三元表達式。

return childVal ? parentVal ? parentVal.concat(childVal) : Array.isArray(childVal) ? childVal : [childVal] : parentVal

第一層,如果 childVal 為真,返回第二層三元表達式,如果為假,返回 parentVal。

第二層,如果 parentVal 為真,返回 parentVal 和 childVal 合并后的數組,如果 parentVal 為假,返回第三層三元表達式。

第三層,如果 childVal 是數組,返回 childVal,否則將 childVal 包裝成數組返回。

new Vue({ created: [ function() { console.log(’沖沖沖!’) }, function() { console.log(’鴨鴨鴨!’) } ]})// => 沖沖沖!// => 鴨鴨鴨!項目實踐

使用 vue 的小伙伴們,當然也少不了在項目中使用 element-ui。比如使用 Table 表格的時候,免不了申明 tableData、total、pageSize 一些 Table 表格、Pagination 分頁需要的參數。

我們可以將重復的 data、methods 寫在一個 tableMixin 中。

export default { data() { return { total: 0, pageNo: 1, pageSize: 10, tableData: [], loading: false } }, created() { this.searchData() }, methods: { // 預申明,防止報錯 searchData() {}, handleSizeChange(size) { this.pageSize = size this.searchData() }, handleCurrentChange(page) { this.pageNo = page this.searchData() }, handleSearchData() { this.pageNo = 1 this.searchData() } }}

當我們需要使用時直接引入即可:

import tableMixin from ’./tableMixin’export default { ... mixins: [tableMixin], methods: { searchData() { ... } }}

我們在組件內會重新申明 searchData 方法。類似這種 methods 對象形式的 key,如果 key 相同,組件內的 key 會覆蓋 tableMixin 中的 key。

當然我們也可以在 mixins 中嵌套 mixins,申明 axiosMixin:

import tableMixin from ’./tableMixin’export default { mixins: [tableMixin], methods: { handleFetch(url) { const { pageNo, pageSize } = this this.loading = true this.axios({method: ’post’,url,data: { ...this.params, pageNo, pageSize} }).then(({ data = [] }) => { this.tableData = data this.loading = false}).catch(error => { this.loading = false}) } }}

引入 axiosMixin:

import axiosMixin from ’./axiosMixin’export default { ... mixins: [axiosMixin], created() { this.handleFetch(’/user/12345’) }}

在 axios 中,我們可以預先處理 axios 的 success、error 的后續調用,是不是少寫了很多代碼。

extend

順便講一下 extend,與 mixins 相似,只能傳入一個 options 對象,并且 mixins 的優先級比較高,會覆蓋 extend 同名 key 值。

// 如果有 child.extends 遞歸調用 mergeOptions 實現屬性拷貝const extendsFrom = child.extendsif (extendsFrom) { parent = mergeOptions(parent, extendsFrom, vm)}// 如果有 child.mixins 遞歸調用 mergeOptions 實現屬性拷貝if (child.mixins) { for (let i = 0, l = child.mixins.length; i < l; i++) { parent = mergeOptions(parent, child.mixins[i], vm) }}

// 如果有 child.extends 遞歸調用 mergeOptions 實現屬性拷貝const extendsFrom = child.extendsif (extendsFrom) { parent = mergeOptions(parent, extendsFrom, vm)}// 如果有 child.mixins 遞歸調用 mergeOptions 實現屬性拷貝if (child.mixins) { for (let i = 0, l = child.mixins.length; i < l; i++) { parent = mergeOptions(parent, child.mixins[i], vm) }}

在 mergeOptions 函數中,會先對 extends 進行屬性拷貝,然后再對 mixin 進行拷貝,在調用 mergeField 函數的時候會優先取 child 的 key。

雖然 extends 的同名 key 會被 mixins 的覆蓋,但是 extends 是優先執行的。

總結

注意一下 vue 中 mixins 的優先級,component > mixins > extends。

我們暫且將 mixins 稱作是組件模塊化,靈活運用組件模塊化,可以將組件內的重復代碼提取出來,實現代碼復用,也使我們的代碼更加清晰,效率也大大提高。

當然,mixins 還有更加神奇的操作等你去探索。

以上就是vue使用mixins優化組件的詳細內容,更多關于vue 用mixins優化組件的資料請關注好吧啦網其它相關文章!

標簽: Vue
相關文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
久久精品五月婷婷| 欧美这里有精品| 在线精品视频一区二区三四| 亚洲欧美日韩久久精品| 欧美91福利在线观看| 欧美一区欧美二区| 久久99精品久久久久久国产越南 | 国产精品日韩二区| 国产欧美一区二区精品性色 | 亚洲成人资源网| 国产精品hd| 国产精品午夜免费| 你懂的亚洲视频| 日本一区二区在线不卡| 欧美不卡福利| 国产精品传媒视频| 国产一区二区三区四区hd| 亚洲国产精品二十页| www.色综合.com| 337p粉嫩大胆色噜噜噜噜亚洲 | 99国产精品一区| 蜜臀91精品一区二区三区| 精品一区二区国语对白| 欧美在线观看视频一区二区| 男人的j进女人的j一区| 在线观看av一区| 美女国产一区二区| 在线观看日韩一区| 奇米四色…亚洲| 亚洲青青青在线视频| 中文一区二区| 亚洲国产精品ⅴa在线观看| 国产精品啊啊啊| 日韩精品久久久久久| 免费在线成人av| 日韩黄色免费电影| 免费成人在线影院| 免费观看在线色综合| 久久久久久久久久码影片| 成人午夜av影视| 大桥未久av一区二区三区中文| 国产精品久久久久久妇女6080| 久久综合狠狠综合久久综合88| 91久久精品网| 成人激情动漫在线观看| 黑人一区二区| 亚洲色大成网站www久久九九| 中文亚洲字幕| 日韩精品乱码免费| 欧美一区二区三区视频免费| 欧美成人免费在线| 亚洲美女屁股眼交| 色94色欧美sute亚洲线路一ni| 国内精品伊人久久久久av一坑| 日韩精品一区二区三区swag| 色综合中文字幕| 亚洲线精品一区二区三区八戒| 欧美亚洲综合一区| 成人av网站在线观看免费| 国产精品天干天干在观线| 一本一本久久a久久精品综合妖精| 天天av天天翘天天综合网| 欧美电影影音先锋| 欧美三区在线| 日韩av电影免费观看高清完整版 | 欧美视频你懂的| zzijzzij亚洲日本少妇熟睡| 国产精品久久久久久久久久久免费看 | 国产午夜亚洲精品羞羞网站| 亚洲日本视频| 久久精品久久99精品久久| 欧美xxxxxxxxx| 亚洲欧洲日本mm| 美日韩一级片在线观看| 欧美xxxxxxxx| aⅴ色国产欧美| 美女高潮久久久| 国产午夜精品福利| 午夜在线精品偷拍| 国产精品一卡二| 亚洲天堂精品在线观看| 91九色02白丝porn| 99国产精品99久久久久久| 一级女性全黄久久生活片免费| 欧美亚洲免费在线一区| 91在线丨porny丨国产| 夜夜嗨av一区二区三区网页| 4438x成人网最大色成网站| 一区二区视频在线观看| 日本伊人午夜精品| 久久精品在线免费观看| 午夜亚洲伦理| 成人国产精品免费网站| 一区二区三区免费网站| 日韩欧美的一区二区| 亚洲欧洲一级| 国产精一区二区三区| 亚洲欧洲国产日本综合| 欧美日韩综合色| 亚洲国产午夜| 懂色av一区二区夜夜嗨| 亚洲资源中文字幕| 日韩欧美中文字幕公布| 一区二区精品在线观看| 韩国成人精品a∨在线观看| 自拍偷自拍亚洲精品播放| 欧美美女黄视频| 国产欧美一级| 99re热视频精品| 日韩和欧美的一区| 国产精品天美传媒沈樵| 欧美一区二区三区播放老司机| 亚洲国产婷婷| 国产精品影音先锋| 一区二区久久久久久| 久久综合久久久久88| 欧美在线综合视频| 99riav国产精品| 91在线云播放| 久久国产精品色婷婷| 亚洲手机成人高清视频| 91精品国产91热久久久做人人| 国产精品久久久久久模特| 成人一区二区三区视频在线观看| 亚洲国产精品天堂| 国产农村妇女毛片精品久久麻豆| 欧美日韩国产美女| 一本色道久久综合亚洲精品不卡 | 一区免费观看视频| 精品国产第一区二区三区观看体验| 91成人免费在线| 韩国av一区| 成人精品高清在线| 理论电影国产精品| 亚洲一二三区在线观看| 国产亚洲精品7777| 91精品国产色综合久久久蜜香臀| 快she精品国产999| 日韩一级免费| 午夜视频一区| av亚洲精华国产精华精华| 精品亚洲成av人在线观看| 午夜激情一区二区| 亚洲精品成人精品456| 国产三级一区二区三区| 欧美一区二区三区系列电影| 视频一区中文字幕| 免费精品视频在线| 色婷婷精品久久二区二区蜜臂av| 日日摸夜夜添夜夜添亚洲女人| 中文字幕日韩精品一区 | 亚洲一级网站| 波多野结衣欧美| 国产一区二区三区四| 日本女优在线视频一区二区| 亚洲国产综合在线| 亚洲同性gay激情无套| 国产精品天美传媒沈樵| 久久人人97超碰com| 欧美日韩国产区一| 欧美日韩中文国产| 色婷婷久久久久swag精品| 久久久精品动漫| 国产精品久久一区二区三区| 亚洲成人在线视频网站| 欧美三区不卡| 女女同性女同一区二区三区91| 成人免费毛片嘿嘿连载视频| 国产剧情一区在线| 国产精品资源在线看| 国产一区二区三区免费| 狠狠色丁香婷婷综合| 精品一二三四区| 国产一区二区三区视频在线播放 | 欧美福利一区二区三区| 色综合色狠狠综合色| gogogo免费视频观看亚洲一| 成人激情图片网| yourporn久久国产精品| zzijzzij亚洲日本少妇熟睡| 懂色av一区二区三区免费观看| 粉嫩绯色av一区二区在线观看| 懂色av一区二区夜夜嗨| 丁香婷婷深情五月亚洲| 成人综合婷婷国产精品久久蜜臀| 国产一区福利在线| 成人性视频免费网站| 国产精品综合在线视频| 国产一区二区三区四区五区入口 | 欧美精品一区二区三区一线天视频 | 国产欧美一区二区在线观看| 日韩免费福利电影在线观看| 日韩欧美在线不卡| 日韩精品一区二区三区在线观看| 欧美一区二区三区思思人| 欧美一级黄色片| 精品人伦一区二区色婷婷| 国产女人18毛片水真多成人如厕| 亚洲少妇30p| 丝袜亚洲另类丝袜在线|