Vuejs进阶
title: Vue.js进阶
date: 2019-09-13 15:47:32
tags: # 这里写的分类会自动汇集到 categories 页面上,分类可以多级
- Vue.js # 一级分类
- Vue.js进阶 # 二级分类
关于h
将 h
作为 createElement
的别名是 Vue 生态系统中的一个通用惯例.
响应式原理
官网解释
当你把一个普通的 JavaScript 对象传入 Vue 实例作为 data 选项,Vue 将遍历此对象所有的属性,并使用 Object.defineProperty 把这些属性全部转为 getter/setter。每个组件实例都对应一个 watcher 实例,它会在组件渲染的过程中把“接触”过的数据属性记录为依赖。之后当依赖项的 setter 触发时,会通知 watcher,从而使它关联的组件重新渲染。
官方解释总结:
- 任何一个 Vue 组件都有一个与之对应的
Watcher
实例。 - Vue 的
data
上的属性会被添加getter
和setter
属性。 - 当 Vue 组件的
render
函数被执行的时候,data
上会被触碰(touch
),即被读,getter
方法会被调用,此时 Vue 会去记录此 Vue 组件所依赖的所有data
。(这一过程被称为依赖收集) data
被改动时(主要是用户操作),即被写,setter
方法会被调用,此时 Vue 会去通知所有依赖于此data
的组件去调用他们的render
函数进行更新。
其他说法:
mvvm 用来初始化数据
observer
用来对初始数据通过Object.defineProperty
添加setter
和getter
,当取数据(即调用 get)的时候添加订阅对象(watcher
)到数组里, 当给数据赋值(即调用 set)的时候就能知道数据的变化,此时调用发布订阅中心的notify
,从而遍历当前这个数据的订阅数组,执行里面所有的watcher
,通知变化update
。
compiler 是用来把 data 编译到 dom 中。分三步:1.先把真实的 dom 移入到内存中fragment
,2.编译:提取想要的元素节点v-model
和文本节点;3.把编译好的 fragment 塞回到页面去。第二步骤中会对编译到 dom 中的data
添加watcher
,当data
变化时,这里的watcher
回调也能收到通知得到执行。
watcher
是oberver
和compiler
之间通信的桥梁。
Object.defineProperty 的缺陷
- 无法监听数组变化。
- 只能劫持对象的属性,因此我们需要对每个对象的每个属性进行遍历,如果属性值也是对象那么需要深度遍历,显然能劫持一个完整的对象是更好的选择。
使用 Proxy 实现 Vue 数据劫持
proxy 定义: Proxy 对象用于定义基本操作的自定义行为(如属性查找,赋值,枚举,函数调用等).
Proxy 第一个参数是目标对象,第二个参数是一个对象,其属性是当执行一个操作时定义代理的行为的函数。这时可以在第二个参数中加入一个 set 方法,这时可以监听到是哪个 key 做了改变。并且通过 Reflect 的 set 方法去模拟真实的 set 方法。
为什么说 Proxy 的性能比 Object.defineProperty 更好呢?
Object.defineProperty
只能监听属性,而 Proxy 能监听整个对象,省去对非对象或数组类型的劫持,也能做到监听。
vue 是对对象每一个属性进行Object.defineProperty
。
第二点,Object.defineProperty
不能监测到数组变化
总结
Proxy 返回的是一个新对象,我们可以只操作新的对象达到目的,而Object.defineProperty
只能遍历对象属性直接修改;
Proxy 作为新标准将受到浏览器厂商重点持续的性能优化,也就是传说中的新标准的性能红利
当然,Proxy 的劣势就是兼容性问题,而且无法用polyfill
实现
Proxy 基本语法
const obj = new Proxy(target, handler);
参数说明如下:
target
: 被代理对象。
handler
: 是一个对象,声明了代理 target 的一些操作。
obj
: 是被代理完成之后返回的对象。
但是当外界每次对 obj 进行操作时,就会执行 handler 对象上的一些方法。handler 中常用的对象方法如下:
1 | 1. get(target, propKey, receiver) |
常用(考)组件之keep-alive
作用: 缓存组件内部状态,避免重新渲染
注意: 和<transition>
相似,<keep-alive>
是一个抽象组件:自身不会渲染一个 DOM 元素,也不会出现在父组件链中.
用法:
缓存动态组件:
<keep-alive>
包裹动态组件时,会缓存不活动的组件实例,而不是销毁它们,此种方式并无太大的实用意义。
1 | <!-- 基本 --> |
使用keep-alive
可以将所有路径匹配到的路由组件都缓存起来,包括路由组件里面的组件,keep-alive
大多数使用场景就是这种。
1 | <keep-alive> |
常用(考)API 之nextTick
作用: $nextTick
是将回调推迟到下次 DOM 更新循环之后再执行,在修改数据之后使用$nextTick
,则可以在回调中获取更新后的 DOM,
Vue 实现响应式并不是数据发生变化之后 DOM 立即变化,而是按一定的策略进行 DOM 的更新。
1 | // 修改数据 |
为了在数据变化之后等待 Vue 完成更新 DOM,可以在数据变化之后立即使用
Vue.nextTick(callback)
。这样回调函数将在 DOM 更新完成后被调用。在组件内使用vm.$nextTick()
实例方法特别方便,因为它不需要全局 Vue,并且回调函数中的 this 将自动绑定到当前的 Vue 实例上.因为$nextTick()
返回一个 Promise 对象,所以可以使用 ES6 语法.
常用(考)API 之 set
返回: 设置的值.
用法: 向响应式对象中添加一个属性,并确保这个新属性同样是响应式的,且触发视图更新。它必须用于向响应式对象上添加新属性,因为 Vue 无法探测普通的新增属性 .
注意对象不能是 Vue 实例,或者 Vue 实例的根数据对象。
常用(考)API 之 watch
一个对象,键是需要观察的表达式,值是对应回调函数。值也可以是方法名,或者包含选项的对象。
Vue 实例将会在实例化时调用 $watch()
,遍历 watch 对象的每一个属性。
1 | <div id="app"> |
取消观察
vm.$watch 返回一个取消观察函数,用来停止触发回调:
1 | var unwatch = vm.$watch("a", cb); |
注意,不应该使用箭头函数来定义
watcher
函数 (例如searchQuery: newValue => this.updateAutocomplete(newValue)
)。理由是箭头函数绑定了父级作用域的上下文,所以this
将不会按照期望指向 Vue 实例,this.updateAutocomplete
将是undefined
。
其他 API
Vue.extend(options)
用法: 使用基础 Vue 构造器创建一个子类.参数是一个包含组件选项的对象.
data 在Vue.extend
中是函数
1 | // 创建构造器 |
Vue.set(target,propertyName/index,value)
返回值: 设置的值
用法: 向响应式添加新属性,并确保这个新属性同样是响应式的.且触发识图更新.它必须用于响应式对象上添加新属性,因为 Vue 无法探测新增的属性.
如果我们在创建实例以后,再在实例上绑定新属性,vue 是无法进行双向绑定的。
注意: 对象不能是 Vue 实例,或者 Vue 实例的根数据对象.
Vue.mixin
用法: 全局注册一个混入,影响注册之后所有创建的每个 Vue 实例.插件作者可以使用混入,向组件注入自定义行为.不推荐在应用代码中使用.
1 | Vue.mixin({ |
虽然文档不建议我们在应用中直接使用 mixin ,但是如果不滥用的话也是很有帮助的,比如可以全局混入封装好的 ajax 或者一些工具函数等等。
mixins
应该是我们最常使用的扩展组件的方式了。如果多个组件中有相同的业务逻辑,就可以将这些逻辑剥离出来,通过mixins
混入代码,比如上拉下拉加载数据这种逻辑等等。
另外需要注意的是mixins
混入的钩子函数会先于组件内的钩子函数执行,并且在遇到同名选项的时候也会有选择性的进行合并.
Vue 页面优化(spa 首屏单页面)
- 压缩代码
- 框架和插件按需加载
- 框架和插件从 CDN 中引入
- 路由懒加载
- SSR 服务端渲染
函数化组件
先设置functional: true
,表示该组件无状态无实例,不能使用this
使用上下文context
进行替换
替换规律:
1 | this.text-----context.props.text |
1 | <div id="app"> |
虚拟 dom
虚拟 DOM 到底是什么,说简单点,就是一个普通的 JavaScript 对象,包含了 tag
、props
、children
三个属性。
虚拟 dom 优点
用 JS 对象模拟 DOM 节点的好处是,页面的更新可以先全部反映在 JS 对象(虚拟 DOM )上,操作内存中的 JS 对象的速度显然要更快,等更新完成后,再将最终的 JS 对象映射成真实的 DOM,交由浏览器去绘制。
1 | <div id="app"> |
上面的 HTML 转换为虚拟 DOM 如下:
1 | { |
该对象就是我们常说的虚拟 DOM 了,因为 DOM 是树形结构,所以使用 JavaScript 对象就能很简单的表示。而原生 DOM 因为浏览器厂商需要实现众多的规范(各种 HTML5 属性、DOM 事件),即使创建一个空的 div 也要付出昂贵的代价。虚拟 DOM 提升性能的点在于 DOM 发生变化的时候,通过 diff 算法比对 JavaScript 原生对象,计算出需要变更的 DOM,然后只对变化的 DOM 进行操作,而不是更新整个视图.
diff 算法
diff 算法用来比较两棵 Virtual DOM 树的差异,如果需要两棵树的完全比较,那么 diff 算法的时间复杂度为 O(n^3)。但是在前端当中,你很少会跨越层级地移动 DOM 元素,所以 Virtual DOM 只会对同一个层级的元素进行对比,如下图所示, div 只会和同一层级的 div 对比,第二层级的只会跟第二层级对比,这样算法复杂度就可以达到 O(n)。
Vue 中的虚拟 dom
Vue 通过建立一个虚拟 DOM 来追踪自己要如何改变真实 DOM。请仔细看这行代码:
1 | return createElement("h1", this.blogTitle); |
createElement 到底会返回什么呢?其实不是一个实际的 DOM 元素。它更准确的名字可能是 createNodeDescription,因为它所包含的信息会告诉 Vue 页面上需要渲染什么样的节点,包括及其子节点的描述信息。我们把这样的节点描述为“虚拟节点 (virtual node)”,也常简写它为“VNode”。“虚拟 DOM”是我们对由 Vue 组件树建立起来的整个 VNode 树的称呼。
render 函数(渲染函数)
1 | <template id="hdom"> |
render 函数的第一个参数
在 render 函数的方法中,参数必须是createElement
,它的类型是function
createElement
的第一个参数必选.类型可以是String|Object|Function
1 | Vue.component("child", { |
render 函数的第二个参数
createElement
的第二个参数可选.参数是数据对象,只能是Object
1 | Vue.component('child',{ |
render 函数的第三个参数
createElement
的第三个参数可选.参数可以是String|Array
,代表子节点
1 | Vue.component("child", { |
在 render 函数中使用this.$slots
第三个参数存的是 VNode,也就是虚拟节点.组件树中的所有 VNode 必须是唯一的.
1 | Vue.component("my-component", { |
在 render 函数中使用 props 传数据
1 | <div id="app"> |
在 render 函数中使用v-model
v-model
作用: 接收input
的内容并绑定到后面的值上.
1 | <div id="app"> |
render 函数中使用作用域插槽
1 | <div id="app"> |