在vue元件中事件如何傳遞
最近的工作需要用到vue,所以最近接觸最多的就是vue,下面我給大家介紹下vue組件間事件傳遞,需要的朋友參考下吧
由於新工作需要用vue,所以最近接觸最多的也是vue,因為之前一直在用react,所以對於vue上手還是很快的。
我也盡量找一些他們兩個的異同點,除了多了一些輔助用的方法以外,最大的不同應該是對於組件間的通信,不僅有props,還有一種事件監聽,也是可以透過組件間傳遞的。
但是,在vue2. 中,vue引入了diff演算法和虛擬dom來提升效率。我們知道這些事為了處理頻繁更新dom元素所提出的一種優化方案,可頻繁變動更新以及事件監聽的初始化之間是否會有矛盾,當組件需要變動時,有沒有對註冊過的事件進行解綁?我們來寫一些簡單的程式碼印證一下。
我們寫兩個p做的按鈕,一個是寫的html程式碼,一個是透過元件的形式插入,兩個按鈕完全一樣,但我們加一個disabled的屬性在外層,並透過if- else來判斷disabled從而顯示不同的按鈕(當然正常場景下我們不會這麼去寫程式碼,這裡只是透過這種方式模擬一種特殊場景,我們自行考慮在我們的業務中是否存在這種場景)。
<template> <p class="test"> <p class="btn" v-if="disabled" @click="handleClick">可点击</p> <p class="btn" v-else >不可点击</p> <Button v-if="disabled" @clickTest="handleClick">可点击</Button> <Button v-else>不可点击</Button> </p> </template> <script> import Button from './Button' export default { data () { return { disabled: true } }, methods: { handleClick() { alert('可点击') } }, components: { Button, }, mounted() { setTimeout(() => { this.disabled = false }, 1000) } } </script> <style> .btn{ margin: 100px auto; width: 200px; line-height: 50px; border: 1px solid #42b983; border-radius: 5px; color: #42b983; } </style>
我們加一點樣式,讓他盡量好看一點,看著很簡單,兩個按鈕,可點擊時為他綁定一個點擊事件,不可點擊時不為他綁定。不同點是一個是直接寫的html程式碼,一個是元件。元件的程式碼如下:
<template> <p class="btn" @click="handleClick"><slot></slot></p> </template> <script> export default { methods: { handleClick() { this.$emit('clickTest') } } } </script>
然後在mounted週期加一個1秒的settimeout將disabled變成false,然後我們測試一下
當disabled還是true得時候,兩個按鈕點選都會跳出可點選的alert。但當disebled變成false的時候,上面用html寫的不會再彈框,可是下面用元件寫的就還是會彈跳窗。
這種問題出現時是非常不好定位的,因為程式碼上很顯然不會去調取這個clicktest事件,而在頁面上,我們也能確定按鈕已經變成不可點擊的那一個了。那為什麼這個事件還是會被調取呢? 這先要從diff演算法說起,傳統的diff tree演算法的演算法複雜度是O(n^3),而react在引入diff演算法時,拋除了跨級移動的情況,即只比對同一層的節點異同,讓演算法複雜度降低到了O(n),讓我們可以肆無忌憚(當然也要適可而止)的頻繁刷新整個頁面。 (呵呵,沒圖)diff有一個策略是擁有相同類別的兩個元件將會產生相似的樹狀結構,擁有不同類別的兩個元件將會產生不同的樹形結構。所以它的比對順序就是1)tree diff2)component diff3)element diff#回到我們的程式碼上,我們在進行component diff時,認為他們是相同的組件,然後進行element diff,即進行新增刪除和移動所以問題就是發生在了這裡,在實例化組件的時候我們初始化了事件監聽,但在替換相同元件裡的dom時,vue並沒有對已加入元件上的事件監聽做刪除。 我們看一下vue的程式碼,###Vue.prototype.$emit = function (event: string): Component { const vm: Component = this if (process.env.NODE_ENV !== 'production') { const lowerCaseEvent = event.toLowerCase() if (lowerCaseEvent !== event && vm._events[lowerCaseEvent]) { tip( `Event "${lowerCaseEvent}" is emitted in component ` + `${formatComponentName(vm)} but the handler is registered for "${event}". ` + `Note that HTML attributes are case-insensitive and you cannot use ` + `v-on to listen to camelCase events when using in-DOM templates. ` + `You should probably use "${hyphenate(event)}" instead of "${event}".` ) } } let cbs = vm._events[event] if (cbs) { cbs = cbs.length > 1 ? toArray(cbs) : cbs const args = toArray(arguments, 1) for (let i = 0, l = cbs.length; i < l; i++) { try { cbs[i].apply(vm, args) } catch (e) { handleError(e, vm, `event handler for "${event}"`) } } } return vm }
: clickTest : Array(1) 0 : ƒ invoker() length :
function sameVnode (a, b) { return ( a.key === b.key && ( ( a.tag === b.tag && a.isComment === b.isComment && isDef(a.data) === isDef(b.data) && sameInputType(a, b) ) || ( isTrue(a.isAsyncPlaceholder) && a.asyncFactory === b.asyncFactory && isUndef(b.asyncFactory.error) ) ) ) }
<Button key="1" v-if="disabled" @clickTest="handleClick">可点击</Button> <Button key="2" v-else>不可点击</Button>
<Button v-if="disabled" :clickTest="handleClick">可点击</Button> <Button v-else>不可点击</Button> props: { 'clickTest': { type: Function } }, methods: { handleClick() { //this.$emit('clickTest') this.clickTest && this.clickTest() } }
虽然vue给了我们更方便的事件传递的方式,但props里是允许我们去传递任何类型的,我的期望是在真实的dom上或者在公共组件的入口处以外的地方,都是通过props的方式来传递结果的。虽然这种方式很不vue,而且也享受不到v-on给我们带来的遍历,但是这样确实可以减少不必要的麻烦。
当然既然用了vue,更好的利用vue给我们带来的便利也很重要,所以对于这种很少会出现的麻烦,我们有一个预期,并可以快速定位并修复问题,就可以了。
上面是我整理给大家的,希望今后会对大家有帮助。
相关文章:
以上是在vue元件中事件如何傳遞的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

在 Vue.js 中使用 Bootstrap 分為五個步驟:安裝 Bootstrap。在 main.js 中導入 Bootstrap。直接在模板中使用 Bootstrap 組件。可選:自定義樣式。可選:使用插件。

可以通過以下步驟為 Vue 按鈕添加函數:將 HTML 模板中的按鈕綁定到一個方法。在 Vue 實例中定義該方法並編寫函數邏輯。

Vue.js 中的 watch 選項允許開發者監聽特定數據的變化。當數據發生變化時,watch 會觸發一個回調函數,用於執行更新視圖或其他任務。其配置選項包括 immediate,用於指定是否立即執行回調,以及 deep,用於指定是否遞歸監聽對像或數組的更改。

Vue.js 返回上一頁有四種方法:$router.go(-1)$router.back()使用 <router-link to="/"> 組件window.history.back(),方法選擇取決於場景。

Vue 多頁面開發是一種使用 Vue.js 框架構建應用程序的方法,其中應用程序被劃分為獨立的頁面:代碼維護性:將應用程序拆分為多個頁面可以使代碼更易於管理和維護。模塊化:每個頁面都可以作為獨立的模塊,便於重用和替換。路由簡單:頁面之間的導航可以通過簡單的路由配置來管理。 SEO 優化:每個頁面都有自己的 URL,這有助於搜索引擎優化。

NetflixusesAcustomFrameworkcalled“ Gibbon” BuiltonReact,notReactorVuedIrectly.1)TeamSperience:selectBasedonFamiliarity.2)ProjectComplexity:vueforsimplerprojects:reactforforforproproject,reactforforforcompleplexones.3)cocatizationneedneeds:reactoffipicatizationneedneedneedneedneedneeds:reactoffersizationneedneedneedneedneeds:reactoffersizatization needefersmoreflexibleise.4)

在 Vue.js 中引用 JS 文件的方法有三種:直接使用 <script> 標籤指定路徑;利用 mounted() 生命週期鉤子動態導入;通過 Vuex 狀態管理庫進行導入。

Vue.js 遍歷數組和對像有三種常見方法:v-for 指令用於遍歷每個元素並渲染模板;v-bind 指令可與 v-for 一起使用,為每個元素動態設置屬性值;.map 方法可將數組元素轉換為新數組。
