首頁  >  文章  >  web前端  >  Vue.js提升必知的幾點總結

Vue.js提升必知的幾點總結

php中世界最好的语言
php中世界最好的语言原創
2018-04-11 13:37:561536瀏覽

這次帶給大家Vue.js提升必知的幾點總結,Vue.js提升的注意事項有哪些,下面就是實戰案例,一起來看一下。

第一個招:化繁為簡的Watchers

# 場景還原:

created(){
  this.fetchPostList()
},
watch: {
  searchInputValue(){
    this.fetchPostList()
  }
}

元件創建的時候我們獲取一次列表,同時監聽input框,每當發生變化的時候重新獲取一次篩選後的列表這個場景很常見,有沒有辦法優化一下呢?

招式解析:

首先,在watchers中,可以直接使用函數的字面量名稱;其次,宣告immediate:true表示建立元件時立刻執行一次。

watch: {
  searchInputValue:{
    handler: 'fetchPostList',
    immediate: true
  }
}

第二招:一勞永逸的元件註冊

# 場景還原:

import BaseButton from './baseButton'
import BaseIcon from './baseIcon'
import BaseInput from './baseInput'
export default {
 components: {
  BaseButton,
  BaseIcon,
  BaseInput
 }
}
<BaseInput
 v-model="searchText"
 @keydown.enter="search"
/>
<BaseButton @click="search">
 <BaseIcon name="search"/>
</BaseButton>

# 我們寫了一堆基礎UI元件,然後每次需要使用這些元件的時候,都得先import,然後就聲明components,很繁瑣!秉持能偷懶就偷懶的原則,我們要辦法優化!

招式解析:

我們需要藉助一下神器webpack,使用 require.context() 方法來建立自己的(模組)上下文,從而實現自動動態require元件。這個方法需要3個參數:要搜尋的資料夾目錄,是否還應該搜尋它的子目錄,以及一個符合檔案的正規表示式

我們在components資料夾中加入一個叫做global.js的文件,在這個文件裡借助webpack動態將需要的基礎元件統統打包進來。

