专栏名称: JavaScript
面向JavaScript爱好人员提供:前端最新资讯、原创内容、JavaScript、HTML5、Ajax、jQuery、Node.js等一系列教程和经验分享。
目录
相关文章推荐
51好读  ›  专栏  ›  JavaScript

7个有用的 Vue开发技巧

JavaScript  · 公众号  · Javascript  · 2019-06-06 08:36

正文

1 状态共享

随着组件的细化,就会遇到多组件状态共享的情况, Vuex当然可以解决这类问题,不过就像 Vuex官方文档所说的,如果应用不够大,为避免代码繁琐冗余,最好不要使用它,今天我们介绍的是 vue.js 2.6 新增加的 Observable API ,通过使用这个 api 我们可以应对一些简单的跨组件数据状态共享的情况。

如下这个例子,我们将在组件外创建一个 store,然后在 App.vue组件里面使用 store.js 提供的 storemutation方法,同理其它组件也可以这样使用,从而实现多个组件共享数据状态。

首先创建一个 store.js,包含一个 store和一个 mutations,分别用来指向数据和处理方法。

  1. import Vue from "vue";


  2. export const store = Vue.observable({ count: 0 });


  3. export const mutations = {

  4. setCount(count) {

  5. store.count = count;

  6. }

  7. };

然后在 App.vue里面引入这个 store.js,在组件里面使用引入的数据和方法

  1. "app">
  2. "25%" src="./assets/logo.png">

  3. count:{{count}}


  • import { store, mutations } from "./store";

  • export default {

  • name: "App",

  • computed: {

  • count() {

  • return store.count;

  • }

  • },

  • methods: {

  • setCount: mutations.setCount

  • }

  • };


  • 你可以点击 在线DEMO 查看最终效果

    2 长列表性能优化

    我们应该都知道 vue会通过 object.defineProperty对数据进行劫持,来实现视图响应数据的变化,然而有些时候我们的组件就是纯粹的数据展示,不会有任何改变,我们就不需要 vue来劫持我们的数据,在大量数据展示的情况下,这能够很明显的减少组件初始化的时间,那如何禁止 vue劫持我们的数据呢?可以通过 object.freeze方法来冻结一个对象,一旦被冻结的对象就再也不能被修改了。

    1. export default {

    2. data: () => ({

    3. users: {}

    4. }),

    5. async created() {

    6. const users = await axios.get("/api/users");

    7. this.users = Object.freeze(users);

    8. }

    9. };

    另外需要说明的是,这里只是冻结了 users的值,引用不会被冻结,当我们需要 reactive数据的时候,我们可以重新给 users赋值。

    1. export default {

    2. data: () => ({

    3. users: {}

    4. }),

    5. async created() {

    6. const users = await axios.get("/api/users");

    7. this.users = Object.freeze(users);

    8. },

    9. methods:{

    10. // 改变值不会触发视图响应

    11. this.data.users[0] = newValue

    12. // 改变引用依然会触发视图响应

    13. this.data.users = newArray

    14. }

    15. };

    3 去除多余的样式

    随着项目越来越大,书写的不注意,不自然的就会产生一些多余的 css,小项目还好,一旦项目大了以后,多余的 css 会越来越多,导致包越来越大,从而影响项目运行性能,所以有必要在正式环境去除掉这些多余的css,这里推荐一个库purgecss,支持 CLI、JavascriptApi、Webpack 等多种方式使用,通过这个库,我们可以很容易的去除掉多余的 css。

    我做了一个测试,在线DEMO

    1. Hello Vanilla!

    2. We use Parcel to bundle this sandbox, you can find more info about Parcel

    3. href="https://parceljs.org" target="_blank" rel="noopener noreferrer">here.

    1. body {

    2. font-family: sans-serif;

    3. }

    4. a {

    5. color: red;

    6. }

    7. ul {

    8. li {

    9. list-style: none;

    10. }

    11. }

    1. import Purgecss from "purgecss";

    2. const purgecss = new Purgecss({

    3. content: ["**/*.html"],

    4. css: ["**/*.css"]

    5. });

    6. const purgecssResult = purgecss.purge();

    最终产生的 purgecssResult结果如下,可以看到多余的 aul标签的样式都没了

    evernotecid://15DE25B8-A8A2-48DA-BEA3-2B892B3566BA/appyinxiangcom/22977527/ENResource/p232

    4 作用域插槽

    利用好作用域插槽可以做一些很有意思的事情,比如定义一个基础布局组件A,只负责布局,不管数据逻辑,然后另外定义一个组件B 负责数据处理,布局组件A 需要数据的时候就去 B 里面去取。假设,某一天我们的布局变了,我们只需要去修改组件A 就行,而不用去修改组件B,从而就能充分复用组件B 的数据处理逻辑,关于这块我之前写过一篇实际案例,可以点击这里查看。

    这里涉及到的一个最重要的点就是父组件要去获取子组件里面的数据,之前是利用 slot-scope,自 vue 2.6.0 起,提供了更好的支持 slotslot-scope 特性的 API 替代方案。

    比如,我们定一个名为 current-user的组件:

    1. {{ user.lastName }}

    父组件引用 current-user的组件,但想用名替代姓(老外名字第一个单词是名,后一个单词是姓):

    1. {{ user.firstName }}

    这种方式不会生效,因为 user对象是子组件的数据,在父组件里面我们获取不到,这个时候我们就可以通过 v-slot来实现。

    首先在子组件里面,将 user作为一个 元素的特性绑定上去:

    1. v-bind:user="user">

    2. {{ user.lastName }}

    之后,我们就可以在父组件引用的时候,给 v-slot带一个值来定义我们提供的插槽 prop 的名字:

    1. {{ slotProps.user.firstName }}

    这种方式还有缩写语法,可以查看独占默认插槽的缩写语法,最终我们引用的方式如下:

    1. "slotProps">

    2. {{ slotProps.user.firstName }}

    相比之前 slot-scope代码更清晰,更好理解。

    5 属性事件传递

    写过高阶组件的童鞋可能都会碰到过将加工过的属性向下传递的情况,如果碰到属性较多时,需要一个个去传递,非常不友好并且费时,有没有一次性传递的呢(比如react里面的 {...this.props})?答案就是 v-bindv-on

    举个例子,假如有一个基础组件 BaseList,只有基础的列表展示功能,现在我们想在这基础上增加排序功能,这个时候我们就可以创建一个高阶组件 SortList

    1. <BaseList v-bind="$props" v-on="$listeners"> BaseList>


    2. import BaseList from "./BaseList";

    3. // 包含了基础的属性定义

    4. import BaseListMixin from "./BaseListMixin";

    5. // 封装了排序的逻辑

    6. import sort from "./sort.js";


    7. export default {

    8. props: BaseListMixin.props,

    9. components: {

    10. BaseList

    11. }

    12. };

    可以看到传递属性和事件的方便性,而不用一个个去传递

    6 函数式组件

    函数式组件,即无状态,无法实例化,内部没有任何生命周期处理方法,非常轻量,因而渲染性能高,特别适合用来只依赖外部数据传递而变化的组件。

    写法如下:

    1. 在 template标签里面标明 functional

    2. 只接受 props

    3. 不需要 script标签

    1. "app">
    2. <List

    3. :items="['Wonderwoman', 'Ironman']"

    4. :item-click="item => (clicked = item)"

    5. />

    6. Clicked hero: {{ clicked }}


  • import List from "./List";


  • export default {

  • name: "App",

  • data: () => ({ clicked: "" }),

  • components: { List }

  • };

    1. for="item in props.items" @click="props.itemClick(item);">

    2. {{ item }}

  • 7 监听组件的生命周期

    比如有父组件 Parent和子组件 Child,如果父组件监听到子组件挂载 mounted就做一些逻辑处理,常规的写法可能如下:

    1. // Parent.vue

    2. <Child @mounted="doSomething"/>


    3. // Child.vue

    4. mounted() {

    5. this.$emit("mounted");

    6. }

    这里提供一种特别简单的方式,子组件不需要任何处理,只需要在父组件引用的时候通过 @hook来监听即可,代码重写如下:

    1. <Child @hook:mounted="doSomething"/>

    当然这里不仅仅是可以监听 mounted,其它的生命周期事件,例如: createdupdated等都可以,是不是特别方便~

    作者:skinnerl
    https://juejin.im/post/5ce3b519f265da1bb31c0d5f


    好文和朋友一起看~
    推荐文章
    电商行业  ·  身价105亿的蔡文胜告诉你:出身、学历不重要,关键在于你交什么朋友
    7 年前
    创伙伴  ·  4.5亿的教训:李彦宏器重他,雷军投资他,结果他却败了
    7 年前
    亲仁书屋  ·  【健康】脾胃好不好,脸上见分晓!(附调养妙招)
    7 年前
    青核桃  ·  最近这些地方招人啦!找工作的呼和浩特人,赶快去看看
    7 年前
    冷笑话  ·  家里有个智障主人怎么办?在线等,挺急的!
    7 年前