全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

Vue之Watcher源码解析(1)

上一节最后再次调用了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  # 跳到  # 的是  # 默认为  # 是一个  # 有什么  # 在这里  # 看了  # 过了  # 一遍  # 不清楚  # 错了  # 第二个  # 可以看到  # 跳转  # 还说  # 报错  # 不接受  # 完后  # 传了  # 大家多多 


相关文章: 宝塔建站助手安装配置与建站模板使用全流程解析  高防网站服务器:DDoS防御与BGP线路的AI智能防护方案  做企业网站制作流程,企业网站制作基本流程有哪些?  学生网站制作软件,一个12岁的学生写小说,应该去什么样的网站?  如何通过虚拟主机快速搭建个人网站?  Avalonia如何实现跨窗口通信 Avalonia窗口间数据传递  建站之星多图banner生成与模板自定义指南  C#如何使用XPathNavigator高效查询XML  网站制作的方法有哪些,如何将自己制作的网站发布到网上?  如何设置并定期更换建站之星安全管理员密码?  网站制作话术技巧,网站推广做的好怎么话术?  如何在阿里云部署织梦网站?  常州企业网站制作公司,全国继续教育网怎么登录?  网站制作培训多少钱一个月,网站优化seo培训课程有哪些?  网站制作软件有哪些,制图软件有哪些?  学校建站服务器如何选型才能满足性能需求?  微课制作网站有哪些,微课网怎么进?  相亲简历制作网站推荐大全,新相亲大会主持人小萍萍资料?  如何在IIS中新建站点并配置端口与IP地址?  如何做静态网页,sublimetext3.0制作静态网页?  制作网站哪家好,cc、.co、.cm哪个域名更适合做网站?  如何快速搭建高效可靠的建站解决方案?  一键网站制作软件,义乌购一件代发流程?  如何在沈阳梯子盘古建站优化SEO排名与功能模块?  如何用手机制作网站和网页,手机移动端的网站能制作成中英双语的吗?  潍坊网站制作公司有哪些,潍坊哪家招聘网站好?  ,购物网站怎么盈利呢?  上海网站制作网站建设公司,建筑电工证网上查询系统入口?  如何高效完成自助建站业务培训?  视频网站app制作软件,有什么好的视频聊天网站或者软件?  如何将凡科建站内容保存为本地文件?  沈阳制作网站公司排名,沈阳装饰协会官方网站?  如何基于云服务器快速搭建网站及云盘系统?  如何高效完成独享虚拟主机建站?  如何在腾讯云服务器快速搭建个人网站?  php8.4新语法match怎么用_php8.4match表达式替代switch【方法】  家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?  西安大型网站制作公司,西安招聘网站最好的是哪个?  西安市网站制作公司,哪个相亲网站比较好?西安比较好的相亲网站?  网站制作网站,深圳做网站哪家比较好?  江苏网站制作公司有哪些,江苏书法考级官方网站?  如何在阿里云服务器自主搭建网站?  西安制作网站公司有哪些,西安货运司机用的最多的app或者网站是什么?  为什么Go需要go mod文件_Go go mod文件作用说明  品牌网站制作公司有哪些,买正品品牌一般去哪个网站买?  长沙企业网站制作哪家好,长沙水业集团官方网站?  建站中国必看指南:CMS建站系统+手机网站搭建核心技巧解析  建站主机如何选?高性价比方案全解析  南京网站制作费用,南京远驱官方网站?  建站之星×万网:智能建站系统+自助建站平台一键生成 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。