1.插值表达式 {{}}
2.指令
2.1 v-if 、v-else、 v-show、v-else-if
v-if: 在切换过程中条件块内的事件监听器和子组件适当地被销毁和重建,开销较高,在运行时条件很少改变时使用。
v-show:调整css dispaly属性,开销较小,在常频繁地切换时使用
2.2 v-for
数组 v-for="item in items"
加key diff算法高效的更新虚拟DOM
对象 v-for="(value, key, index) in object"
2.3 v-text 、v-html
v-text {{xxx}}取值有个弊端,当网速很慢或javascript出错时,会在页面显示{{xxx}},Vue提供的v-text可以解决这个问题
v-html 用于输出html代码
2.4 v-on
<button v-on:click="add">加分</button>
缩写 <button @click="add">加分</button>
2.5 v-moudle
需要在data中声明初始值
单选 radio 多选 checkbox 下拉 select/option
2.6 v-bind
<img v-bind:src="imgSrc" width="200px">
缩写 <img :src="imgSrc" width="200px">
2.7 v-pre、v-cloak、v-once
v-pre 在模板中跳过vue的编译,直接输出原始值,如果在标签中加入v-pre就不会输出vue中的data值了
v-cloak 在vue渲染完指定的整个DOM后才进行显示。它必须和CSS样式一起使用
[v-cloak] {
display: none;
}
v-once 只显示DOM第一次渲染的值,以后不改变了
生命周期
new vue() 实例化对象 > 初始化生命周期和事件 >
beforeCreate 实例刚被创建还没实例化,做一些初始化操作 > 初始化依赖注入和校验
created 实例已创建,属性已绑定,dom未生成,$el属性不存在,此时可发起网络请求,结束加载动画 >
has 'el' option 存在则继续,不存在则等到vm.$mount(el)继续编译 >
has 'template' option 有则继续编译 render函数首次执行 否则用html
beforeMount 完成虚拟dom,模板已编译,data里的数据和模板生成html,未挂载到页面上 >
給vue实例添加$el成员,并替换掉挂载的dom元素
mounted dom结构完成,可做网络请求 >
beforeUpdate data数据发生变化,触发组件重新渲染,组件发生变化之前,虚拟dom已发生更改 >
重新渲染虚拟dom,并通过diff算法对比vnode节点差异更新真实dom
updated 组件更新之后
beforeDestory 组件销毁之前 > 拆卸观察者、子组件、事件监听者
destoryed 解绑,事件监听器移除,子实例销毁
常用选项
methods
用于html中绑定的方法
计算属性
对于任何复杂逻辑,都应当使用计算属性
提供的函数将用作 vm.data. 的 getter 函数
计算属性是基于它们的响应式依赖进行缓存的。只在相关响应式依赖发生改变时它们才会重新求值,否则不会重新执行函数
不是响应式依赖不缓存
计算属性的getter/setter
侦听器
watch
当需要在数据变化时执行异步或开销较大的操作时,这个方式是最有用的
对引用数据类型监听
filters过滤器
vue-cli
安装 npm vue-cli -g npm install -g @vue/cli
升级 npm update -g @vue/cli
检查版本 vue -V
初始化项目 vue init <template-name> <project-name>
<template-name>:表示模板名称,vue-cli官方提供的5种模板,一般使用webpack
自动打开浏览器 config>index>autoOpenBrowser属性改为true
vue组件传值 传值取数据
事件总线
vue事件总线就像所有组件的事件中心,在组件中,我们可以使用 $emit,$on,$off 分别来分发、监听、取消监听事件
全局事件总线创建
全局事件总线操作
事件总线
创建一个单独的event-bus.js文件
在vue页面销毁时,同时移除EventBus 事件监听
}
EventBus.$off() -----------移除当前组件的所有事件监听者
EventBus.$off(param:string)------------移除当前组件对某个事件的所有监听
EventBus.$off(param:string,param)-------移除某个方法对某个事件的监听如:this.addTodo
$parent / $children与 ref
ref:如果在普通的dom元素上使用,引用指向的就是dom元素,如果用在子组件上,引用就是指向组件实例
this.$refs['string']
this.$parent.
this.$children[].
这两种方法的弊端是,无法在跨级或兄弟间通信
$attrs/$listeners
$attrs包含了父作用域中不被prop所识别(且获取)的特性绑定,当一个组件没有声明任何prop时,会包含所有父作用域的绑定,
并通过v-bind='$attrs'传入组件内部,通常配合inheritAttrs选项一起使用
inheritAttrs: false, // 可以关闭自动挂载到组件根元素上的没有在props声明的属性
$listeners包含了父作用域中(不含.native修饰器的)v-on事件监听器,它可以通过v-on='$listeners'传入组件内部
是$emit的偷懒写法
provide与inject 用于父组件向子孙组件传递数据
使用方法
provide在父组件中返回要传给下级的数据,inject在需要使用这个数据的子辈组件或者孙辈等下级组件中注入数据
provide:是一个对象,或者是一个返回对象的函数。里面呢就包含要给子孙后代的东西,也就是属性和属性值。
inject:一个字符串数组,或者是一个对象。属性值可以是一个对象,包含from和default默认值
父组件调用子组件方法
通过 ref
通过 $emit发射事件传事件
通过 $emit/$on
通过 $children[0]
通过总线、vuex、
子组件调用父组件方法
通过 props传函数
通过 $parent
通过 $emit发生事件传信号
通过总线、vuex
vuex
vuex是vue.js应用程序开发的状态管理模式
安装 npm install vuex --save
使用
访问数据 this.$store.state.
用计算属性优雅
使用mapState
vuex的getters,相当于vuex的计算属性,state参数必传
修改状态 Mutations
使用方法 this.$store.commit('setName')
使用mapMutations
异步修改状态Actions
使用方法 this.$store.dispatch('asyncNewName',{})
使用mapActions
import {mapActions} from 'vuex'
在methods中解构
moudle 把vuex模块化封装
vue-router
#后hash值得变化,不会导致刷新页面,会触发hashchange
widow.addEventLister('hashChange',update)
引:history 模式,H5提供pushState,replaceState两个API,通过API修改url地址不会发送请求刷新页面。
因为没有#号更加美观,但当用户刷新页面时还是会发送请求,为了避免这种个情况,需要服务端支持,把所有路由重定向到根页面
安装 npm install vue-router --save
路由跳转
通过router-link标签跳转
<router-link :to="{path:'/first',params:{id:1}}">first</router-link>,获取不到参数,路由不能加/:id,否则不能跳转
<router-link :to="{path:'/first',query:{id:1}}">first</router-link>
<router-link :to="{name:'first',params:{id:1}}">first</router-link>
获取参数 this.$route.params.id 路由配置/first/:id 刷新不会消失
<router-link :to="{name:'first',query:{id:1}}">first</router-link>
获取参数 this.$route.query.id 相当于get传参
编程式跳转
this.$router.push()
this.$router.push('/home')
this.$router.push({name:'home'})
this.$router.push({path:'/home'})
this.$router.push({name:'home',query: {id:'1'}})
this.$router.push({path:'/home',query: {id:'1'}})
this.$router.push({name:'home',params: {id:'1'}})
this.$router.replace() (用法同push)
this.$router.go(n)
路由守卫
Vue 路由 导航守卫(全局守卫、路由独享守卫、组件内守卫)
to:即将要进入的目标路由对象
from:当前导航即将要离开的路由对象
next :调用该方法后,才能进入下一个钩子函数(afterEach)
next() 进入管道的下一个钩子
next(false) 中断当前导航,回到from路由对应的地址
next('/') 或者next({path:'/'})跳转到其他路由
next(error) 导航终止,且该错误会传递给router.onError()注册过的回调
全局守卫
router.beforeEach((to,from,next)=>{})
router.beforeResolve((to,from,next)=>{}) 全局解析守卫,路由跳转前,所有组件内守卫和异步路由组件被解析之后触发
router.afterEach((to,from)=>{})
路由守卫
{
path:'first',
name:'first',
component:'first',
beforeEnter:(to,from,next)=>{}
}
组件守卫
beforeRouteEnter(to,from,next){}
beforeRouteUpdate(to,from){} 组件内子路由更新
beforeRouteLeave(to,from){}
面试题汇总
vue修饰符
.lazy 输入框光标离开时才改变 .trim过滤空格
.number 将值转换为数字 先输入数字,只保留数字,先输入字母,number修饰符无效
.stop 阻止冒泡 .capture 由外往内捕获
.prevent 阻止默认事件 .native加在自定义时间上,保证事件能执行
对象新属性无法更新视图,删除属性无法更新视图
原因:Object.definePropertym没有对对象的新属性进行劫持
使用Vue.$set(obj,key,value)
删除用 Vue.$delete(obj,key)
直接arr[index] = xxx无法更新视图
原因:Vue没有对数组进行数据劫持
使用数组的splice方法
使用Vue.$set(arr,index,value)
nextTick的用处
Vue是异步更新 此方法是在更新之后
SSR服务端渲染
Vue响应式原理怎么实现的
对象内部使用defineReactive方法,使用Object.defineProperty将属性进行劫持,数组则是重写,每个属性都有自己的dep属性
存放他所依赖的watcher,属性变化时通知自己对应的watcher去更新
为什么不对数组进行数据劫持?
数据量
实际面试总结
Vue页面加载优化
路由懒加载
路由懒加载的几种方法
component:()=>import('@/views/login.vue') 解构
component:resolve=>require(['@/views/login'],resolve) 引入并赋值
const login = ()=>import('./views/login.vue')
component:login
webpack提供的require.ensure()实现懒加载
const HelloWorld=resolve=>{
require.ensure(['@/components/HelloWorld'],()=>{
resolve(require('@/components/HelloWorld'))
})
}
路由懒加载做了什么事
主要作用是将路由对应的组件打包成一个个的js代码块,只有在这个路由被访问到的时候,才加载对应的组件,否则不加载!
组件懒加载
const one = ()=>import('./one')
components:{one}
components:{
one:resolve=>require(['./one'],resolve)
}
按需加载 elementui
较大图片进行压缩/换成网络地址
CDN引入
不生成.map文件
build出来的dist文件夹里面有很多的.map文件,这些文件主要是帮助线上调试代码,查看样式.
由于基本都是本地调试好,线上不需要怎么修改,所以就禁止生成这些文件
config>index.js中 productionSourceMap改为false
使用gzip压缩
安装compression-webpack-plugin
Vue改变数组的方法 七种
push pop shift unshift reverse sort spilice
vue侦听器
方法格式的侦听器
watch:{
name(newName,oldName){
}
}
缺点:首次进入浏览器,无法立即触发第一次。侦听对象,对象数据发生变化,不会触发。
对象格式的侦听器
优点:可以通过immediate属性,控制侦听器自动触发一次(默认是false)。可以 通过deep属性,控制侦听器深度监听到对象中每一个属性的变化
watch:{
obj:{
handler(newval){
},
immediate:true,
deep:true
}
}
如果想要侦听对象里某一个子属性的变化,可以通过''单引号进行包裹
watch:{
'obj.name'(newval,oldval){
}
}
侦听器的特点
它的命名不能随意,必须和你想侦听的属性名称完全一致
它不能手动调用,它是自动触发
不需要有任何返回值
它可以侦听异步操作中数据的变化
computed/watch/filter的区别
computed当页面中需要使用大量的表达式处理数据时
watch数据变化时执行异步或开销较大的操作时
filter与computed的主要区别在于它可以当做函数使用
computed 对于其中变量的依赖是多个的 watch 的依赖则是单个的
vue前端权限控制
接口权限
接口权限目前一般采用jwt的形式来验证,没有通过的话一般返回401,跳转到登录页面重新进行登录
登录完拿到token,将token存起来,通过axios请求拦截器进行拦截,每次请求的时候头部携带token
axios.interceptors.request.use(config => {
config.headers['token'] = cookie.get('token')
return config
})
axios.interceptors.response.use(res=>{},{response}=>{
if (response.data.code === 40099 || response.data.code === 40098) { //token过期或者错误
router.push('/login')
}
})