App下載

10個簡單的技巧讓你的 vue.js 代碼更優(yōu)雅

猿友 2020-09-04 14:02:25 瀏覽數(shù) (3062)
反饋

文章轉(zhuǎn)載自公眾號:前端人

前言

作為深度代碼潔癖,我們都希望能寫出簡單高效的代碼,讓我們的代碼看起來更加優(yōu)雅,讓我們拋棄繁雜的代碼,一起開啟簡單的旅程~~

slots 新語法向 3.0 看齊

使用帶有“#”的新命名插槽縮寫語法,在Vue 2.6.0+中可用

舉個例子:

構(gòu)建插槽時,最好規(guī)劃一下布局。這就是我的文章布局。構(gòu)建插槽與構(gòu)建組件沒有什么不同。本質(zhì)上,插槽是具有超強功能的組件,讓我們細(xì)分一下上面的布局,組件的外觀如下:

<!-- TidbitPage.vue -->
<template>
  <article-layout>


    <template #articleHeader>
      <h1>I am the header</h1>
    </template>


    <template #articleContent>
      <p>I am the content</p>
    </template>


    <template #articleFooter>
      <footer>I am the footer</footer>
    </template>


    <template #side>
      <aside>I am the side stuff</aside>
    </template>


    <template #banner>
      <div>I am the banner</div>
    </template>


  </article-layout>
<template>

動態(tài)指令參數(shù)

指令參數(shù)現(xiàn)在可以接受動態(tài) JavaScript 表達(dá)式 動態(tài)參數(shù)值應(yīng)該是字符串,但允許null作為一個明確指示應(yīng)該刪除綁定的特殊值,那將會很方便。任何其他非字符串值都可能出錯,并會觸發(fā)警告。(僅適用于v-bind和v-on

<div v-bind:[attr]="attributeName"></div>
//簡寫
<div :[attr]="attributeName"></div>

這里的 attributeName 會被作為一個JavaScript表達(dá)式進(jìn)行動態(tài)求值,求得的值將會作為最終的參數(shù)來使用。例如,如果你的 Vue 實例有一個 data 屬性 attributeName,其值為 href,那么這個綁定將等價于 v-bind:href

同樣地,你可以使用動態(tài)參數(shù)為一個動態(tài)的事件名綁定處理函數(shù):

<button v-on:[eventName]="handler"></button>
//簡寫
<button @[eventName]="handler"></button>

當(dāng) eventName 的值為 focus 時,v-on:[eventName] 將等價于v-on:focus。

同樣可以適用于插槽綁定:

<my-component>
<template v-slot:[slotName]>
Dynamic slot name
</template>
</my-component>
//簡寫
<foo>
<template #[name]>
Default slot
</template>
</foo>

動態(tài)參數(shù)預(yù)期會求出一個字符串,異常情況下值為 null。這個特殊的 null 值可以被顯性地用于移除綁定。任何其它非字符串類型的值都將會觸發(fā)一個警告。

<!-- 這會觸發(fā)一個編譯警告 且 無效 -->
<a v-bind:['foo' + bar]="value"> ... </a>

變通的辦法是使用沒有空格或引號的表達(dá)式,或用計算屬性替代這種復(fù)雜表達(dá)式。

@hook那些事

處理組件內(nèi)定時器的步驟。通常我們一般都是這樣操作的:

<script>
  export default {
    mounted() {
      this.timer = setInterval(() => { ... }, 1000);
    },
    beforeDestroy() {
      clearInterval(this.timer);
    }
  };
</script>

但是其實更好的做法是:

<script>
  export default {
    mounted() {
      const timer = setInterval(() => { ... }, 1000);
      this.$once('hook:beforeDestroy', () => clearInterval(timer);)
    }
  };
</script>

設(shè)想一個場景 如果我們需要在數(shù)據(jù)渲染到頁面的之前讓頁面 loading。mounted 之后停止 loading。beforeUpdata 時開始 loadingupdatad 之后停止 loading。

最簡單的方法就是改寫組件的生命周期函數(shù),使其在 mounted/beforeUpdata /updatad 時通知父組件顯示或者隱藏 loading。

這樣做顯示不好,因為侵入了自組件的邏輯,增加的邏輯也和組件本身的功能好不關(guān)聯(lián)。最好的辦法就是使用 v-on="hook:xxx" 的方式:

<v-chart
    @hook:mounted="loading = false"
    @hook:beforeUpdated="loading = true"
    @hook:updated="loading = false"
    :data="data"
/>

這樣,就實現(xiàn)了對子組件生命周期的監(jiān)聽。對任意的組件都有效果,包括引入的第三方組件。

vue中的$props、$attrs$listeners(可用來二次封裝組件)

$props:當(dāng)前組件接收到的 props 對象。Vue 實例代理了對其 props 對象屬性的訪問。

假如有個 input 輸入框。我們有很多的原生屬性,比如:name、placeholder、disabled 等等。我們?nèi)绻级x在props顯示接收,未免太過繁瑣。所以官網(wǎng)出現(xiàn)了:v-bind="$props"這樣的操作。如果父組件傳遞很多的原生屬性,那么我們在子組件中直接可以:

