Vue.js最佳实践(五招助你成为vuejs大师)菜鸟教程网_编程入门基础知识

本文面向对象是有一定Vue.js编程经验的开发者。如果有人需要Vue.js入门系列的文章可以在评论区告诉我,有空就给你们写。对大部分人来说,掌握Vue.js基本的几个API后就已经能够正常地开发前端网站。但如果你想更加高效地使用Vue来开发,成为Vue.js大师,那下面我要传授的这五招你一定得认真学习一下了。第一招:化繁为简的Watchers场景还原:cre

Vue.js最佳实践(五招助你成为vuejs大师)菜鸟教程网

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

Vue.js最佳实践(五招助你成为vuejs大师)菜鸟教程网_编程入门基础知识

对大部分人来说,掌握Vue.js基本的几个API后就已经能够正常地开发前端网站。但如果你想更加高效地使用Vue来开发,成为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('focus', $event)"
   @input="$emit('input', $event.target.value)"
  >
 </label>
</template>

有下面几个优化点:

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

<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传回给父组件而已,那其实和上面类似,我完全没必要显式地申明:

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

海计划公众号
(0)
上一篇 2020/03/29 01:39
下一篇 2020/03/29 01:39

您可能感兴趣的内容

  • 首页白屏优化实践小白攻略_优化使用帮助

    前言自从前端三大框架React、Vue、Angular面世以来,前端开发逐渐趋向规范化、统一化,大多数时候新建前端项目,首先想到使用的技术一定是三大框架之一,框架给前端开发带来了极大的便利和规范,但是由于这三大框架都是JS驱动,在JS没有解析加载完成之前页面无法展示,会处于长时间的白屏,带来了一定的用户体验问题,接下来本篇文章会介绍本人最近在白屏优化时遇到的

    2020/03/24
  • Awesome-qr.js入门基础_优雅的花式JS二维码生成器

    Awesome-qr.js入门基础 官方网址:https://www.bitcat.cc/webapp/awesome-qr/index.html GitHub:http://ww…

    2020/03/10
  • Webpack 4正式发布了!菜鸟教程下载_webpack小白帮助

    webpack 4(Legato)正式发布了!你可以使用 yarn 或者 npm 获得它: yarn add webpack –dev或者npm i webpack –save-dev为什么叫 Legato?首先我们会开始一个新传统:为我们以后的每个大版本设定代号!因此,我们决定将命名这个特权给予我们最大的 OpenCollective 捐赠者:triv

    2020/04/05
  • multiscroll.js零基础入门一款jQuery左右垂直反向滚动插件

    multiscroll.js小白常识 官方网址:http://alvarotrigo.com/multiScroll/ GitHub:https://github.com/alva…

    2020/03/06
  • anyupload小白常识_一个极度纯净的上传插件

    anyupload小白常识 GitHub:https://github.com/dianbaer/anyupload 简介描述:一个极度纯净的上传插件 anyupload是一个极度…

    2020/03/11
  • 码良使用教程_高扩展的在线网页制作平台

    码良使用教程 官方网址:https://godspen.ymm56.com/ GitHub:https://github.com/ymm-tech/gods-pen 简介描述:高扩…

    2020/03/11
  • ReactJS Components: 基础指南入门基础_Component菜鸟教程

    创建和管理React组件的各种方式,涌现的大量状态管理工具等等都是这些挑战的焦点。我们今天能做的就是在React(基于社区选择)中将最常用的做法引入桌面并讨论它们。 常规的 React Component按照常规,我的意思是常见的,你可能在大多数代码库和文章中看到过:var Hello = React.createClass({render: functi

    2020/04/05
  • 和讯科技基础入门_报道科技中国 分享全球智慧

    和讯科技基础入门 官方网址:http://tech.hexun.com/ 简介描述:报道科技中国 分享全球智慧 和讯科技是和讯网核心资讯频道,以报道科技中国,分享全球智慧为目标,为…

    2020/03/06
  • typescript中类型断言理解小白攻略_断言入门基础教程

    typescript中类型断言好理解也好用类型断言是个好用的玩意,虽然typescript很强大,但是有时还不如我们知道一个值的类型,导致在开发过程中总是报一些令人头痛的类型错误。使用断言,简单来说就是先做好一个假设,使得编译通过。我一开始接触类型断言时是有点不明白的,后来我了解到原因是 “**类型断言更像是类型的选择,而不是类型转换”。**我发现不少博客文

    2020/03/22
  • 程序猿搞笑趣图菜鸟指南_程序员使用说明

    1、客户需求vs最终产品2、程序员的一天寂寞的时候干什么?写程序写程序写程序失恋的时候干什么?写程序写程序写程序发骚的时候干什么?写程序写程序写程序剩下的时候干什么?调程序调程序调程序3、开发平台的差异4、C++ vs C…You have no class(没地位,无级别)…5、凌晨三点,灯火阑珊处十年生死两茫茫,写程序,到天亮。千行代码,Bug何

    2020/04/03
  • js调用函数的几种方法基础入门ES5/ES6的函数调用方式_函数小白知识

    这篇文章主要介绍ES5中函数的4种调用,在ES5中函数内容的this指向和调用方法有关。以及ES6中函数的调用,使用箭头函数,其中箭头函数的this是和定义时有关和调用无关一、ES5函数调用模式包括函数名()和匿名函数调用,this指向window function getSum() {console.log(this) //window}getSum()(

    2020/04/05
  • js之map及转换json、Object使用攻略_json基础知识

    搞后段HashMap用的还是很多,其实js里也有提供了map。搞一段map怎么玩let map = new Map()
    map.set(‘a’,’a’)
    map.set(‘b’,’b’)
    console.log(map)
    console.log(map.get(‘a’))
    以上简单的创建了一个map,显然存值用set、取值用get,和java 的还是蛮相似

    2020/03/24
  • hamburgers指南攻略_超酷图标变形动画特效

    hamburgers指南攻略 官方网址:https://jonsuh.com/hamburgers GitHub:https://github.com/jonsuh/hamburg…

    2020/03/06
  • 利用canvas实现环形进度条菜鸟教程_canvas基础入门

    前提:有时候在项目中会有用到进度条的情况,使用css3也可以实现,但是对于性能不好的设备,或者网络不好的情况下,卡顿现象非常明显,避免出现不流畅的尴尬情况,所以记录一下,使用canvas来实现的方法。效果图: DOM中,首先定义canvas画板元素: <canvas id="canvas" width="500" height="500" style="

    2020/03/30
  • 微信小程序分享到朋友圈方法与技巧攻略教程_技巧小白指南

    小程序提供onShareAppMessage 函数,此函数只支持分享给我微信朋友。小程序如何分享到朋友圈呢?我提供的方法是,使用canvas绘制一张图片,并用wx.previewImage预览图片,然后长按图片保存图片到手机。再通过发朋友圈的方式,选择保存的图片,当用户浏览朋友圈时,可以长按图片、识别图中二维码进入小程序。效果展示 准备

    2020/04/05
  • React Fiber入门指南_线程使用教程

    背景前段时间准备前端招聘事项,复习前端React相关知识;复习React16新的生命周期:弃用了componentWillMount、componentWillReceivePorps,componentWillUpdate三个生命周期, 新增了getDerivedStateFromProps、getSnapshotBeforeUpdate来代替弃用的三个钩

    2020/03/29