Vue之Watcher源码解析(1)
发布时间 - 2026-01-11 02:24:57 点击率:次上一节最后再次调用了mount函数,我发现竟然跳到了7000多行的那个函数,之前我还说因为声明早了被覆盖,看来我错了!

就是这个函数:
// Line-7531
Vue$3.prototype.$mount = function(el, hydrating) {
el = el && inBrowser ? query(el) : undefined;
return mountComponent(this, el, hydrating)
};
第一步query就不用看了,el此时是一个DOM节点,所以直接返回,然后调用了mountComponent函数。
// Line-2375
function mountComponent(vm, el, hydrating) {
vm.$el = el;
/* 检测vm.$options.render */
// 调用钩子函数
callHook(vm, 'beforeMount');
var updateComponent;
/* istanbul ignore if */
if ("development" !== 'production' && config.performance && mark) {
/* 标记vue-perf */
} else {
updateComponent = function() {
vm._update(vm._render(), hydrating);
};
}
// 生成中间件watcher
vm._watcher = new Watcher(vm, updateComponent, noop);
hydrating = false;
// 调用最后一个钩子函数
if (vm.$vnode == null) {
vm._isMounted = true;
callHook(vm, 'mounted');
}
return vm
}
这个函数做了三件事,调用beforeMount钩子函数,生成Watcher对象,接着调用mounted钩子函数。
数据双绑、AST对象处理完后,这里的Watcher对象负责将两者联系到一起,上一张网上的图片:
可以看到,之前以前把所有的组件都过了一遍,目前就剩一个Watcher了。
构造新的Watcher对象传了3个参数,当前vue实例、updateComponent函数、空函数。
// Line-2697
var Watcher = function Watcher(vm, expOrFn, cb, options) {
this.vm = vm;
// 当前Watcher添加到vue实例上
vm._watchers.push(this);
// 参数配置 默认为false
if (options) {
this.deep = !!options.deep;
this.user = !!options.user;
this.lazy = !!options.lazy;
this.sync = !!options.sync;
} else {
this.deep = this.user = this.lazy = this.sync = false;
}
this.cb = cb;
this.id = ++uid$2;
this.active = true;
this.dirty = this.lazy; // for lazy watchers
this.deps = [];
this.newDeps = [];
// 内容不可重复的数组对象
this.depIds = new _Set();
this.newDepIds = new _Set();
// 把函数变成字符串形式`
this.expression = expOrFn.toString();
// parse expression for getter
if (typeof expOrFn === 'function') {
this.getter = expOrFn;
} else {
this.getter = parsePath(expOrFn);
if (!this.getter) {
this.getter = function() {};
"development" !== 'production' && warn(
"Failed watching path: \"" + expOrFn + "\" " +
'Watcher only accepts simple dot-delimited paths. ' +
'For full control, use a function instead.',
vm
);
}
}
// 不是懒加载类型调用get
this.value = this.lazy ?
undefined :
this.get();
};
该构造函数添加了一堆属性,第二个参数由于是函数,直接作为getter属性加到watcher上,将字符串后则作为expression属性。
最后有一个value属性,由于lazy为false,调用原型函数gei进行赋值:
// Line-2746
Watcher.prototype.get = function get() {
pushTarget(this);
var value;
var vm = this.vm;
if (this.user) {
try {
value = this.getter.call(vm, vm);
} catch (e) {
handleError(e, vm, ("getter for watcher \"" + (this.expression) + "\""));
}
} else {
// 调用之前的updateComponent
value = this.getter.call(vm, vm);
}
// "touch" every property so they are all tracked as
// dependencies for deep watching
if (this.deep) {
traverse(value);
}
popTarget();
this.cleanupDeps();
return value
};
// Line-750
Dep.target = null;
var targetStack = [];
function pushTarget(_target) {
// 默认为null
if (Dep.target) {
targetStack.push(Dep.target);
}
// 依赖目前标记为当前watcher
Dep.target = _target;
}
function popTarget() {
Dep.target = targetStack.pop();
}
原型方法get中,先设置了依赖收集数组Dep的target值,user属性暂时不清楚意思,跳到了else分支,调用了getter函数。而getter就是之前的updateComponent函数:
// Line-2422
updateComponent = function() {
vm._update(vm._render(), hydrating);
};
这个函数不接受参数,所以说传进来的两个vm并没有什么卵用,调用这个函数会接着调用_update函数,这个是挂载到vue原型的方法:
// Line-2422
Vue.prototype._render = function() {
var vm = this;
var ref = vm.$options;
var render = ref.render;
var staticRenderFns = ref.staticRenderFns;
var _parentVnode = ref._parentVnode;
// 检测是否已挂载
if (vm._isMounted) {
// clone slot nodes on re-renders
for (var key in vm.$slots) {
vm.$slots[key] = cloneVNodes(vm.$slots[key]);
}
}
// 都没有
vm.$scopedSlots = (_parentVnode && _parentVnode.data.scopedSlots) || emptyObject;
if (staticRenderFns && !vm._staticTrees) {
vm._staticTrees = [];
}
vm.$vnode = _parentVnode;
// render self
var vnode;
try {
// 调用之前的render字符串函数
vnode = render.call(vm._renderProxy, vm.$createElement);
} catch (e) {
/* handler error */
}
// return empty vnode in case the render function errored out
if (!(vnode instanceof VNode)) {
/* 报错 */
vnode = createEmptyVNode();
}
// set parent
vnode.parent = _parentVnode;
return vnode
};
方法获取了一些vue实例的参数,比较重点的是render函数,调用了之前字符串后的ast对象:
在这里有点不一样的地方,接下来的跳转有点蒙,下节再说。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# Vue
# Watcher
# Vue之Watcher源码解析(2)
# vue如何实现observer和watcher源码解析
# 简单实现Vue的observer和watcher
# 跳到
# 的是
# 默认为
# 是一个
# 有什么
# 在这里
# 看了
# 过了
# 一遍
# 不清楚
# 错了
# 第二个
# 可以看到
# 跳转
# 还说
# 报错
# 不接受
# 完后
# 传了
# 大家多多
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
iOS正则表达式验证手机号、邮箱、身份证号等
如何快速搭建个人网站并优化SEO?
HTML5空格和margin有啥区别_空格与外边距的使用场景【说明】
如何快速搭建高效简练网站?
香港服务器选型指南:免备案配置与高效建站方案解析
网站制作企业,网站的banner和导航栏是指什么?
Laravel如何从数据库删除数据_Laravel destroy和delete方法区别
标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?
如何在建站主机中优化服务器配置?
如何为不同团队 ID 动态生成多个“认领值班”按钮
手机软键盘弹出时影响布局的解决方法
Laravel如何实现多语言支持_Laravel本地化与国际化(i18n)配置教程
php读取心率传感器数据怎么弄_php获取max30100的心率值【指南】
Laravel怎么使用artisan命令缓存配置和视图
JavaScript如何实现类型判断_typeof和instanceof有什么区别
小米17系列还有一款新机?主打6.9英寸大直屏和旗舰级影像
如何用ChatGPT准备面试 模拟面试问答与职场话术练习教程
Laravel PHP版本要求一览_Laravel各版本环境要求对照
黑客如何通过漏洞一步步攻陷网站服务器?
Laravel如何获取当前用户信息_Laravel Auth门面获取用户ID
制作旅游网站html,怎样注册旅游网站?
Laravel路由Route怎么设置_Laravel基础路由定义与参数传递规则【详解】
悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】
如何在 Go 中优雅地映射具有动态字段的 JSON 对象到结构体
JavaScript常见的五种数组去重的方式
Laravel如何记录日志_Laravel Logging系统配置与自定义日志通道
如何在云服务器上快速搭建个人网站?
如何快速搭建二级域名独立网站?
Laravel如何实现用户注册和登录?(Auth脚手架指南)
如何确保FTP站点访问权限与数据传输安全?
谷歌浏览器下载文件时中断怎么办 Google Chrome下载管理修复
如何用腾讯建站主机快速创建免费网站?
bing浏览器学术搜索入口_bing学术文献检索地址
微信小程序 五星评分(包括半颗星评分)实例代码
PHP怎么接收前端传的文件路径_处理文件路径参数接收方法【汇总】
简单实现Android验证码
javascript中数组(Array)对象和字符串(String)对象的常用方法总结
Laravel软删除怎么实现_Laravel Eloquent SoftDeletes功能使用教程
详解jQuery中基本的动画方法
JavaScript数据类型有哪些_如何准确判断一个变量的类型
PHP 500报错的快速解决方法
Laravel怎么配置不同环境的数据库_Laravel本地测试与生产环境动态切换【方法】
Laravel怎么使用Markdown渲染文档_Laravel将Markdown内容转HTML页面展示【实战】
Laravel如何使用Gate和Policy进行权限控制_Laravel权限判定与策略规则配置
如何用手机制作网站和网页,手机移动端的网站能制作成中英双语的吗?
在线ppt制作网站有哪些软件,如何把网页的内容做成ppt?
python中快速进行多个字符替换的方法小结
iOS UIView常见属性方法小结
,在苏州找工作,上哪个网站比较好?
Laravel如何配置和使用队列处理异步任务_Laravel队列驱动与任务分发实例

