本教程操作环境:windows7系统、vue3版,dell g3电脑。
vue为什么要求组件模板只能有一个根元素?下面一起来了解一下。
vue 实例首先,我觉得这个问题要先从 vue 的实例开始讲起。vue 的实例一般都是长成下面这个样子,不同的只是 id 名的不同。
<div id="app"></div>
var vm = new vue({ el: '#app', data: {}, methods: {} ...})
这就是 vue 实例的基本结构,并不陌生。从这里可以看到,el 的指定是一个 id 为 app 的 div 元素,vue 实例接管了对它的控制,减少了我们的 dom 操作,需要被 vm 控制的元素全部加在它的内部。如果是需要控制不同的部分,这就需要多个 vue 的实例来实现。疑问就来了,为什么需要不同的 vue 实例来接管?
入口问题在 vue 中指定 el 选项是给 vue 实例指定一个 spa 入口,有可能你的页面会长成像下面这样:
<div id="app"></div><div id="app1"></div><div id="app2"></div>
vue 实例其实并不知道哪一个是入口,它应该接管哪一个部分,所以你要给它指定一个唯一的元素作为入口。每一个入口可以看作是一个 vue 的类,vue 要把这个入口进去的所有东西都取出来进行轮循渲染一遍,再把它重新挂载回页面中的 dom 里面去。打给比方来说,一个 vue 实例只拥有一个钥匙,一个钥匙只能开一把锁,但是页面上有很多把锁,如果你不说清楚它是哪把锁的钥匙,vue 实例就不知道接下来要怎么做了。
当然,这只是一个比较浅显的理解。你可能会说,我指定几个入口让 vue 实例去一个一个试就好了,我们往下看。
虚拟 dom“虚拟 dom”是我们对由 vue 组件树建立起来的整个 vnode 树的称呼
学习 vue 不得不说的就是2.0引入的virtual dom,引入虚拟 dom 后,在框架的内部将虚拟 dom 树型结构与真实 dom 做了映射,让我们不用再命令式的去操作 dom。
引用里面的一张图片:
从这张图可以看出来虚拟 dom 的一个渲染过程,那我们再回到本文的话题:为什么只能由一个根元素?
我们来看一个例子,假设你的 vue 实例接管的 dom 结构长成这个样子:
<div id="app"> <h1 id="h3">my title</h1> <span>content</span> other text <!-- annotation text --></div>
它在浏览器内部的表现是一个这样的 dom 树:
原谅我画图技术差,不过我想展示的效果达到了。从这可以看出它是一个树的结构,每个元素、文字、注释都是一个节点,虚拟 dom 遵循的也是这样的一个树的数据结构。
回到正题,我们的指定的 el 也就是整个 dom 结构的根。现在就很好说了,我们只有指定了唯一的 el 根元素,才能交给 vue 实例在内部通过 createelement 方法生成一个对应的虚拟 dom 结构映射真实的 dom 元素进行操作渲染成真正的 html
换句话来说,可以把 el 对应的元素理解成 vue 接管部分中的一个顶级标签,就像基本的 html 结构中,顶级标签是 <html></html>,只能有一个这样的标签存在。对应到 vue 中也是这样,如果你给它两个顶级标签,那么对应的 dom 结构就无法生成了,这也就解释了之前的疑惑:为什么不能指定多个入口让 vue 实例一个个的试。
不知道我这样的解释有没有说明白这个问题,如果没清楚我们下面再来看看。
vue-cli现在实际的项目开发中,使用脚手架 vue-cli 开发居多,我们来看看。
vue-cli 的形式是单文件组件,一个 .vue 页面的基本结构是这样的:
<template> <div></div></template><script>export default {}</script><style></style>
在这里,<template> 标签下也只能有一个根元素 div,这是为什么?
在说这个话题之前,我们需要了解 h5 新标签 <template> 的一些特性,可以参考文档,它保证了内部的内容有效但不会被渲染。vue-cli 本质上是会把 .vue 文件通过 webpack 配置打包成一系列的 js/css 文件注入到一个 html 文件中交给浏览器进行解释执行,我们看一个打包好的文件目录:
这也就是说,每个 .vue 文件都会是一个 vue 的实例,而 <template> 标签中的内容就是 vue 实例接管形成虚拟 dom 的那部分内容。如果在 template 下有多个 div,那么虚拟 dom 树就没办法生成了。
问题抽象并回答其实这个问题归结到最后,也可以抽象为一个问题:为什么抽象出来的 dom 树只能有一个根?
从查找和遍历的角度来说,如果有多个根,那么我们的查找和遍历的效率会很低。
如果一个树有多个根,说明可以优化,肯定会有一个节点是可以访问到所有的节点,那这个节点就会成为新的根节点。
再从 vue 本身来说,如果说一个组件有多个入口多个根,那不就意味着你的组件还可以进一步拆分成多个组件,进一步组件化,降低代码之间的耦合程度。
【相关视频教程推荐:vue视频教程、web前端入门】
以上就是vue为什么只有一个根标签的详细内容。
