解析vue中的$mount


Posted in Javascript onDecember 21, 2017

本文主要是带领大家分析$mount。

$mount所做的工作从大体来讲主要分为3步:

1.如果你的option里面没有 render 函数,那么,通过 compileToFunctions 将HTML模板编译成可以生成VNode的Render函数。

2.new 一个 Watcher 实例,触发 updateComponent 方法。

3.生成vnode,经过patch,把vnode更新到dom上。 由于篇幅有限,这里先说前两步,第三步下篇说。 好,下面具体的说。首先,我们来到 $mount 函数,如下图:

解析vue中的$mount 

我们呢可以看到,代码首先判断option里面有没有render函数,没有的话,进一步判断有没有template,没有的话就用dom元素的outerHTML。得到template以后干什么了呢?如下图。

解析vue中的$mount  

我们可以看到,调用了 compileToFunctions 将template转成render函数。这里面有两个过程:

  • 将template解析成ast语法树。
  • 通过ast语法树生成render函数。

具体的将template解析成ast语法树在本文就不说了,有时间单独开一个章节分析。好,这下我们拿到render函数了,那么接下来一步干什么了呢?没错,就开始 mountComponent 了。如下图:

解析vue中的$mount  

可以从上图看到,程序声明了一个 updateComponent 方法,这个是将要被 Watcher 实例调用的更新组件的方法,过一会分析到 Watcher 的时候将会看到。至于为什么会有个判断语句来根据条件声明 updateComponent 方法,其实从 performance 可以看出,其中一个方法是用来测试 render 和 update 性能的。好我们终于该到 Watcher 了,先看这句代码:

// we set this to vm._watcher inside the watcher's constructor
// since the watcher's initial patch may call $forceUpdate (e.g. inside child
// component's mounted hook), which relies on vm._watcher being already defined
new Watcher(vm, updateComponent, noop, null, true /* isRenderWatcher */);

我们先来分析一下注释里所说的 _watcher 是啥玩意呢?其实看看 forceupdate 的代码就知道了:

Vue.prototype.$forceUpdate = function () {
 var vm = this;
 if (vm._watcher) {
  vm._watcher.update();
 }
 };

就是调用这个vm的 _watcher 的 update 方法。用来强制更新。为什么叫强制更新呢?vue里面有判断,如果新值 == 旧值, 那么就不触发watcher更新视图了~ 所以,如果非要更新就要调用 forceupdate 来强制更新了。好,让我们来看一看传进去的参数吧:

  • vm:当前的vm实例
  • updateComponent 这个非常重要,用来在后面将vnode更新到dom上的。
  • noop 无意义的函数
  • null option选项,没有则为null
  • true 主要是用来判断是哪个watcher的。因为computed计算属性和如果你要在options里面配置watch了同样也是使用了 new Watcher ,加上这个用以区别这三者。好,我们来看看 new Watcher 都做了什么事,如下图。

解析vue中的$mount 

首先,我们看到代码有个这个判断

if (isRenderWatcher) {
 vm._watcher = this;
}

可以看到,如果声明这个watcher的上下文是用来渲染视图的,也就是说是在 mountComponent 这里调用的 new Watcher 的时候,才会把this赋值给_watcher。然后把 watcher push到 _watchers 里面,目的是等到组件销毁时顺便把watcher也销毁掉。然后就是初始化watcher的成员,代码如下:

this.deep = this.user = this.lazy = this.sync = false;<br />

接下来,就是赋值给 getter , this.getter = expOrFn 。还记得刚才传过来的 updateComponent 函数么,没错,就是这个赋值给我 getter 。然后我们就到了:

this.value = this.lazy
 ? undefined
 : this.get();

进入到 get 方法里面,我们看看到底做了什么。get代码如下图:

解析vue中的$mount 

我们可以看到,首先它执行的是 pushTarget(this) , pushTarget(this) 代码如下:

function pushTarget (_target) {
 if (Dep.target) { targetStack.push(Dep.target); }
 Dep.target = _target;
}

也就是说如果当前有 Dep.target 的话,就把target放到 targetStack 里面,如果没有的话,就设为当前的target,也就是这个watcher。 接着,就是执行了它的 getter 属性,也就是刚刚传入 updateComponent 函数。而 updateComponent 就是我们开篇提到第三步了。

总结