//good
<input v-bind="$props">

 
//bad
//而不是下面這樣,如果很多的屬性就特別繁瑣
<input :name="name" :placeholder="placeholder" :disabled="disabled">

我們可以利用以下方式$attrs 將原生屬性直接傳遞給子組件,這是 Vue 在2.4.0新增的屬性,包含了父作用域中不作為 prop 被識別 (且獲取) 的特性綁定 (class 和 style 除外)。當(dāng)一個組件沒有聲明任何 prop 時,這里會包含所有父作用域的綁定 (class 和 style 除外),并且可以通過 v-bind="$attrs" 傳入內(nèi)部組件——在創(chuàng)建高級別的組件時非常有用。

<input
   v-bind="$attrs"
   :value="value"
   @focus=$emit('focus', $event)"
   @input="$emit('input', $event.target.value)"
>

$listeners:包含了父作用域中的 (不含 .native修飾器的) v-on 事件監(jiān)聽器。它可以通過 v-on="$listeners" 傳入內(nèi)部組件——在創(chuàng)建更高層次的組件時非常有用。

如果子組件不在父組件的根目錄下,則可以將所有事件偵聽器從父組件傳遞到子組件,如下所示:

<template>
<div>
  ...
<childComponent v-on="$listeners" />...
</div>
</template>

響應(yīng)式數(shù)據(jù)(2.6.0新增)

我們習(xí)慣于用 Vuex 去解決狀態(tài)的共享問題,但是在小項目中使用就會有增大代碼體積和將代碼復(fù)雜化的煩惱,所以在后來的版本中Vue新增

Vue.observable( object ) 讓一個對象可響應(yīng),Vue 內(nèi)部會用它來處理 data 函數(shù)返回的對象。

返回的對象可以直接用于渲染函數(shù)和 計算屬性 內(nèi),并且會在發(fā)生改變時觸發(fā)相應(yīng)的更新。也可以作為最小化的跨組件狀態(tài)存儲器,用于簡單的場景:

官方示例:

const state = Vue.observable({ count: 0 })


const Demo = {
  render(h) {
    return h('button', {
      on: { click: () => { state.count++ }}
    }, `count is: ${state.count}`)
  }
}

jsx模板組件

以下面的一組狀態(tài)判斷按鈕為例,我們很容易就下意識地在模板內(nèi)寫下這種代碼

<button v-if="status === 1" class="btn1" :class="status === 1" @click="">未開始</button>
<button v-if="status === 2" class="btn2" :class="status === 2" @click="">進(jìn)行中</button>
<button v-if="status === 3" class="btn3" :class="status === 3" @click="">可領(lǐng)取</button>
<button v-if="status === 4" class="btn4" :class="status === 4" @click="">已領(lǐng)取</button>

但是如果我們利用渲染函數(shù)可以將上面的代碼抽取成優(yōu)雅的使用組件

<!DOCTYPE html>
<html lang="en">


<body>
    <div id="app">
        <child :status="status"></child>
    </div>
    <script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    <script>
        Vue.component('child', {
            props: {
                status: {
                    type: Number,
                    required: true
                }
            },
            render(createElement) {
                const innerHTML = ['未開始', '進(jìn)行中', '可領(lǐng)取', '已領(lǐng)取'][this.status]
                return createElement('button', {
                    class: {
                        active: this.status
                    },
                    attrs: {
                        id: 'btn'
                    },
                    domProps: {
                        innerHTML
                    },
                    on: {
                        click: () => console.log(this.status)
                    }
                })
            }
        })
        var app = new Vue({
            el: '#app',
            data: {
                status: 0
            }
        })
    </script>
</body>


</html>

我們將所有的邏輯封裝進(jìn)渲染函數(shù)內(nèi),外部只需要傳遞一個狀態(tài)參數(shù)即可改變

