自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

10個(gè)Vue開發(fā)技巧

開發(fā) 前端
在組件中使用 $route 會(huì)使之與其對(duì)應(yīng)路由形成高度耦合,從而使組件只能在某些特定的 URL 上使用,限制了其靈活性。

路由參數(shù)解耦

一般在組件內(nèi)使用路由參數(shù),大多數(shù)人會(huì)這樣做:

  1. export default { 
  2.     methods: { 
  3.         getParamsId() { 
  4.             return this.$route.params.id 
  5.         } 
  6.     } 

在組件中使用 $route 會(huì)使之與其對(duì)應(yīng)路由形成高度耦合,從而使組件只能在某些特定的 URL 上使用,限制了其靈活性。

正確的做法是通過(guò) props 解耦

  1. const router = new VueRouter({ 
  2.     routes: [{ 
  3.         path: '/user/:id', 
  4.         component: User, 
  5.         props: true 
  6.     }] 
  7. }) 

將路由的 props 屬性設(shè)置為 true 后,組件內(nèi)可通過(guò) props 接收到 params 參數(shù)

  1. export default { 
  2.     props: ['id'], 
  3.     methods: { 
  4.         getParamsId() { 
  5.             return this.id 
  6.         } 
  7.     } 

另外你還可以通過(guò)函數(shù)模式來(lái)返回 props

  1. const router = new VueRouter({ 
  2.     routes: [{ 
  3.         path: '/user/:id', 
  4.         component: User, 
  5.         props: (route) => ({ 
  6.             id: route.query.id 
  7.         }) 
  8.     }] 
  9. }) 

文檔:router.vuejs.org/zh/guide/es…

函數(shù)式組件

函數(shù)式組件是無(wú)狀態(tài),它無(wú)法實(shí)例化,沒(méi)有任何的生命周期和方法。創(chuàng)建函數(shù)式組件也很簡(jiǎn)單,只需要在模板添加 functional 聲明即可。一般適合只依賴于外部數(shù)據(jù)的變化而變化的組件,因其輕量,渲染性能也會(huì)有所提高。

組件需要的一切都是通過(guò) context 參數(shù)傳遞。它是一個(gè)上下文對(duì)象,具體屬性查看文檔。這里 props 是一個(gè)包含所有綁定屬性的對(duì)象。

函數(shù)式組件:

  1. <template functional> 
  2.     <div class="list"> 
  3.         <div class="item" v-for="item in props.list" :key="item.id" @click="props.itemClick(item)"> 
  4.             <p>{{item.title}}</p> 
  5.             <p>{{item.content}}</p> 
  6.         </div> 
  7.     </div> 
  8. </template> 

父組件使用:

  1. <template> 
  2.     <div> 
  3.         <List :list="list" :itemClick="item => (currentItem = item)" /> 
  4.     </div> 
  5. </template> 
  1. import List from '@/components/List.vue' 
  2. export default { 
  3.     components: { 
  4.         List 
  5.     }, 
  6.     data() { 
  7.         return { 
  8.             list: [{ 
  9.                 title: 'title', 
  10.                 content: 'content' 
  11.             }], 
  12.             currentItem: '' 
  13.         } 
  14.     } 

文檔:cn.vuejs.org/v2/guide/re…

樣式穿透

在開發(fā)中修改第三方組件樣式是很常見,但由于 scoped 屬性的樣式隔離,可能需要去除 scoped 或是另起一個(gè) style 。這些做法都會(huì)帶來(lái)副作用(組件樣式污染、不夠優(yōu)雅),樣式穿透在css預(yù)處理器中使用才生效。

我們可以使用 >>> 或 /deep/ 解決這一問(wèn)題:

  1. <style scoped> 
  2. 外層 >>> .el-checkbox { 
  3.   display: block; 
  4.   font-size: 26px; 
  5.  
  6.   .el-checkbox__label { 
  7.     font-size: 16px; 
  8.   } 
  9. </style> 
  1. <style scoped> 
  2. /deep/ .el-checkbox { 
  3.   display: block; 
  4.   font-size: 26px; 
  5.  
  6.   .el-checkbox__label { 
  7.     font-size: 16px; 
  8.   } 
  9. </style> 

watch高階使用

1. 立即執(zhí)行

watch 是在監(jiān)聽屬性改變時(shí)才會(huì)觸發(fā),有些時(shí)候,我們希望在組件創(chuàng)建后 watch 能夠立即執(zhí)行。

可能想到的的方法就是在 create 生命周期中調(diào)用一次,但這樣的寫法不優(yōu)雅,或許我們可以使用這樣的方法:

  1. export default { 
  2.     data() { 
  3.         return { 
  4.             name: 'Joe' 
  5.         } 
  6.     }, 
  7.     watch: { 
  8.         name: { 
  9.             handler: 'sayName', 
  10.             immediate: true 
  11.         } 
  12.     }, 
  13.     methods: { 
  14.         sayName() { 
  15.             console.log(this.name) 
  16.         } 
  17.     } 

深度監(jiān)聽在監(jiān)聽對(duì)象時(shí),對(duì)象內(nèi)部的屬性被改變時(shí)無(wú)法觸發(fā) watch ,我們可以為其設(shè)置深度監(jiān)聽:

  1. export default { 
  2.     data: { 
  3.         studen: { 
  4.             name: 'Joe', 
  5.             skill: { 
  6.                 run: { 
  7.                     speed: 'fast' 
  8.                 } 
  9.             } 
  10.         } 
  11.     }, 
  12.     watch: { 
  13.         studen: { 
  14.             handler: 'sayName', 
  15.             deep: true 
  16.         } 
  17.     }, 
  18.     methods: { 
  19.         sayName() { 
  20.             console.log(this.studen) 
  21.         } 
  22.     } 

2. 觸發(fā)監(jiān)聽

執(zhí)行多個(gè)方法使用數(shù)組可以設(shè)置多項(xiàng),形式包括字符串、函數(shù)、對(duì)象:

  1. export default { 
  2.     data: { 
  3.         name: 'Joe' 
  4.     }, 
  5.     watch: { 
  6.         name: [ 
  7.             'sayName1', 
  8.             function(newVal, oldVal) { 
  9.                 this.sayName2() 
  10.             }, 
  11.             { 
  12.                 handler: 'sayName3', 
  13.                 immaediate: true 
  14.             } 
  15.         ] 
  16.     }, 
  17.     methods: { 
  18.         sayName1() { 
  19.             console.log('sayName1==>', this.name) 
  20.         }, 
  21.         sayName2() { 
  22.             console.log('sayName2==>', this.name) 
  23.         }, 
  24.         sayName3() { 
  25.             console.log('sayName3==>', this.name) 
  26.         } 
  27.     } 

文檔:cn.vuejs.org/v2/api/#wat…

watch監(jiān)聽多個(gè)變量

watch本身無(wú)法監(jiān)聽多個(gè)變量。但我們可以將需要監(jiān)聽的多個(gè)變量通過(guò)計(jì)算屬性返回對(duì)象,再監(jiān)聽這個(gè)對(duì)象來(lái)實(shí)現(xiàn)“監(jiān)聽多個(gè)變量”

  1. export default { 
  2.     data() { 
  3.         return { 
  4.             msg1: 'apple', 
  5.             msg2: 'banana' 
  6.         } 
  7.     }, 
  8.     compouted: { 
  9.         msgObj() { 
  10.             const { msg1, msg2 } = this 
  11.             return { 
  12.                 msg1, 
  13.                 msg2 
  14.             } 
  15.         } 
  16.     }, 
  17.     watch: { 
  18.         msgObj: { 
  19.             handler(newVal, oldVal) { 
  20.                 if (newVal.msg1 != oldVal.msg1) { 
  21.                     console.log('msg1 is change') 
  22.                 } 
  23.                 if (newVal.msg2 != oldVal.msg2) { 
  24.                     console.log('msg2 is change') 
  25.                 } 
  26.             }, 
  27.             deep: true 
  28.         } 
  29.     } 

事件參數(shù)$event

$event 是事件對(duì)象的特殊變量,在一些場(chǎng)景能給我們實(shí)現(xiàn)復(fù)雜功能提供更多可用的參數(shù)

1. 原生事件

在原生事件中表現(xiàn)和默認(rèn)的事件對(duì)象相同:

  1. <template> 
  2.     <div> 
  3.         <input type="text" @input="inputHandler('hello', $event)" /> 
  4.     </div> 
  5. </template> 
  1. export default { 
  2.     methods: { 
  3.         inputHandler(msg, e) { 
  4.             console.log(e.target.value) 
  5.         } 
  6.     } 

2. 自定義事件

在自定義事件中表現(xiàn)為捕獲從子組件拋出的值

my-item.vue:

  1. export default { 
  2.     methods: { 
  3.         customEvent() { 
  4.             this.$emit('custom-event', 'some value') 
  5.         } 
  6.     } 
  7. 復(fù)制代碼 

App.vue:

  1. <template> 
  2.     <div> 
  3.         <my-item v-for="(item, index) in list" @custom-event="customEvent(index, $event)"> 
  4.             </my-list> 
  5.     </div> 
  6. </template> 
  1. export default { 
  2.     methods: { 
  3.         customEvent(index, e) { 
  4.             console.log(e) // 'some value' 
  5.         } 
  6.     } 

文檔:

  • cn.vuejs.org/v2/guide/ev…
  • cn.vuejs.org/v2/guide/co…

自定義組件雙向綁定

組件 model 選項(xiàng):

允許一個(gè)自定義組件在使用 v-model 時(shí)定制 prop 和 event。默認(rèn)情況下,一個(gè)組件上的 v-model 會(huì)把 value 用作 prop 且把 input 用作 event,但是一些輸入類型比如單選框和復(fù)選框按鈕可能想使用 value prop 來(lái)達(dá)到不同的目的。使用 model 選項(xiàng)可以回避這些情況產(chǎn)生的沖突。

input 默認(rèn)作為雙向綁定的更新事件,通過(guò) $emit 可以更新綁定的值

  1. <my-switch v-model="val"></my-switch> 
  1. export default { 
  2.     props: { 
  3.         value: { 
  4.             type: Boolean, 
  5.             default: false 
  6.         } 
  7.     }, 
  8.     methods: { 
  9.         switchChange(val) { 
  10.             this.$emit('input', val) 
  11.         } 
  12.     } 

修改組件的 model 選項(xiàng),自定義綁定的變量和事件

  1. <my-switch v-model="num" value="some value"></my-switch> 
  1. export default { 
  2.     model: { 
  3.         prop: 'num', 
  4.         event: 'update' 
  5.     }, 
  6.     props: { 
  7.         value: { 
  8.             type: String, 
  9.             default: '' 
  10.         }, 
  11.         num: { 
  12.             type: Number, 
  13.             default: 0 
  14.         } 
  15.     }, 
  16.     methods: { 
  17.         numChange() { 
  18.             this.$emit('update', this.num++) 
  19.         } 
  20.     } 

文檔:cn.vuejs.org/v2/api/#mod…

監(jiān)聽組件生命周期

通常我們監(jiān)聽組件生命周期會(huì)使用 $emit ,父組件接收事件來(lái)進(jìn)行通知

子組件:

  1. export default { 
  2.     mounted() { 
  3.         this.$emit('listenMounted') 
  4.     } 

父組件:

  1. <template> 
  2.     <div> 
  3.         <List @listenMounted="listenMounted" /> 
  4.     </div> 
  5. </template> 

其實(shí)還有一種簡(jiǎn)潔的方法,使用 @hook 即可監(jiān)聽組件生命周期,組件內(nèi)無(wú)需做任何改變。同樣的, created 、 updated 等也可以使用此方法。

  1. <template> 
  2.     <List @hook:mounted="listenMounted" /> 
  3. </template> 

程序化的事件偵聽器

比如,在頁(yè)面掛載時(shí)定義計(jì)時(shí)器,需要在頁(yè)面銷毀時(shí)清除定時(shí)器。這看起來(lái)沒(méi)什么問(wèn)題。但仔細(xì)一看 this.timer 唯一的作用只是為了能夠在 beforeDestroy 內(nèi)取到計(jì)時(shí)器序號(hào),除此之外沒(méi)有任何用處。

  1. export default { 
  2.     mounted() { 
  3.         this.timer = setInterval(() => { 
  4.             console.log(Date.now()) 
  5.         }, 1000) 
  6.     }, 
  7.     beforeDestroy() { 
  8.         clearInterval(this.timer) 
  9.     } 

如果可以的話最好只有生命周期鉤子可以訪問(wèn)到它。這并不算嚴(yán)重的問(wèn)題,但是它可以被視為雜物。

我們可以通過(guò) $on 或 $once 監(jiān)聽頁(yè)面生命周期銷毀來(lái)解決這個(gè)問(wèn)題:

  1. export default { 
  2.     mounted() { 
  3.         this.creatInterval('hello') 
  4.         this.creatInterval('world') 
  5.     }, 
  6.     creatInterval(msg) { 
  7.         let timer = setInterval(() => { 
  8.             console.log(msg) 
  9.         }, 1000) 
  10.         this.$once('hook:beforeDestroy', function() { 
  11.             clearInterval(timer) 
  12.         }) 
  13.     } 

使用這個(gè)方法后,即使我們同時(shí)創(chuàng)建多個(gè)計(jì)時(shí)器,也不影響效果。因?yàn)樗鼈儠?huì)在頁(yè)面銷毀后程序化的自主清除。

文檔:cn.vuejs.org/v2/guide/co…

手動(dòng)掛載組件

在一些需求中,手動(dòng)掛載組件能夠讓我們實(shí)現(xiàn)起來(lái)更加優(yōu)雅。比如一個(gè)彈窗組件,最理想的用法是通過(guò)命令式調(diào)用,就像 elementUI 的 this.$message 。而不是在模板中通過(guò)狀態(tài)切換,這種實(shí)現(xiàn)真的很糟糕。

先來(lái)個(gè)最簡(jiǎn)單的例子:

  1. import Vue from 'vue' 
  2. import Message from './Message.vue' 
  3.  
  4. // 構(gòu)造子類 
  5. let MessageConstructor = Vue.extend(Message) 
  6. // 實(shí)例化組件 
  7. let messageInstance = new MessageConstructor() 
  8. // $mount可以傳入選擇器字符串,表示掛載到該選擇器 
  9. // 如果不傳入選擇器,將渲染為文檔之外的的元素,你可以想象成 document.createElement()在內(nèi)存中生成dom 
  10. messageInstance.$mount() 
  11. // messageInstance.$el獲取的是dom元素 
  12. document.body.appendChild(messageInstance.$el) 

下面實(shí)現(xiàn)一個(gè)簡(jiǎn)易的 message 彈窗組件

Message/index.vue:

  1. <template> 
  2.     <div class="wrap"> 
  3.         <div class="message" :class="item.type" v-for="item in notices" :key="item._name"> 
  4.             <div class="content">{{item.content}}</div> 
  5.         </div> 
  6.     </div> 
  7. </template> 
  1. // 默認(rèn)選項(xiàng) 
  2. const DefaultOptions = { 
  3.     duration: 1500, 
  4.     type: 'info', 
  5.     content: '這是一條提示信息!', 
  6. let mid = 0 
  7. export default { 
  8.     data() { 
  9.         return { 
  10.             notices: [] 
  11.         } 
  12.     }, 
  13.     methods: { 
  14.         add(notice = {}) { 
  15.             // name標(biāo)識(shí) 用于移除彈窗 
  16.             let _name = this.getName() 
  17.             // 合并選項(xiàng) 
  18.             notice = Object.assign({ 
  19.                 _name 
  20.             }, DefaultOptions, notice) 
  21.  
  22.             this.notices.push(notice) 
  23.  
  24.             setTimeout(() => { 
  25.                 this.removeNotice(_name) 
  26.             }, notice.duration) 
  27.         }, 
  28.         getName() { 
  29.             return 'msg_' + (mid++) 
  30.         }, 
  31.         removeNotice(_name) { 
  32.             let index = this.notices.findIndex(item => item._name === _name) 
  33.             this.notices.splice(index, 1) 
  34.         } 
  35.     } 
  1. .wrap { 
  2.     position: fixed; 
  3.     top: 50px; 
  4.     left: 50%; 
  5.     display: flex; 
  6.     flex-direction: column; 
  7.     align-items: center; 
  8.     transform: translateX(-50%); 
  9.  
  10. .message { 
  11.     --borderWidth: 3px; 
  12.     min-width: 240px; 
  13.     max-width: 500px; 
  14.     margin-bottom: 10px; 
  15.     border-radius: 3px; 
  16.     box-shadow: 0 0 8px #ddd; 
  17.     overflow: hidden; 
  18.  
  19. .content { 
  20.     padding: 8px; 
  21.     line-height: 1.3; 
  22.  
  23. .message.info { 
  24.     border-left: var(--borderWidth) solid #909399; 
  25.     background: #F4F4F5; 
  26.  
  27. .message.success { 
  28.     border-left: var(--borderWidth) solid #67C23A; 
  29.     background: #F0F9EB; 
  30.  
  31. .message.error { 
  32.     border-left: var(--borderWidth) solid #F56C6C; 
  33.     background: #FEF0F0; 
  34.  
  35. .message.warning { 
  36.     border-left: var(--borderWidth) solid #E6A23C; 
  37.     background: #FDF6EC; 

Message/index.js:

  1. import Vue from 'vue' 
  2. import Index from './index.vue' 
  3.  
  4. let messageInstance = null 
  5. let MessageConstructor = Vue.extend(Index) 
  6.  
  7. let init = () => { 
  8.     messageInstance = new MessageConstructor() 
  9.     messageInstance.$mount() 
  10.     document.body.appendChild(messageInstance.$el) 
  11.  
  12. let caller = (options) => { 
  13.     if (!messageInstance) { 
  14.         init(options) 
  15.     } 
  16.     messageInstance.add(options) 
  17.  
  18. export default { 
  19.     // 返回 install 函數(shù) 用于 Vue.use 注冊(cè) 
  20.     install(vue) { 
  21.         vue.prototype.$message = caller 
  22.     } 

main.js:

  1. import Message from '@/components/Message/index.js' 
  2.  
  3. Vue.use(Message) 

使用:

  1. this.$message({ 
  2.     type: 'success', 
  3.     content: '成功信息提示', 
  4.     duration: 3000 
  5. }) 

文檔:cn.vuejs.org/v2/api/#vm-…

 

責(zé)任編輯:趙寧寧 來(lái)源: 前端工匠
相關(guān)推薦

2022-12-15 16:38:17

2020-03-31 09:47:04

Vue開發(fā)代碼

2015-07-27 09:36:09

storyboard

2020-06-07 16:16:01

Python開發(fā)工具

2020-06-10 10:30:48

Python 開發(fā)編程語(yǔ)言

2017-11-06 14:33:54

Web開發(fā)服務(wù)器網(wǎng)絡(luò)

2019-07-23 09:00:00

vuejavascript前端

2025-02-03 00:00:00

Vue.js組件緩存

2025-01-06 08:57:19

Vue技巧

2020-06-15 10:29:10

JavaScript開發(fā) 技巧

2020-04-14 09:26:43

開發(fā)工具Chrome

2014-08-20 10:02:54

GitGit能力

2019-02-25 15:15:44

Windows 10Windows技巧

2018-11-28 08:15:09

2011-07-07 10:21:56

2022-06-23 09:22:57

Vue技巧前端

2025-04-09 09:10:00

開發(fā)ViteVue

2020-06-08 07:52:31

Python開發(fā)工具

2015-12-21 10:54:37

Docker云計(jì)算

2021-03-15 08:13:19

JavaScript開發(fā)代碼
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)