成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

vue源碼解讀-component機(jī)制

1treeS / 1982人閱讀

摘要:通過的合并策略合并添加項(xiàng)到新的構(gòu)造器上緩存父構(gòu)造器處理和相關(guān)響應(yīng)式配置項(xiàng)在新的構(gòu)造器上掛上的工具方法緩存組件構(gòu)造器在上總的來說是返回了一個(gè)帶有附加配置相的新的的構(gòu)造器。在函數(shù)中,構(gòu)造器叫做,等待時(shí)候初始化。

身為原來的jquery,angular使用者。后面接觸了react和vue。漸漸的喜歡上了vue。抱著學(xué)習(xí)的態(tài)度呀。看看源碼。果然菜要付出代價(jià)。一步步單步調(diào)試。頭好疼??吹侥睦镉浀侥睦?。來一點(diǎn)點(diǎn)心得。錯(cuò)誤的地方請(qǐng)幫我指出來。謝謝。最近看的是vue component部分。

先上一段最簡(jiǎn)單的代碼,來剖析component機(jī)制:
{{a}}
111
{{a}}

我們按照瀏覽器的思維逐行來。執(zhí)行到腳本時(shí)。首先執(zhí)行了

Vue.component("my-component", {
  name: "my-component",
  template: "
A custom component!
", components: { Child:Child }, created(){ console.log(this); }, mounted(){ console.log(this); } })

我們來看看這個(gè)函數(shù)經(jīng)歷了什么:
vue.js初始化的時(shí)候。調(diào)用了initGlobalAPI(vue),為vue掛上了工具函數(shù)vue.component
經(jīng)過initGlobalAPI(vue)中的initAssetRegisters (Vue) 后。變?yōu)?/p>

vue.component = function (
      id,
      definition
    ) {
      if (!definition) {
        return this.options[type + "s"][id]
      } else {
        /* istanbul ignore if */
        {
          if (type === "component" && config.isReservedTag(id)) {
            warn(
              "Do not use built-in or reserved HTML elements as component " +
              "id: " + id
            );
          }
        }
        if (type === "component" && isPlainObject(definition)) {
          definition.name = definition.name || id;
          definition = this.options._base.extend(definition);
        }
        if (type === "directive" && typeof definition === "function") {
          definition = { bind: definition, update: definition };
        }
        this.options[type + "s"][id] = definition;//全局的組件或者指令和過濾器。統(tǒng)一掛在vue.options上。等待init的時(shí)候利用策略合并侵入實(shí)例。供實(shí)例使用
        return definition
      }
    };

this.options._base在initGlobalAPI(vue)中為Vue.options._base = Vue;
so vue.component調(diào)用了vue.extend。找到了源頭。我們來好好看看這個(gè)vue.extend這個(gè)function。代碼如下:

Vue.extend = function (extendOptions) {
    extendOptions = extendOptions || {};
    var Super = this;
    var SuperId = Super.cid;
    var cachedCtors = extendOptions._Ctor || (extendOptions._Ctor = {});
    if (cachedCtors[SuperId]) {
      return cachedCtors[SuperId]//如果組件已經(jīng)被緩存在extendOptions上則直接取出
    }

    var name = extendOptions.name || Super.options.name;
    {
      if (!/^[a-zA-Z][w-]*$/.test(name)) {
        warn(
          "Invalid component name: "" + name + "". Component names " +
          "can only contain alphanumeric characters and the hyphen, " +
          "and must start with a letter."//校驗(yàn)組件名
        );
      }
    }

    var Sub = function VueComponent (options) {
      this._init(options);
    };
    Sub.prototype = Object.create(Super.prototype);
    Sub.prototype.constructor = Sub;//將vue上原型的方法掛在Sub.prototype中,Sub的實(shí)例同時(shí)也繼承了vue.prototype上的所有屬性和方法。
    Sub.cid = cid++;
    Sub.options = mergeOptions(
      Super.options,
      extendOptions//通過vue的合并策略合并添加項(xiàng)到新的構(gòu)造器上
    );
    Sub["super"] = Super;緩存父構(gòu)造器

    // For props and computed properties, we define the proxy getters on
    // the Vue instances at extension time, on the extended prototype. This
    // avoids Object.defineProperty calls for each instance created.
    if (Sub.options.props) {
      initProps$1(Sub);
    }
    if (Sub.options.computed) {   //處理props和computed相關(guān)響應(yīng)式配置項(xiàng)
      initComputed$1(Sub);
    }

    // allow further extension/mixin/plugin usage
    Sub.extend = Super.extend;
    Sub.mixin = Super.mixin;
    Sub.use = Super.use;

    // create asset registers, so extended classes
    // can have their private assets too.              //在新的構(gòu)造器上掛上vue的工具方法
    ASSET_TYPES.forEach(function (type) {
      Sub[type] = Super[type];
    });
    // enable recursive self-lookup
    if (name) {
      Sub.options.components[name] = Sub;
    }

    // keep a reference to the super options at extension time.
    // later at instantiation we can check if Super"s options have
    // been updated.
    Sub.superOptions = Super.options;
    Sub.extendOptions = extendOptions;
    Sub.sealedOptions = extend({}, Sub.options);

    // cache constructor
    cachedCtors[SuperId] = Sub;//緩存組件構(gòu)造器在extendOptions上
    return Sub
  };
}