import Vue from 'vue'
function capitalizeFirstLetter(string) {
 return string.charAt(0).toUpperCase() + string.slice(1)
}
const requireComponent = require.context(
 '.', false, /\.vue$/
  //找到components文件夹下以.vue命名的文件
)
requireComponent.keys().forEach(fileName => {
 const componentConfig = requireComponent(fileName)
 const componentName = capitalizeFirstLetter(
  fileName.replace(/^\.\//, '').replace(/\.\w+$/, '')
  //因为得到的filename格式是: './baseButton.vue', 所以这里我们去掉头和尾,只保留真正的文件名
 )
 Vue.component(componentName, componentConfig.default || componentConfig)
})

最後我們在main.js中import 'components/global.js',然後我們就可以隨時隨地使用這些基礎元件,無需手動引入了。

第三招:釜底抽薪的router key

# 場景還原:

下面這個場景真的是傷透了很多程式設計師的心...先默認大家用的是Vue-router來實現路由的控制。

假設我們在寫一個部落格網站,需求是從/post-page/a,跳到/post-page/b。然後我們驚人的發現,頁面跳轉後資料竟然沒更新? !原因是vue-router"智能地"發現這是同一個元件,然後它就決定要重複使用這個元件,所以你在created函數裡寫的方法壓根就沒執行。通常的解決方案是監聽$route的變化來初始化數據,如下:

data() {
 return {
  loading: false,
  error: null,
  post: null
 }
}, 
watch: {
 '$route': {
  handler: 'resetData',
  immediate: true
 }
},
methods: {
 resetData() {
  this.loading = false
  this.error = null
  this.post = null
  this.getPost(this.$route.params.id)
 },
 getPost(id){
 }
}

bug是解決了,但每次這麼寫也太不優雅了吧?秉持著能偷懶則偷懶的原則,我們希望程式碼這樣寫:

data() {
 return {
  loading: false,
  error: null,
  post: null
 }
},
created () {
 this.getPost(this.$route.params.id)
},
methods () {
 getPost(postId) {
  // ...
 }
}

招式解析:

那要怎麼樣才能實現這樣的效果呢,答案是為router-view添加一個unique的key,這樣即使是公用元件,只要url變化了,就一定會重新建立這個元件。 (雖然損失了一丟丟效能,但避免了無限的bug)。同時,注意我將key直接設定為路由的完整路徑,一舉兩得。

<router-view :key="$route.fullpath"></router-view>

第四招: 無所不能的render函數

# 場景還原:

vue要求每一個元件只能有一個根元素,當你有多個根元素時,vue就會給你報錯

<template>
 <li
  v-for="route in routes"
  :key="route.name"
 >
  <router-link :to="route">
   {{ route.title }}
  </router-link>
 </li>
</template>
 ERROR - Component template should contain exactly one root element. 
  If you are using v-if on multiple elements, use v-else-if 
  to chain them instead.

招式解析:

那有沒有辦法化解呢,答案是有的,只不過這時候我們需要使用render()函數來創建HTML,而不是template。其實用js來產生html的好處就是極度的靈活功能強大,而且你不需要去學習使用vue的那些功能有限的指令API,比如v-for, v-if。 (reactjs就完全丟棄了template)

functional: true,
render(h, { props }) {
 return props.routes.map(route =>
  <li key={route.name}>
   <router-link to={route}>
    {route.title}
   </router-link>
  </li>
 )
}

第五招:無招勝有招的高階組件

# 劃重點:這招威力無窮,請務必掌握

當我們寫元件的時候,通常我們都需要從父元件傳遞一系列的props到子元件,同時父元件監聽子元件emit過來的一系列事件。舉例:

//父组件
<BaseInput 
  :value="value"
  label="密码" 
  placeholder="请填写密码"
  @input="handleInput"
  @focus="handleFocus>
</BaseInput>
//子组件
<template>
 <label>
  {{ label }}
  <input
   :value="value"
   :placeholder="placeholder"
   @focus=$emit(&#39;focus&#39;, $event)"
   @input="$emit(&#39;input&#39;, $event.target.value)"
  >
 </label>
</template>

有下面幾個優化點:

1.每一个从父组件传到子组件的props,我们都得在子组件的Props中显式的声明才能使用。这样一来,我们的子组件每次都需要申明一大堆props, 而类似placeholer这种dom原生的property我们其实完全可以直接从父传到子,无需声明。方法如下:

<input
   :value="value"
   v-bind="$attrs"
   @input="$emit(&#39;input&#39;, $event.target.value)"
  >

$attrs包含了父作用域中不作为 prop 被识别 (且获取) 的特性绑定 (class 和 style 除外)。当一个组件没有声明任何 prop 时,这里会包含所有父作用域的绑定,并且可以通过 v-bind="$attrs" 传入内部组件——在创建更高层次的组件时非常有用。

2.注意到子组件的@focus=$emit('focus', $event)"其实什么都没做,只是把event传回给父组件而已,那其实和上面类似,我完全没必要显式地申明:

<input
  :value="value"
  v-bind="$attrs"
  v-on="listeners"
>
computed: {
 listeners() {
  return {
   ...this.$listeners,
   input: event => 
    this.$emit('input', event.target.value)
  }
 }
}

$listeners包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on="$listeners" 传入内部组件——在创建更高层次的组件时非常有用。

3.需要注意的是,由于我们input并不是BaseInput这个组件的根节点,而默认情况下父作用域的不被认作 props 的特性绑定将会“回退”且作为普通的 HTML 特性应用在子组件的根元素上。所以我们需要设置inheritAttrs:false,这些默认行为将会被去掉, 以上两点的优化才能成功。

相信看了本文案例你已经掌握了方法,更多精彩请关注php中文网其它相关文章!

推荐阅读:

vue的新手入门教程

Vue项目怎样分环境打包

以上是Vue.js提升必知的幾點總結的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn