Vue.js最佳实践(五招助你成为vuejs大师)

转自https://www.jb51.net/article/139448.htm

本文面向对象是有一定Vue.js编程经验的开发者。如果有人需要Vue.js入门系列的文章可以在评论区告诉我,有空就给你们写。

对大部分人来说,掌握Vue.js基本的几个API后就已经能够正常地开发前端网站。但如果你想更加高效地使用Vue来开发,成为Vue.js大师,那下面我要传授的这五招你一定得认真学习一下了。

第一招:化繁为简的Watchers

场景还原:

?


1

2

3

4

5

6

7

8

created(){

  this.fetchPostList()

},

watch: {

  searchInputValue(){

    this.fetchPostList()

  }

}

组件创建的时候我们获取一次列表,同时监听input框,每当发生变化的时候重新获取一次筛选后的列表这个场景很常见,有没有办法优化一下呢?

招式解析:

首先,在watchers中,可以直接使用函数的字面量名称;其次,声明immediate:true表示创建组件时立马执行一次。

?


1

2

3

4

5

6

watch: {

  searchInputValue:{

    handler: ‘fetchPostList‘,

    immediate: true

  }

}

第二招:一劳永逸的组件注册

场景还原:

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

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动态将需要的基础组件统统打包进来。

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

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 的变化来初始化数据,如下:

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

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是解决了,可每次这么写也太不优雅了吧?秉持着能偷懒则偷懒的原则,我们希望代码这样写:

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

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直接设置为路由的完整路径,一举两得。

?


1

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

第四招: 无所不能的render函数

场景还原:

vue要求每一个组件都只能有一个根元素,当你有多个根元素时,vue就会给你报错

?


1

2

3

4

5

6

7

8

9

10

<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)

?


1

2

3

4

5

6

7

8

9

10

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过来的一系列事件。举例子:

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

//父组件

<BaseInput

  :value="value"

  label="密码"

  placeholder="请填写密码"

  @input="handleInput"

  @focus="handleFocus>

</BaseInput>

//子组件

<template>

 <label>

  {{ label }}

  <input

   :value="value"

   :placeholder="placeholder"

   @focus=$emit(‘focus‘, $event)"

   @input="$emit(‘input‘, $event.target.value)"

  >

 </label>

</template>

有下面几个优化点:

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

?


1

2

3

4

5

<input

   :value="value"

   v-bind="$attrs"

   @input="$emit(‘input‘, $event.target.value)"

  >

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

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

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

<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 ,这些默认行为将会被去掉, 以上两点的优化才能成功。

结尾

原文地址:https://www.cnblogs.com/lhuser/p/11219035.html

时间: 2024-10-04 16:08:10

Vue.js最佳实践(五招助你成为vuejs大师)的相关文章

Vue.js最佳实践

Vue.js最佳实践 第一招:化繁为简的Watchers 场景还原: created(){ this.fetchPostList() }, watch: { searchInputValue(){ this.fetchPostList() } } 组件创建的时候我们获取一次列表,同时监听input框,每当发生变化的时候重新获取一次筛选后的列表这个场景很常见,有没有办法优化一下呢? 招式解析: 首先,在watchers中,可以直接使用函数的字面量名称:其次,声明immediate:true表示创建

require.js 最佳实践

require.js(官网:http://www.requirejs.org/)是一个js库,相关的基础知识,前面转载了两篇博文:Javascript模块化编程(require.js), Javascript模块化工具require.js教程,RequireJS 参考文章 1. require.js的主要作用是js的工程化,规范化: 1)它是一个js脚本的加载器,它遵循AMD(Asynchronous Module Definition)规范,实现js脚本的异步加载,不阻塞页面的渲染和其后的脚本

Vue 工程化最佳实践

目录结构 总览 api 目录用于存放 api 请求,文件名与模型名称基本一致,文件名使用小驼峰,方法名称与后端 restful 控制器一致. enums 目录存放 常量,与后端的常量目录对应 icons 目录用于存放图标,element-ui 提供的图标实在是太少啦.所以我通常会使用 阿里的 iconfont lang 目录存放多语言 layouts 目录存放布局 上面展示的是一个后台系统,empty 为一个空布局.用于登录页面,其他页面则使用 default 布局.布局不需要过多介绍,写过 l

Doodoo.js 发布 1.1.0,Koa.js+ Nuxt.js 最佳实践

doodoo.js发布1.1.0 -- 中文最佳实践Node.js Web快速开发框架,支持Koa.js, Express.js中间件.包含多项功能改进,及Bug修复.更新内容: 1.[新增]新增支持.js7格式文件2.[新增]新增支持redis密码设置3.[更新]更新默认不连接mysql4.[新增]新增支持mongodb数据库5.[新增]新增支持设置api路由前缀6.[移除]移除默认static静态服务器7.[新增]新增static插件8.[新增]新增自定义plugin9.[新增]新增webh

Vue.js 开发实践:实现精巧的无限加载与分页功能

https://segmentfault.com/a/1190000005351971#articleHeader9 本篇文章是一篇Vue.js的教程,目标在于用一种常见的业务场景--分页/无限加载,帮助读者更好的理解Vue.js中的一些设计思想.与许多Todo List类的入门教程相比,更全面的展示使用Vue.js完成一个需求的思考过程:与一些构建大型应用的高阶教程相比,又更专注于一些零碎细节的实现,方便读者快速掌握.致用. 需求分析 当一个页面中信息量过大时(例如一个新闻列表中有200条新闻

前端开发工具vue.js开发实践总结

最近有很长时间没有更新博客了,换了公司,全部的心思都放在项目上了.通过这次项目的上线,让我感受最深的是前后端分离后,前端页面的模块化管理,以及前端页面的数据邦定.在接触vue.js之前,我之前端要用到的dom结构,都是通过拼接字符串的方式进行输出的.这种方式最大的痛点是拼接很麻烦,也不是很直观,几乎无法复用,和数据紧密的偶合在一起,维护不方便,太多太多的问题.在同事的推荐下,开始接触vue.js这个开源项目.如果你不知道什么是vue,那么请看这里的介绍http://vuejs.org/. vue

[DIV/CSS] Vue.js 开发实践:实现精巧的无限加载与分页功能

本篇文章是一篇Vue.js的教程,目标在于用一种常见的业务场景--分页/无限加载,帮助读者更好的理解Vue.js中的一些设计思想.与许多Todo List类的入门教程相比,更全面的展示使用Vue.js完成一个需求的思考过程:与一些构建大型应用的高阶教程相比,又更专注于一些零碎细节的实现,方便读者快速掌握.致用. 需求分析 当一个页面中信息量过大时(例如一个新闻列表中有200条新闻需要展示),就会产生问题,例如: 数据量过大,影响加载速度 用户体验差,很难定位到之前自己看过的某篇文章 扩展性差,如

Vue.js实战 章五:内置指令

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta http-equiv="X-UA-Compatible" con

vue组件最佳实践

看了老外的一篇关于组件开发的建议(强烈建议阅读英文原版),感觉不错翻译一下加深理解. 这篇文章制定一个统一的规则来开发你的vue程序,以至于达到一下目的.1.让开发者和开发团队更容易发现一些事情.2.让你更好的利用你的IDE.3.让你更加容易的使用打包工具4.让你的代码更容易碎片化以达到复用的目的. 基于模块开发 用一些功能单一的小模块来组织你的应用 Why? 对于你自己和你团队的人来说较小的模块更容易看懂 维护 复用和调试. How? 每个组件应该保持单一 独立 可复用 可测试把你很大的组件拆