function initProps$1 (Comp) {
  var props = Comp.options.props;
  for (var key in props) {
    proxy(Comp.prototype, "_props", key);
  }
}

function initComputed$1 (Comp) {
  var computed = Comp.options.computed;
  for (var key in computed) {
    defineComputed(Comp.prototype, key, computed[key]);
  }
}

總的來說vue.extend是返回了一個(gè)帶有附加配置相的新的vue的構(gòu)造器。在函數(shù)中,構(gòu)造器叫做Sub,等待render時(shí)候初始化。
經(jīng)過vue.component的調(diào)用。vue增加了一個(gè)全局組件my-component;此時(shí)vue.options.component如下圖:

前三個(gè)是vue內(nèi)置的三個(gè)組件,在initgloabalapi的時(shí)候初始化。
至此全局組件創(chuàng)建完成。全局組件放置在最底層。在以后的策略合并里會(huì)在子組件中的component項(xiàng)的__proto__中。

通過組件的遞歸創(chuàng)建渲染來看vue整體的生命周期(理解vue如何巧妙構(gòu)建應(yīng)用)

上圖:

vue官方的生命周期圖,其實(shí)也就是vue組件的構(gòu)成的生命周期。沿著new Vue()我們來大概看看這些生命周期在什么階段

Vue.prototype._init = function (options) {
    var vm = this;
    // a uid
    vm._uid = uid$1++;

    var startTag, endTag;
    /* istanbul ignore if */
    if ("development" !== "production" && config.performance && mark) {
      startTag = "vue-perf-init:" + (vm._uid);
      endTag = "vue-perf-end:" + (vm._uid);
      mark(startTag);
    }

    // a flag to avoid this being observed
    vm._isVue = true;
    // merge options
    if (options && options._isComponent) {
      // optimize internal component instantiation
      // since dynamic options merging is pretty slow, and none of the
      // internal component options needs special treatment.
      initInternalComponent(vm, options);//內(nèi)部組件調(diào)用此快捷方法
    } else {
      vm.$options = mergeOptions(
        resolveConstructorOptions(vm.constructor),//策略合并,每項(xiàng)屬性都有對(duì)應(yīng)的合并規(guī)則
        options || {},
        vm
      );
    }
    /* istanbul ignore else */
    {
      initProxy(vm);//屬性代理,即vm.xx = vm.data.xx
    }
    // expose real self
    vm._self = vm;
    initLifecycle(vm);//初始化生命周期狀態(tài)變量,建立子父關(guān)系初始值,如$children,$parent.
    initEvents(vm);//?初始化事件
    initRender(vm);//初始化render核心函數(shù)_$createElement和$slots $scopedSlots等
    callHook(vm, "beforeCreate");
    initInjections(vm); // resolve injections before data/props
    initState(vm);//利用數(shù)據(jù)劫持做響應(yīng)式
    initProvide(vm); //resolve provide after data/props
    callHook(vm, "created");

    /* istanbul ignore if */
    if ("development" !== "production" && config.performance && mark) {
      vm._name = formatComponentName(vm, false);
      mark(endTag);
      measure(((vm._name) + " init"), startTag, endTag);
    }

    if (vm.$options.el) {
      vm.$mount(vm.$options.el);//如果有el配置相則主動(dòng)掛載。觸發(fā)之后的compile.render
    }
  };