<child :status="status"></child>

動態(tài)組件

通過 Vue 的 元素加一個特殊的 is attribute 可以實現(xiàn)動態(tài)組件的效果

如圖,這是一個 v-for 渲染的列表(只是目前這個版塊才剛開始做,目前只有一個),圓圈內(nèi)的就是一個組件,也就是要 v-for 動態(tài)組件

實際使用

一開始就是基本的組件引入了

import ColorIn from '@/components/Magic/ColorIn.vue'
import LineIn from "@/components/Magic/LineIn.vue";
import Header from "@/components/Magic/Header.vue";
import Footer from "@/components/Magic/Footer.vue";


export default{
      components:{
        ColorIn,
        LineIn,
        Header,
        Footer
    }
}

接下來就是動態(tài) v-for 動態(tài)組件的使用,componentList:['ColorIn','LineIn','Header','Footer']使用下面的代碼即可將代碼依次循環(huán)

<component v-for="(item,index) in componentList" :key="index" :is="item"></component>

編譯以后的效果就是

<ColorIn></ColorIn>
<LineIn></LineIn>
<Header></Header>
<Footer></Footer>

Vue.filter

簡單介紹一下過濾器,顧名思義,過濾就是一個數(shù)據(jù)經(jīng)過了這個過濾之后出來另一樣?xùn)|西,可以是從中取得你想要的,或者給那個數(shù)據(jù)添加點什么裝飾,那么過濾器則是過濾的工具。例如,從['abc','abd','ade']數(shù)組中取得包含‘a(chǎn)b’的值,那么可通過過濾器篩選出來‘a(chǎn)bc’和‘a(chǎn)bd’;把‘Hello’變成‘Hello World’,那么可用過濾器給值‘Hello’后面添加上‘ World’;或者把時間節(jié)點改為時間戳等等都可以使用過濾器。

場景:時間戳轉(zhuǎn)化成年月日這是一個公共方法,所以可以抽離成過濾器使用

// 使用
// 在雙花括號中
{{ message | capitalize }}


// 在 `v-bind` 中
<div v-bind:id="rawId | formatId"></div>


// 全局注冊
Vue.filter('stampToYYMMDD', (value) =>{
  // 處理邏輯
})


// 局部注冊
filters: {
  stampToYYMMDD: (value)=> {
    // 處理邏輯
  }
}


// 多個過濾器全局注冊
// /src/common/filters.js
let dateServer = value => value.replace(/(\d{4})(\d{2})(\d{2})/g, '$1-$2-$3')
export { dateServer }
// /src/main.js
import * as custom from './common/filters/custom'
Object.keys(custom).forEach(key => Vue.filter(key, custom[key]))

.sync 語法糖

sync 就是為了實現(xiàn)prop 進(jìn)行“雙向綁定”僅此而已(父對子,子對父,來回傳)

當(dāng)你有需要在子組件修改父組件值的時候這個方法很好用,它的實現(xiàn)機制和v-model是一樣的。

利用 object.freeze 提升性能

Object.freeze() 方法可以凍結(jié)一個對象。一個被凍結(jié)的對象再也不能被修改;凍結(jié)了一個對象則不能向這個對象添加新的屬性,不能刪除已有屬性,不能修改該對象已有屬性的可枚舉性、可配置性、可寫性,以及不能修改已有屬性的值。此外,凍結(jié)一個對象后該對象的原型也不能被修改。 freeze() 返回和傳入的參數(shù)相同的對象。

比方我們需要渲染一個非常大的數(shù)組對象,例如用戶列表,對象列表,文章列表等等。

export default {
  data: () => ({
    users: {}
  }),
  async created() {
    const users = await axios.get("/api/users");
    this.users = users;
  }
};

vue 會將 data 對象中的所有的屬性加入到 vue 的響應(yīng)式系統(tǒng)中,當(dāng)這些屬性的值發(fā)生改變時,視圖將會產(chǎn)生 響應(yīng),若對象的體積比較大,會消耗很多瀏覽器解析時間。

所以我們可以通過減少數(shù)據(jù)的響應(yīng)式轉(zhuǎn)換來提供前端的性能。

export default {
  data: () => ({
    users: {}
  }),
  async created() {
    const users = await axios.get("/api/users");
    this.users = Object.freeze(users);
  }
};

以上就是W3Cschool編程獅關(guān)于10個簡單的技巧讓你的 vue.js 代碼更優(yōu)雅的相關(guān)介紹了,希望對大家有所幫助。

0 人點贊