# createElement

在上一节,我们知道了render函数执行的时候,会调用$createElement或者_c方法,也知道了它们最后其实调用的是同一个createElement方法,只不过最后一个参数有点区别。在这一节,我们来详细分析一下createElement方法的实现逻辑。

createElement是定义在src/core/vdom/create-element.js文件中,其代码如下:

const SIMPLE_NORMALIZE = 1
const ALWAYS_NORMALIZE = 2
export function createElement (
  context: Component,
  tag: any,
  data: any,
  children: any,
  normalizationType: any,
  alwaysNormalize: boolean
): VNode | Array<VNode> {
  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)
}

在分析代码之前,我们来看一下$createElement_c方法最后一个不相同的参数,在createElement中体现在什么地方。我们可以从最后一个参数命名猜测其作用,对于模板编译调用_c时,其alwaysNormalize传递的是false,因为_c只会在内部使用,因此其方法调用的时候参数格式是比较规范的,我们不需要过多的进行normalize。而$createElement是提供给用户使用的,为了让$createElement更加简洁和实用,允许用户传递不同形式的参数来调用$createElement,这也就造成了用户手写的render,我们必须始终进行normalize

在上述分析完毕后,我们就知道了$createElement_c最后一个不相同的参数,体现在什么地方了:调用_c时对children进行简单规范化,调用$createElement时必须始终对children进行规范化。

回到正题,我们发现createElement其实是对_createElement方法的一层包裹,之所以这样做是为了让createElement达到一种类似于函数重载的功能(JavaScript实际并没有这个概念)。其中第三个参数data是可以不传的。

// 不传递data 
createElement(this, 'div', 'Hello, Vue', 1, false)
// 传递data
createElement(this, 'div', undefined, 'Hello, Vue', 1, false)

当不传递data的时候,我们需要把第三、第四个参数往后移动一个位置,然后把data赋值为undefined,最后在把处理好的参数传递给_createElement。接下来,我们先看一下_createElement方法几个参数的具体作用:

  • contextVNode当前上下环境。
  • tag:标签,可以是正常的HTML元素标签,也可以是Component组件。
  • dataVNode的数据,其类型为VNodeData,可以在根目录flow/vnode.js文件中看到其具体定义。
  • childrenVNode的子节点。
  • normalizationTypechildren子节点规范化类型。

其具体实现代码如下:

export function _createElement (
  context: Component,
  tag?: string | Class<Component> | Function | Object,
  data?: VNodeData,
  children?: any,
  normalizationType?: number
): VNode | Array<VNode> {
  // ...省略代码
  if (!tag) {
    // in case of component :is set to falsy value
    return createEmptyVNode()
  }
  if (normalizationType === ALWAYS_NORMALIZE) {
    children = normalizeChildren(children)
  } else if (normalizationType === SIMPLE_NORMALIZE) {
    children = simpleNormalizeChildren(children)
  }
  let vnode, ns
  if (typeof tag === 'string') {
    let Ctor
    ns = (context.$vnode && context.$vnode.ns) || config.getTagNamespace(tag)
    if (config.isReservedTag(tag)) {
      // platform built-in elements
      if (process.env.NODE_ENV !== 'production' && isDef(data) && isDef(data.nativeOn)) {
        warn(
          `The .native modifier for v-on is only valid on components but it was used on <${tag}>.`,
          context
        )
      }
      vnode = new VNode(
        config.parsePlatformTagName(tag), data, children,
        undefined, undefined, context
      )
    } else if ((!data || !data.pre) && isDef(Ctor = resolveAsset(context.$options, 'components', tag))) {
      // 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)
  }
  // ...省略代码
}

_createElement的代码看起来有点多,但它主要做两件事情:规范化子节点和创建VNode节点,接下来我们围绕这两个方面来详细介绍。

  • 规范化子节点:因为虚拟DOM是一个树形结构,每一个节点都应该是VNode类型,但是children参数又是任意类型的,所以如果有子节点,我们需要把它进行规范化成VNode类型,如果没有子节点,那么children就是undefined。至于如何规范化,则是通过normalizationType参数来实现的,其中normalizationType可能的值有三种:undefined表示不进行规范化,1表示简单规范化,2表示始终规范化。我们先来看当值为1的情况,它调用了simpleNormalizeChildren,这个方法和normalizeChildren是定义在同一个地方src/core/vdom/helpers/normalize-children.js文件中,其代码如下:
export function simpleNormalizeChildren (children: any) {
  for (let i = 0; i < children.length; i++) {
    if (Array.isArray(children[i])) {
      return Array.prototype.concat.apply([], children)
    }
  }
  return children
}

simpleNormalizeChildren的作用是把多维数组降低一个维度,例如二维数组降低到一维数组,三维数组降低到二维数组,这样做的目的是为了方便后续遍历children

// 展示使用,实例为VNode
let children = ['VNode', ['VNode', 'VNode'], 'VNode']

// 简单规范化子节点
children = simpleNormalizeChildren(children)

// 规范化后
console.log(children) // ['VNode', 'VNode', 'VNode', 'VNode']

接下来我们来看值为2的情况,它调用了normalizeChildren,其代码如下:

export function normalizeChildren (children: any): ?Array<VNode> {
  return isPrimitive(children)
    ? [createTextVNode(children)]
    : Array.isArray(children)
      ? normalizeArrayChildren(children)
      : undefined
}

normalizeChildren的代码不是很多,也不是很复杂。当children是基础类型值的时候,直接返回一个文本节点的VNode数组,createTextVNode我们在之前已经介绍过了。如果不是,则再判断是否为数组,不是则其children就是undefined,是的话就调用normalizeArrayChildren来规范化。接下来,我们重点分析以下normalizeArrayChildren的实现,它和normalizeChildren是定义在同一个位置,其实现代码如下:

function normalizeArrayChildren (children: any, nestedIndex?: string): Array<VNode> {
  const res = []
  let i, c, lastIndex, last
  for (i = 0; i < children.length; i++) {
    c = children[i]
    if (isUndef(c) || typeof c === 'boolean') continue
    lastIndex = res.length - 1
    last = res[lastIndex]
    //  nested
    if (Array.isArray(c)) {
      if (c.length > 0) {
        c = normalizeArrayChildren(c, `${nestedIndex || ''}_${i}`)
        // merge adjacent text nodes
        if (isTextNode(c[0]) && isTextNode(last)) {
          res[lastIndex] = createTextVNode(last.text + (c[0]: any).text)
          c.shift()
        }
        res.push.apply(res, c)
      }
    } else if (isPrimitive(c)) {
      if (isTextNode(last)) {
        // merge adjacent text nodes
        // this is necessary for SSR hydration because text nodes are
        // essentially merged when rendered to HTML strings
        res[lastIndex] = createTextVNode(last.text + c)
      } else if (c !== '') {
        // convert primitive to vnode
        res.push(createTextVNode(c))
      }
    } else {
      if (isTextNode(c) && isTextNode(last)) {
        // merge adjacent text nodes
        res[lastIndex] = createTextVNode(last.text + c.text)
      } else {
        // default key for nested array children (likely generated by v-for)
        if (isTrue(children._isVList) &&
          isDef(c.tag) &&
          isUndef(c.key) &&
          isDef(nestedIndex)) {
          c.key = `__vlist${nestedIndex}_${i}__`
        }
        res.push(c)
      }
    }
  }
  return res
}

虽然normalizeArrayChildren的代码很多,但做的事情并不复杂,我们只要关注遍历过程中几个重要的逻辑分支即可。

  1. 遍历项为数组:这种情况稍微复杂一点,多见于v-for或者slot的时候,会出现嵌套VNode数组的情况,如果存在嵌套VNode的情况会递归调用normalizeArrayChildren,我们以下面这个例子为例:
<template>
  <div id="app">
    <p>{{msg}}</p>
    <span v-for="(item, index) in list" :key="index">{{item}}</span>
  </div>
</template>
<script>
export default {
  name: 'App',
  data () {
    return {
      msg: 'message',
      list: [1, 2, 3]
    }
  }
}
</script>

App组件render函数执行的时候,其children子节点会出现VNode嵌套数组的情况,可以用以下代码示例说明:

const children = [
  [ { tag: 'p' }, ... ],
  [
    [ { tag: 'span', ... } ],
    [ { tag: 'span', ... } ],
    [ { tag: 'span', ... } ]
  ]
]

递归调用normalizeArrayChildren方法后,嵌套数组被处理成了一维数组,如下:

const children = [
  [ { tag: 'p' }, ... ],
  [ { tag: 'span', ... } ],
  [ { tag: 'span', ... } ],
  [ { tag: 'span', ... } ]
]
  1. 遍历项为基础类型:当为基础类型的时候,调用封装的createTextVNode方法来创建一个文本节点,然后push到结果数组中。
  2. 遍历项已经是VNode类型:这种情况最简单,如果不属于以上两种情况,那么代表本身已经是VNode类型了,这时候我们什么都不需要做,直接push到结果数组中即可。

在这三个逻辑分支中,都判断了isTextNode,这部分的代码主要是用来优化文本节点:如果存在两个连续的文本节点,则将其合并成一个文本节点。

// 合并前
const children = [
  { text: 'Hello ', ... },
  { text: 'Vue.js', ... },
]

// 合并后
const children = [
  { text: 'Hello Vue.js', ... }
]
  • 创建VNode节点:创建VNode节点的逻辑有两大分支,tagstring类型和component类型,其中string类型又存在几个小的逻辑判断分支。在createElement章节,我们重点介绍类型为string的分支。在这个分支中,首先判断tag提供的标签名是不是平台保留标签(htmlsvg标签),如果是则直接创建对应标签的VNode节点,如果不是则尝试在已经全局或者局部注册的组件中去匹配,匹配成功则使用createComponent去创建组件节点,如果没有匹配上则创建一个未知标签的VNode节点,例如:
<template>
  <div id="app">
    <div>{{msg}}</div>
    <hello-world :msg="msg" />
    <cms>12321321</cms>
  </div>
</template>
<script>
import HelloWorld from '@/components/HelloWorld.vue'
export default {
  name: 'App',
  data () {
    return {
      msg: 'message',
    }
  },
  components: {
    HelloWorld
  }
}
</script>

tagcms,但它既不像div一样是平台保留标签,又不像hello-world一样是已经局部注册过的组件,它属于未知的标签。这里之所以直接创建未知标签的VNode而不是报错,这是因为子节点在createElement的过程中,有可能父节点会为其提供一个namespace,真正做未知标签校验的过程发生在path阶段,path的过程我们将在后续进行介绍。

最后更新时间: 2/28/2023, 8:33:37 PM