介紹了大概的_init函數(shù),我們繼續(xù)往下看程序的運(yùn)行。完成了vue.component()之后。開始執(zhí)行new vue(),創(chuàng)建實(shí)例。
對(duì)照_init函數(shù)。我們知道它分別進(jìn)行了對(duì)傳入?yún)?shù)的合并。初始化實(shí)例參數(shù)。創(chuàng)建響應(yīng)的響應(yīng)式。最后掛載:vm.$mount(vm.$options.el);
簡(jiǎn)單說說掛載。好吧。我們還是往方法里面看,掛載的時(shí)候發(fā)生了什么:

// public mount method
Vue$3.prototype.$mount = function (
  el,
  hydrating
) {
  el = el && inBrowser ? query(el) : undefined;
  return mountComponent(this, el, hydrating)
};

var mount = Vue$3.prototype.$mount//緩存mount,用來觸發(fā)render
Vue$3.prototype.$mount = function (//核心mount用來構(gòu)建render函數(shù)
  el,
  hydrating
) {
  el = el && query(el);

  /* istanbul ignore if */
  if (el === document.body || el === document.documentElement) {
    "development" !== "production" && warn(
      "Do not mount Vue to  or  - mount to normal elements instead."//檢測(cè),排除不可掛載的元素
    );
    return this
  }

  var options = this.$options;
  // resolve template/el and convert to render function
  if (!options.render) {
    var template = options.template;//假如輸入的是template模版時(shí)。
    if (template) {
      if (typeof template === "string") {
        if (template.charAt(0) === "#") {
          template = idToTemplate(template);
          /* istanbul ignore if */
          if ("development" !== "production" && !template) {
            warn(
              ("Template element not found or is empty: " + (options.template)),
              this
            );
          }
        }
      } else if (template.nodeType) {
        template = template.innerHTML;//輸入的是dom節(jié)點(diǎn)時(shí)
      } else {
        {
          warn("invalid template option:" + template, this);
        }
        return this
      }
    } else if (el) {
      template = getOuterHTML(el);//如果是一個(gè)id,如此次初始化掛載的id=app,會(huì)取到id=app的html
    }
    if (template) {
      /* istanbul ignore if */
      if ("development" !== "production" && config.performance && mark) {
        mark("compile");
      }

      var ref = compileToFunctions(template, {
        shouldDecodeNewlines: shouldDecodeNewlines,//核心compile函數(shù)。用于生成render函數(shù)。這里不細(xì)說
        delimiters: options.delimiters
      }, this);
      var render = ref.render;
      var staticRenderFns = ref.staticRenderFns;
      options.render = render;//掛載render到實(shí)例options中。待調(diào)用
      options.staticRenderFns = staticRenderFns;//靜態(tài)的元素區(qū)分開。提升性能,后續(xù)虛擬dom樹比較時(shí),不會(huì)比較靜態(tài)節(jié)點(diǎn)

      /* istanbul ignore if */
      if ("development" !== "production" && config.performance && mark) {
        mark("compile end");
        measure(((this._name) + " compile"), "compile", "compile end");
      }
    }
  }
  return mount.call(this, el, hydrating)//利用緩存的mount調(diào)用準(zhǔn)備好的render
};

$mount方法的核心其實(shí)就是準(zhǔn)備好組件的render函數(shù)。這里最核心的一個(gè)方法就是:

var ref = compileToFunctions(template, {
        shouldDecodeNewlines: shouldDecodeNewlines,//核心compile函數(shù)。用于生成render函數(shù)。這里不細(xì)說
        delimiters: options.delimiters
      }, this);

compileToFunctions這個(gè)函數(shù)中主要做了兩件事:
1:對(duì)模版進(jìn)行compile(按標(biāo)簽解析,生成ast(抽象語法樹)
2:利用generate(ast, options),生成render函數(shù)語法

我們來看看最后實(shí)例生成的render函數(shù):

沒有錯(cuò)就是這個(gè)樣子,很有感覺。生成的render函數(shù)保存在options中,等待調(diào)用

好吧。開始調(diào)用吧。

mount.call(this, el, hydrating)=》mountComponent(this, el, hydrating)=》updateComponent = function () {
      vm._update(vm._render(), hydrating);
    };=》vm._watcher = new Watcher(vm, updateComponent, noop);

new Watcher中會(huì)主動(dòng)調(diào)用updateComponent去touch依賴(給頁面中引用過的data中的變量假如監(jiān)聽)正式調(diào)用render函數(shù)。既然都說了。那就來看看render函數(shù):

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 = [];
    }
    // set parent vnode. this allows render functions to have access
    // to the data on the placeholder node.
    vm.$vnode = _parentVnode;
    // render self
    var vnode;
    try {
      vnode = render.call(vm._renderProxy, vm.$createElement);//核心函數(shù),調(diào)用render
    } catch (e) {
      handleError(e, vm, "render function");
      // return error render result,
      // or previous vnode to prevent render error causing blank component
      /* istanbul ignore else */
      {
        vnode = vm.$options.renderError
          ? vm.$options.renderError.call(vm._renderProxy, vm.$createElement, e)
          : vm._vnode;
      }
    }
    // return empty vnode in case the render function errored out
    if (!(vnode instanceof VNode)) {
      if ("development" !== "production" && Array.isArray(vnode)) {
        warn(
          "Multiple root nodes returned from render function. Render function " +
          "should return a single root node.",
          vm
        );
      }
      vnode = createEmptyVNode();
    }
    // set parent
    vnode.parent = _parentVnode;
    return vnode
  };

_render()間接調(diào)用了vnode = render.call(vm._renderProxy, vm.$createElement);

然后結(jié)合render函數(shù)??纯窗l(fā)生了什么。vm.$createElement是核心的創(chuàng)建虛擬dom的函數(shù)。

繼續(xù)看看核心構(gòu)建虛擬dom函數(shù):

function createElement (
  context,
  tag,
  data,
  children,//children是該元素下的所有子元素
  normalizationType,
  alwaysNormalize
) {
  if (Array.isArray(data) || isPrimitive(data)) {
    normalizationType = children;
    children = data;
    data = undefined;
  }
  if (isTrue(alwaysNormalize)) {
    normalizationType = ALWAYS_NORMALIZE;
  }
  return _createElement(context, tag, data, children, normalizationType)
}

function _createElement (
  context,
  tag,
  data,
  children,
  normalizationType
) {
  if (isDef(data) && isDef((data).__ob__)) {
    "development" !== "production" && warn(
      "Avoid using observed data object as vnode data: " + (JSON.stringify(data)) + "
" +
      "Always create fresh vnode data objects in each render!",
      context
    );
    return createEmptyVNode()
  }
  if (!tag) {
    // in case of component :is set to falsy value
    return createEmptyVNode()
  }
  // support single function children as default scoped slot
  if (Array.isArray(children) &&
      typeof children[0] === "function") {
    data = data || {};
    data.scopedSlots = { default: children[0] };
    children.length = 0;
  }
  if (normalizationType === ALWAYS_NORMALIZE) {
    children = normalizeChildren(children);
  } else if (normalizationType === SIMPLE_NORMALIZE) {
    children = simpleNormalizeChildren(children);
  }
  var vnode, ns;
  if (typeof tag === "string") {
    var Ctor;
    ns = config.getTagNamespace(tag);
    if (config.isReservedTag(tag)) {
      // platform built-in elements
      vnode = new VNode(
        config.parsePlatformTagName(tag), data, children,
        undefined, undefined, context
      );
    } else if (isDef(Ctor = resolveAsset(context.$options, "components", tag))) {假如是組件則從上下文中取出組件的構(gòu)造相關(guān)參數(shù)
      // component
      vnode = createComponent(Ctor, data, context, children, tag);
    } else {
      // unknown or unlisted namespaced elements
      // check at runtime because it may get assigned a namespace when its
      // parent normalizes children
      vnode = new VNode(
        tag, data, children,
        undefined, undefined, context
      );
    }
  } else {
    // direct component options / constructor
    vnode = createComponent(tag, data, context, children);
  }
  if (vnode !== undefined) {
    if (ns) { applyNS(vnode, ns); }
    return vnode
  } else {
    return createEmptyVNode()
  }
}