以上所述是小编给大家介绍的vue中的$mount,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
服务器安全设置的几个注册表设置
Jul 28 Javascript
iframe自适应宽度、高度 ie6 7 8,firefox 3.86下测试通过
Jul 29 Javascript
JavaScript中为元素加上name属性的方法
May 09 Javascript
jQuery绑定事件不执行但alert后可以正常执行
Jun 03 Javascript
jQuery中remove()方法用法实例
Dec 25 Javascript
浅析JavaScript中浏览器的兼容问题
Apr 19 Javascript
浅谈jquery采用attr修改form表单enctype不起作用的问题
Nov 25 Javascript
关于JavaScript语句后面的分号问题
Dec 07 Javascript
JS实现的3des+base64加密解密算法完整示例
May 18 Javascript
vue prop属性传值与传引用示例
Nov 13 Javascript
JS出现404错误原理及解决方案
Jul 01 Javascript
Element-ui树形控件el-tree自定义增删改和局部刷新及懒加载操作
Aug 31 Javascript
vue中使用refs定位dom出现undefined的解决方法
Dec 21 #Javascript
js中bool值的转换及“&amp;&amp;”、“||”、 “!!”详解
Dec 21 #Javascript
利用node实现一个批量重命名文件的函数
Dec 21 #Javascript
详解webpack3编译兼容IE8的正确姿势
Dec 21 #Javascript
利用vue开发一个所谓的数独方法实例
Dec 21 #Javascript
开发Vue树形组件的示例代码
Dec 21 #Javascript
详解使用vuex进行菜单管理
Dec 21 #Javascript
You might like
多重?l件?合查?(二)
2006/10/09 PHP
require(),include(),require_once()和include_once()的异同
2007/01/02 PHP
不重新编译PHP为php增加openssl模块的方法
2011/06/14 PHP
php简单获取复选框值的方法
2016/05/11 PHP
微信公众号开发之获取位置信息php代码
2018/06/13 PHP
TP - 比RBAC更好的权限认证方式(Auth类认证)
2021/03/09 PHP
AJAX分页的代码(后台asp.net)
2011/02/14 Javascript
jquery实现的可隐藏重现的靠边悬浮层实例代码
2013/05/27 Javascript
Javascript实现带关闭按钮的网页漂浮广告代码
2014/01/12 Javascript
JavaScript调试工具汇总
2014/12/23 Javascript
JavaSacript中charCodeAt()方法的使用详解
2015/06/05 Javascript
简述JavaScript对传统文档对象模型的支持
2015/06/16 Javascript
举例讲解jQuery对DOM元素的向上遍历、向下遍历和水平遍历
2016/07/07 Javascript
javascript中apply/call和bind的使用
2017/02/15 Javascript
bootstrap实现的自适应页面简单应用示例
2017/03/09 Javascript
vue实现简单的MVVM框架
2018/08/05 Javascript
用Object.prototype.toString.call(obj)检测对象类型原因分析
2018/10/11 Javascript
Javascript的this详解
2019/03/23 Javascript
Vue退出登录时清空缓存的实现
2019/11/12 Javascript
jQuery 实现扁平式小清新导航
2020/07/07 jQuery
[02:17]《辉夜杯》TRG战队巡礼
2015/10/26 DOTA
详解Python实现多进程异步事件驱动引擎
2017/08/25 Python
Python算法之求n个节点不同二叉树个数
2017/10/27 Python
Python中使用支持向量机SVM实践
2017/12/27 Python
Python实现简易版的Web服务器(推荐)
2018/01/29 Python
python实现人脸识别经典算法(一) 特征脸法
2018/03/13 Python
修复 Django migration 时遇到的问题解决
2018/06/14 Python
python 根据网易云歌曲的ID 直接下载歌曲的实例
2019/08/24 Python
Python高并发解决方案实现过程详解
2020/07/31 Python
Python中的面向接口编程示例详解
2021/01/17 Python
纯CSS绘制漂亮的圆形图案效果
2014/05/07 HTML / CSS
英国翻新电子产品购物网站:Tech Trade
2017/12/25 全球购物
武汉世纪畅想数字传播有限公司 .NET笔试题
2015/06/13 面试题
租赁协议书范本
2014/04/22 职场文书
党性心得体会
2014/09/03 职场文书
详解如何修改nginx的默认端口
2021/03/31 Servers