vue是一个前端js库,用来动态构建用户界面,采用MVVM模式,重要技术为模板和数据绑定,组件和虚拟DOM
Vue专注于MVVM中的viewmodel层,通过双向数据绑定,把view层和model层连接起来。核心是用数据来驱动DOM。
优点:
缺点:
指令过多,自带模板不容易拓展,组件数据传递没有react直观和明显
Vue实列从创建到销毁的过程,叫做Vue的生命周期。也就是从开始创建,初始化数据,编译模板,挂载DOM——>渲染,更新——>渲染,卸载等一系列过程。
vue生命周期中有多个事件钩子,他让我们在控制vue实例的过程中更容易形成好的逻辑
共有8个阶段,分别为创建前后,挂载前后,更新前后,销毁前后
beforeCreate, created,beforMount, mounted
DOM渲染在mounted周期已完成
生命周期 | 应用场景 |
---|---|
beforeCreate | 加入loading事件 |
created | 初始化完成时的事件写在这里,如在这结束loading事件,异步请求也适宜在这里调用 |
mounted | 挂载元素,获取到DOM节点 |
updated | 如果对数据统一处理,在这里写上相应函数 |
beforeDestroy | 可以做一个确认停止事件的确认框 |
nextTick | 更新数据后立即操作dom |
生命周期 | 状态 |
---|---|
beforeCreate | vue实例的挂载元素el和数据对象data都是undefined,还没有初始化 |
created | vue实例的数据对象data有了,可以访问里面的数据和方法,未挂载在DOM,el还没有 |
beforeMount | vue实例的el和data都初始化了,但是挂载之前为虚拟的dom节点 |
mounted | vue实例挂载到真实DOM上,可以通过DOM获取DOM节点 |
beforeUpdate | 响应式数据更新时调用,发生在虚拟DOM打补丁之前,适合在更新之前访问现有的DOM,比如手动移除已添加的事件监听器 |
updated | 虚拟DOM重新渲染和打补丁之后调用,组成新的DOM已经更新,避免在这个钩子函数中操作数据,防止死循环 |
beforeDestroy | 实例销毁前调用,还可以用this能获取到实例,常用于销毁定时器,解绑事件 |
destroyed | 实例销毁后调用,调用后所有事件监听器会被移除,所有的子实例都会被销毁 |
MVVM是前端视图层的开发思想,主要关注于视图层。
M,model代表数据模型,负责保存数据(可以在model中定义数据和操作的业务逻辑)
V,View代表UI组件,她负责将数据模型展现出来。
VM,ViewModel监听数据模型的改变和控制视图行为,处理用户交互,简单来说就是同步view和model的对象
用户操作view层,view数据变化会同步到Model,Model数据变化会立即反应到view中。viewModel通过双向数据绑定把view层和Model层连接了起来
M,Model数据层,用来保存数据
C,controller控制层,负责接收View和用户响应,将数据传递给Model或者从Model中获取数据交给View展现
V,View视图层,负责从controller层获取数据显示数据及响应用户操作
Model和View没有直接通信,都是通过中间人Controller来传递,所以Controller中的负荷越来越重
SPA是在web页面初始化时加载相应的html,css,js,一旦加载完成,就不会因为用户的操作而进行页面的重新加载和跳转,取而代之的是利用路由机制实现html内容的转换,ui与用户的交互,避免用户的重新加载
优点:
缺点:
区别:
computed是计算属性,它依赖于其他属性,并且computed具有缓存,只有他依赖的属性值发生变化时,下次获取的computed值才会重新计算。
watch,更多的是观察者,是一个监听函数,每当监听的数据变化时都会执行回调函数进行后续操作。
应用场景:
每当我们需要进行数值计算时,并且需要依赖其他属性时,我们应该使用computed,因为我们可以利用computed的缓存特性,避免没次获取值时都需要重新计算
当我们需要在数据发生变化时执行异步操作或者开销较大的操作时,应该使用watch
相似处
功能: 动态构建用户界面
虚拟DOM: 内部都通过虚拟DOM提高效率
组件化: 都使用组件化编程思想编写项目
Props: 组件间通信的基本方式
构建工具: 脚手架, webpack
不同点
界面: JSX与模板
界面更新: 状态管理 与 对象属性监视
数据绑定: 单向与双向
vue的思想是响应式的,也是基于数据可变的,其整体思想仍然是拥抱经典的html,css,js鼓励开发者使用template模板,并提供指令供开发者使用,因此在开发web应用有一种再写经典web应用的感觉
react整体是函数式的思想,把组件设计成纯组件,状态和逻辑通过参数传入,其中组件使用jsx语法,将html,css全都融入js中jsx语法相对比较灵活,当组件调用setState或者props发生变化时,组件内部的render会重新渲染,子组件也会重新渲染。
Object.defineProperty只能劫持对象的属性,proxy却可以劫持到整个对象并返回一个新对象,proxy不仅可以代理对象,还可以代理数组
proxy优势
Object.defineProperty 的优势如下
兼容性好
因为组件是可以复用的,js中对象是引用关系,如果data是一个对象,那么就会与子组件中的data值互相污染,产生副作用。当是一个函数时,每个实例可以维护一份被返回对象的独立拷贝
keep-alive是vue一个内置的组件,可以使被包含的组件保留状态,防止被重新渲染
key是为vue中vnodel的唯一标识,通过这个key我们可以更加准确、迅速的进行diff操作
vue.js采用数据劫持的方法,结合发布者订阅者通过object.definepropertty来劫持各个属性的setter,getter以监听数据的变化,在数据变化的时候发布消息给订阅者,触发相应监听回调函数
vue虚拟DOM在渲染过程中,会跟踪每一个组件的依赖关系,不需要渲染整个组件树,而对于react来说,当组件的状态改变时,以该组件为根的所有子组件都需要被重新渲染
v-show的值无论是true还是false,元素都会存在在html代码中,而v-show只有为true时,原素才存在
单向数据绑定优点是,流动方向可以跟踪,流动单一没有状态,这样使得单向绑定能够避免状态管理在复杂度上升时产生的各种问题,程序的调试相对简单,单向数据流更利于状态的维护和优化,更利于组件之间的通信以及组件复用
双向数据流的优点双向绑定在一些需要实时反应用户输入的场合非常方便,用户在视图的修改会自动保存在数据模型中,数据模型中的值也会立刻同步到视图中
缺点:双向数据流是自动管理状态的,但在实际应用中会有很多不得不手动操作状态变化的逻辑,使程序复杂度上升,无法追踪局部状态的变化,同时也导致问题的源头难以被追踪到
双绑跟单向绑定之间的差异只在于,双向绑定把数据变更的操作隐藏在框架内部,调用者并不会直接感知。
通过vm对象来代理data中所有属性的操作,主要是简化vm中的data对象的数据
实现原理:通过Object.defineProperty()给vm添加data对象中的属性,所有添加的属性都有getter/setter,在getter/setter内部操作data中对应的属性数据
还可以通过 e m i t 方 法 出 发 一 个 消 息 , 然 后 emit方法出发一个消息,然后 emit方法出发一个消息,然后on接收这个消息
祖先组件中通过 provider 来提供变量,然后在子孙组件中通过 inject 来注入变量。 provide / inject API 主要解决了跨级组件间的通信问题,不过它的使用场景,主要是子组件获取上级组件的状态,跨级组件间建立了一种主动提供与依赖注入的关系
vuex是一个状态管理模式,对vue应用中多个组件的共享状态进行集中式的管理。每一个vuex应用的核心是store(仓库),它包含着你应用中的大部分状态
vuex的状态存储是响应式的,当vue组件从中读取状态的时候,若store中的状态发生变化,那么相对应的组件也会得到更新
改变store中状态的唯一途径是commit一个mutation
store中主要包括以下几个模块
vue-router 有 3 种路由模式:hash、history、abstract
hash模式:在浏览器中符号 ‘#’ 以及后面的字符称之为hash,用window.loaction.hash读取,不会交给服务器端,只在浏览器段使用
history模式: 利用HTML5新特性pushState()/replaceState()及popState事件监听,路径会提交给后台
导航守卫是主要功能是监视路由跳转和控制路由跳转
(1)全局导航守卫
=============================全局前置守卫 ==================================
router.beforeEach((to, from, next) => {
next();
});
=============================全局解析守卫 ==================================
router.beforeResolve((to, from, next) => {
next();
});
=============================全局后置钩子 ==================================
router.afterEach((to, from) => {
console.log('afterEach 全局后置钩子');
})
(2)路由独享的守卫
你可以在路由配置上直接定义 beforeEnter 守卫:
const router = new VueRouter({
routes: [
{
path: '/foo',
component: Foo,
beforeEnter: (to, from, next) => {
// ...
}
}
]
})
(3)组件内的守卫
const Foo = {
template: `...`,
beforeRouteEnter (to, from, next) {
// 在渲染该组件的对应路由被 confirm 前调用
// 不!能!获取组件实例 `this`
// 因为当守卫执行前,组件实例还没被创建
next(vm => {
// 通过 `vm` 访问组件实例
})
},
beforeRouteUpdate (to, from, next) {
// 在当前路由改变,但是该组件被复用时调用
// 举例来说,对于一个带有动态参数的路径 /foo/:id,在 /foo/1 和 /foo/2 之间跳转的时候,
// 由于会渲染同样的 Foo 组件,因此组件实例会被复用。而这个钩子就会在这个情况下被调用。
// 可以访问组件实例 `this`
},
beforeRouteLeave (to, from, next) {
// 导航离开该组件的对应路由时调用
// 可以访问组件实例 `this`
}
}
1)一个函数 VueRouter:路由构建函数,用来创建路由对象,配置路由
2)两个对象
$route :代表当前的路由对象,包含当前路由的相关信息(path,params参数,query参数) 后面跟属性
$router :代表路由器对象,包含路由跳转的各种方法(push,replace,back)后面跟方法
3)三个标签
<router-link>: 路由链接, 生成路由链接,相当于a标签
<router-view>: 路由视图, 显示当前路由组件
<keep-alive>: 缓存路由组件对象