這里其實(shí)我們不難看出vue在構(gòu)造虛擬dom時(shí)。遞歸的去調(diào)用createElement去生成虛擬dom樹。當(dāng)children是組件或者是普通元素時(shí)。做不同的處理。這里我們關(guān)注的是。當(dāng)子元素是組建時(shí)。這里調(diào)用了
vnode = createComponent(tag, data, context, children);
細(xì)心的人可以在去看看這個(gè)函數(shù)做了什么。簡(jiǎn)單來說這個(gè)函數(shù)將組件的構(gòu)造參數(shù)取出來,放置在元素的componentOptions上。供后續(xù)創(chuàng)建真實(shí)dom時(shí)。標(biāo)記該元素是組件。遞歸初始化。
跳過這些沉重的。我們直接看看我們的這個(gè)html生成的最終的虛擬dom長(zhǎng)什么樣。如下:

我們?cè)趤砜纯次覀兊膍y-component組件長(zhǎng)什么樣子:

componentOptios上存著初始化組件需要的參數(shù)。

構(gòu)建好虛擬dom后。vue進(jìn)入update階段:
這個(gè)階段vue會(huì)判斷先前有無該元素。是否為第一次渲染。假如是第一次。那么直接創(chuàng)建。如果不是有先前的ovnode,則比較差異。最小化更新??纯淳唧w函數(shù):

nction lifecycleMixin (Vue) {
  Vue.prototype._update = function (vnode, hydrating) {
    var vm = this;
    if (vm._isMounted) {
      callHook(vm, "beforeUpdate");
    }
    var prevEl = vm.$el;
    var prevVnode = vm._vnode;//取出緩存的久的虛擬dom
    var prevActiveInstance = activeInstance;
    activeInstance = vm;
    vm._vnode = vnode;//緩存當(dāng)前vnode,供下次更新使用
    // Vue.prototype.__patch__ is injected in entry points
    // based on the rendering backend used.
    if (!prevVnode) {//假如第一次渲染。直接創(chuàng)建
      // initial render
      vm.$el = vm.__patch__(
        vm.$el, vnode, hydrating, false /* removeOnly */,
        vm.$options._parentElm,
        vm.$options._refElm
      );
    } else {
      // updates
      vm.$el = vm.__patch__(prevVnode, vnode);//更新時(shí)。會(huì)比較差異
    }
    activeInstance = prevActiveInstance;
    // update __vue__ reference
    if (prevEl) {
      prevEl.__vue__ = null;
    }
    if (vm.$el) {
      vm.$el.__vue__ = vm;
    }
    // if parent is an HOC, update its $el as well
    if (vm.$vnode && vm.$parent && vm.$vnode === vm.$parent._vnode) {
      vm.$parent.$el = vm.$el;
    }
    // updated hook is called by the scheduler to ensure that children are
    // updated in a parent"s updated hook.
  };

__patch__函數(shù)我們就不細(xì)看了。算了看一下:

  return function patch (oldVnode, vnode, hydrating, removeOnly, parentElm, refElm) {
    if (isUndef(vnode)) {
      if (isDef(oldVnode)) { invokeDestroyHook(oldVnode); }
      return
    }

    var isInitialPatch = false;
    var insertedVnodeQueue = [];

    if (isUndef(oldVnode)) {
      // empty mount (likely as component), create new root element
      isInitialPatch = true;//假如第一次渲染。直接創(chuàng)建
      createElm(vnode, insertedVnodeQueue, parentElm, refElm);
    } else {
      var isRealElement = isDef(oldVnode.nodeType);
      if (!isRealElement && sameVnode(oldVnode, vnode)) {//假如更新并且前后虛擬dom相似,這里相似有自己的一個(gè)算法。比如tag,key必需一致。才會(huì)去diff比較
        // patch existing root node
        patchVnode(oldVnode, vnode, insertedVnodeQueue, removeOnly);
      } else {
        if (isRealElement) {
          // mounting to a real element
          // check if this is server-rendered content and if we can perform
          // a successful hydration.
          if (oldVnode.nodeType === 1 && oldVnode.hasAttribute(SSR_ATTR)) {
            oldVnode.removeAttribute(SSR_ATTR);
            hydrating = true;
          }
          if (isTrue(hydrating)) {
            if (hydrate(oldVnode, vnode, insertedVnodeQueue)) {
              invokeInsertHook(vnode, insertedVnodeQueue, true);
              return oldVnode
            } else {
              warn(
                "The client-side rendered virtual DOM tree is not matching " +
                "server-rendered content. This is likely caused by incorrect " +
                "HTML markup, for example nesting block-level elements inside " +
                "

, or missing . Bailing hydration and performing " + "full client-side render." ); } } // either not server-rendered, or hydration failed. // create an empty node and replace it oldVnode = emptyNodeAt(oldVnode); } // replacing existing element var oldElm = oldVnode.elm; var parentElm$1 = nodeOps.parentNode(oldElm); createElm( vnode, insertedVnodeQueue, // extremely rare edge case: do not insert if old element is in a // leaving transition. Only happens when combining transition + // keep-alive + HOCs. (#4590) oldElm._leaveCb ? null : parentElm$1, nodeOps.nextSibling(oldElm) ); if (isDef(vnode.parent)) { // component root element replaced. // update parent placeholder node element, recursively var ancestor = vnode.parent; while (ancestor) { ancestor.elm = vnode.elm; ancestor = ancestor.parent; } if (isPatchable(vnode)) { for (var i = 0; i < cbs.create.length; ++i) { cbs.create[i](emptyNode, vnode.parent); } } } if (isDef(parentElm$1)) { removeVnodes(parentElm$1, [oldVnode], 0, 0); } else if (isDef(oldVnode.tag)) { invokeDestroyHook(oldVnode); } } } invokeInsertHook(vnode, insertedVnodeQueue, isInitialPatch); return vnode.elm }

patch方法中核心的是createElm:看懂這個(gè)函數(shù)非常重要代碼如下

function createElm (vnode, insertedVnodeQueue, parentElm, refElm, nested) {
    vnode.isRootInsert = !nested; // for transition enter check
    if (createComponent(vnode, insertedVnodeQueue, parentElm, refElm)) {//根據(jù)之前保存的componentoptions來識(shí)別是否為組件。若是。則進(jìn)這個(gè)邏輯
      return
    }

    var data = vnode.data;
    var children = vnode.children;
    var tag = vnode.tag;
    if (isDef(tag)) {
      {
        if (data && data.pre) {
          inPre++;
        }
        if (
          !inPre &&
          !vnode.ns &&
          !(config.ignoredElements.length && config.ignoredElements.indexOf(tag) > -1) &&
          config.isUnknownElement(tag)
        ) {
          warn(
            "Unknown custom element: <" + tag + "> - did you " +
            "register the component correctly? For recursive components, " +
            "make sure to provide the "name" option.",
            vnode.context
          );
        }
      }
      vnode.elm = vnode.ns
        ? nodeOps.createElementNS(vnode.ns, tag)
        : nodeOps.createElement(tag, vnode);
      setScope(vnode);

      /* istanbul ignore if */
      {
        createChildren(vnode, children, insertedVnodeQueue);
        if (isDef(data)) {
          invokeCreateHooks(vnode, insertedVnodeQueue);
        }
        insert(parentElm, vnode.elm, refElm);
      }

      if ("development" !== "production" && data && data.pre) {
        inPre--;
      }
    } else if (isTrue(vnode.isComment)) {
      vnode.elm = nodeOps.createComment(vnode.text);
      insert(parentElm, vnode.elm, refElm);
    } else {
      vnode.elm = nodeOps.createTextNode(vnode.text);
      insert(parentElm, vnode.elm, refElm);
    }
  }

我們這邊還是先關(guān)注自己的組件部分。當(dāng)children是組件元素時(shí),很顯然調(diào)用了createComponent(vnode, insertedVnodeQueue, parentElm, refElm);

var componentVNodeHooks = {
  init: function init (
    vnode,
    hydrating,
    parentElm,
    refElm
  ) {
    if (!vnode.componentInstance || vnode.componentInstance._isDestroyed) {
      var child = vnode.componentInstance = createComponentInstanceForVnode(
        vnode,
        activeInstance,
        parentElm,//調(diào)用了組件內(nèi)部的_init方法遞歸創(chuàng)建子組件。正式進(jìn)入子組件的生命周期
        refElm
      );
      child.$mount(hydrating ? vnode.elm : undefined, hydrating);觸發(fā)子組件的掛載。出發(fā)子組件的編譯和render。又重新來一遍/直到子組件完全渲染好。再開始creelem下一個(gè)child
    } else if (vnode.data.keepAlive) {
      // kept-alive components, treat as a patch
      var mountedNode = vnode; // work around flow
      componentVNodeHooks.prepatch(mountedNode, mountedNode);
    }
  },

這里就是遞歸創(chuàng)建子組件的核心部分.

總結(jié): 第一次寫這個(gè)vue。失敗了。切模塊切的不夠細(xì)。組件機(jī)制感覺用了好多東西。這個(gè)面太大了。自己講的時(shí)候也不知道該細(xì)講還是。。。

總的來說:vue在創(chuàng)建虛擬dom的時(shí)候,如果元素是組件。則準(zhǔn)備好組件的構(gòu)造參數(shù)。包括模版和數(shù)據(jù)等等。組件中的元素如slot,和child放在組件元素的children下。供后面的內(nèi)容分發(fā)用組件中的元素也是在父組件的作用域內(nèi)編譯的??础猒render()函數(shù)就知道。然后在vue需要將虛擬dom變?yōu)檎鎸?shí)dom時(shí)。遇到組件元素時(shí)。開始遞歸初始化。直到把組件compile,render構(gòu)建完后。開始構(gòu)建下一個(gè)元素。最后添加到真實(shí)id=app上。并且把舊的刪了。哈哈。隨便寫了

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/87027.html

相關(guān)文章

  • Vue 中 extend / component / mixins / extends 的區(qū)別

    摘要:父組件在前,子組件在后。,,首先會(huì)在子組件里查找,如果沒有,才會(huì)沿著原型鏈向上,找父組件中對(duì)應(yīng)的屬性。 在segmentfault上看到了一個(gè)問題vue.extend, vue.component 區(qū)別,突然有些方,好歹也寫了幾個(gè)小vue項(xiàng)目,自己都弄不清楚這些東西... new Vue()、component 首先我們來約定一個(gè)選項(xiàng)對(duì)象 baseOptions,后面的代碼會(huì)用到. ...

    Kylin_Mountain 評(píng)論0 收藏0
  • Vue源碼解讀(1)--src/core/index.js 入口文件

    摘要:生產(chǎn)版本設(shè)為可以啟用檢查。只適用于開發(fā)模式和支持的瀏覽器上指定組件的渲染和觀察期間未捕獲錯(cuò)誤的處理函數(shù)為的運(yùn)行時(shí)警告賦予一個(gè)自定義處理函數(shù)。注意這只會(huì)在開發(fā)者環(huán)境下生效,在生產(chǎn)環(huán)境下它會(huì)被忽略。 Vue源碼主入口:src/core/index.js import Vue from ./instance/index // 引用Vue構(gòu)造器 import { initGlobalAPI }...

    tangr206 評(píng)論0 收藏0
  • (解析)vue源碼解讀

    摘要:前言為什么要做源碼解讀我們新到一個(gè)環(huán)境,第一件事情就是熟悉環(huán)境熟悉項(xiàng)目,這個(gè)很考驗(yàn)閱讀源碼的能力以及耐心。構(gòu)造函數(shù)拉到最下面,導(dǎo)出的是一個(gè)構(gòu)造函數(shù)。 前言 A: 為什么要做源碼解讀?Q: 我們新到一個(gè)環(huán)境,第一件事情就是熟悉環(huán)境熟悉項(xiàng)目,這個(gè)很考驗(yàn)閱讀源碼的能力以及耐心。vue是個(gè)很好的庫,知名度高,對(duì)js的學(xué)習(xí)具有向上性,所以搞清楚邏輯是有好處的。A: 閱讀源碼的程度?Q: 我們完全...

    mingde 評(píng)論0 收藏0
  • 來一打前端博客壓壓驚

    前言 本文所有內(nèi)容全部發(fā)布再個(gè)人博客主頁 https://github.com/muwoo/blogs歡迎訂閱。不過最近因?yàn)槭虑楸容^多,有一段時(shí)間沒有更新了,后面打算繼續(xù)不斷學(xué)習(xí)更新,歡迎小伙伴一起溝通交流~ 最近更新 前端單測(cè)的那些事 基于virtual dom 的canvas渲染 js Event loop 機(jī)制簡(jiǎn)介 axios 核心源碼實(shí)現(xiàn)原理 JS 數(shù)據(jù)類型、賦值、深拷貝和淺拷貝 j...

    wangbinke 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<