第十八节:带你梳理Vue2: Vue组件中的注意事项和特例


1. Vue组件名推荐使用驼峰命名

现在我们来看看为什么在Vue中推荐注册组件时使用驼峰写法, 在了解这个之前,相信大家应该都能明白为什么在Vue中, 局部组件的使用频率高于全局组件.

推荐使用驼峰写法也是和局部组件有关系

我们先看一个示例

<div id="app"><!-- 3. 在注册了局部组件的实例中使用局部组件 --><my-component></my-component>
</div><script>// 1. 创建局部组件的选项对象let MyComponent = {template: `<div><h2>局部组件</h2></div>`,}const vm = new Vue({el:"#app",// 2. 将选项对象注册为局部组件components: {"my-component": MyComponent}})
</script>

通过前面的学习,这个例子应该已经熟悉了,

  1. 首先定义一个选项对象myComponent
  2. 在Vue实例中通过components注册为局部组件
  3. 在HTML模板中通过<my-component>自定义标签使用组件

示例中,不使用驼峰命名组件,依然可以正常运行,那么为什么组件名还要推荐使用驼峰命名


那先看下面几点:
  1. 首先在定义选项对象的时候使用的变量名绝对不可以是连字符, 这是标识符命名规范,因此如果要使用连字符就必须加引号

  2. 在注册组件时,components中的属性名my-component就是我们的组件名称, 其值MyComponent就是将谁注册为组件


好此时大家想一想,如果组件名和选项对象的变量名一样会怎么样

是不是就会变成如下的写法

const vm = new Vue({el:"#app",components: {MyComponent: MyComponent}
})

在思考一下,我们之前在学习ES6的时候讲过在定义对象的是有一种简便写法. 当属性跟值长得一样时,就可以简写
因此这里我们就可以简写为

const vm = new Vue({el:"#app",components: {MyComponent}
})

所以,为什么推荐使用驼峰写法,这里就应该可以看出端倪了, 如果我们定义组件的组件名使用驼峰写法,也就是和需要被注册为组件的选项对象一致时,我们注册组件将变得简单

如果此时需要注册n个组件,就可以如下写法

const vm = new Vue({el:"#app",components: {one,two,three,four.....}
})

是不是感觉很优雅,


总结:

  1. 在定义组件时推荐使用驼峰写法,最好组件名和需要被创建为组件的选项对象名一致
  2. 使用组件时推荐使用连字符

2. 组件中的template 选项

尽管语法糖简化了组件注册,但在template选项中拼接HTML元素比较麻烦,这也导致了HTML和JavaScript的高耦合性。

Vue.js提供了两种方式将定义在JavaScript中的HTML模板分离出来。


2.1 使用script标签将template模板分离出来

在使用script标签将template模板分离出来时,要注意script标签的type类型选择,

<div id="app"><!-- 3. 使用组件 --><my-component></my-component>
</div><!-- 组件模板 -->
<!-- 注意: 如果不添加type属性,可能会显示效果,但是会报错, -->
<script id="myComponent" type="text/x-template"><div><h2>我想被创建为局部组件</h2></div>
</script><script>// 1. 创建组件选项对象let MyComponent = {// 此时的模板template的值就是一个选择器template: "#myComponent",}const vm = new Vue({el:"#app",// 2. 注册组件components: {"my-component": MyComponent}})</script>

template选项现在不再是HTML元素,而是一个id,Vue.js根据这个id查找对应的元素,然后将这个元素内的HTML作为模板进行编译。

注意:

使用<script> 标签时,type指定为text/x-template,意在告诉浏览器这不是一段js脚本,浏览器在解析HTML文档时会忽略<script>标签内定义的内容。


2.2 使用template 标签处理模板

如果使用<template>标签,则不需要指定type属性。

<div id="app"><!-- 3. 使用组件 --><my-component></my-component>
</div><!-- 组件模板 -->
<!-- template标签不需要指定type,标签的本意就是告诉浏览器这是模板 -->
<template id="myComponent"><div><h2>我想被创建为局部组件</h2></div>
</template><script>// 1. 创建组件选项对象let MyComponent = {// 此时的模板template的值就是一个选择器template: "#myComponent",}const vm = new Vue({el:"#app",// 2. 注册组件components: {"my-component": MyComponent}})</script>

在理解了组件的创建和注册过程后,我建议使用<script><template>标签来定义组件的HTML模板。

这使得HTML代码和JavaScript代码是分离的,便于阅读和维护。

另外,在Vue.js中,可创建.vue后缀的文件,.vue文件就是一个组件,称为单文件组件,这个内容我会在后面的文章介绍。


3. 组件选项对象中的特例

组件中的选项基本与实例选项对象一致, 但是有两个选项是特例,分别为el 和 data 属性

这个说的特例是指组件的选项对象和实例选项对象使用的不同


3.1 在组件中不能使用el

el属性的作用我们都了解了, 就是在实例中使用,决定Vue需要接管的DOM元素. 组件是在实例中使用,所以不需要el属性,
如果一个选项对象被注册为组件,添加el属性就会造成报错,

// 1. 创建局部组件的选项对象
let MyComponent = {el:"",template: "#myComponent",}const vm = new Vue({el:"#app",// 2. 将选项对象注册为局部组件components: {"my-component": MyComponent}
})// 会报错, 告诉你el属性只能在实例中使用

组件使用el选项报错.png

可以通过报错信息了解到, el选项只能使用在new创建的vue实例上


3.2 组件中data属性值必须是一个函数,

组件中的data选项必须是一个函数,返回一个数据对象.

为什么需要是一个函数,而不可以像Vue实例的选项对象一样是一个对象呢


3.2.1 组件data属性值如果是一个对象的问题

不能使用对象的原因:

  1. 首先因为组件会被多次复用,
  2. 而对象是引用数据类型,如果组件数据使用对象的话,那么组件所有的复用都共享这些数据,

这样就会出现问题,看代码:

示例代码如下:

<div id="app"><!-- 3. 使用组件 --><my-component></my-component><my-component></my-component><my-component></my-component>
</div><template id="myComponent"><div><h2>组件{{num}}</h2><button @click="handleClick">点击+1</button></div>
</template><script>// 组件共享数据let data = {num : 10}// 1. 选项对象let MyComponent = {template: "#myComponent",data: function(){return data},methods: {handleClick(){this.num++}}}const vm = new Vue({el:"#app",// 2. 注册组件components: {"my-component": MyComponent}})
</script>

示例结果:

组件data数据.png

实例说明:

  1. 如果在组件的data中直接写对象就会报错,
  2. 因此我们将data函数中返回的对象提取出来.这样所有的组件都共用一个数据对象.
  3. 一次来模拟组件data选项值为对象的情景

通过案例我们就会发现如果所有的组件都共享数据,当有一个组件中的数据发生了变化,所有的组件显示的数据都会发生变化, 这不是我们想要的,


3.2.2 解决组件共享数据的问题

所以组件的data数据属性才会需要函数,每次初始化化的时候都会执行函数,将返回的结果作为组件的数据,
这样每次执行函数都会给每个组件创建一个独立的数据

<div id="app"><!-- 3. 使用组件 --><my-component></my-component><my-component></my-component><my-component></my-component>
</div><template id="myComponent"><div><h2>组件{{num}}</h2><button @click="handleClick">点击+1</button></div>
</template><script>// 1. 选项对象let MyComponent = {template: "#myComponent",data: function(){// 每次函数执行都会返回新的数据,非共享数据return {num : 10}},methods: {handleClick(){this.num++}}}const vm = new Vue({el:"#app",// 2. 注册组件components: {"my-component": MyComponent}})</script>

显示结果:

组件data数据方法.png

通过示例就会发现, 当一个组件数据发生改变的时候, 其他组件的数据不会变化,因为每个组件都有自己独立的数据


3.3 组件的命名

当注册组件 (或者 prop) 时,可以使用 kebab-case (短横线分隔命名)、camelCase (驼峰式命名) 或 PascalCase (单词首字母大写命名)。

// 在组件定义中
components: {// 使用 kebab-case 注册'kebab-cased-component': { /* ... */ },// 使用 camelCase 注册 俗称小驼峰'camelCasedComponent': { /* ... */ },// 使用 PascalCase 注册 俗称大驼峰'PascalCasedComponent': { /* ... */ }
}

在 HTML 模板中,请使用 kebab-case:

<!-- 在 HTML 模板中始终使用 kebab-case -->
<kebab-cased-component></kebab-cased-component>
<camel-cased-component></camel-cased-component>
<pascal-cased-component></pascal-cased-component>

这个问题已经在前面认真探讨过了,这里不再详细阐述


4. 组件使用问题

通过上面上的例子我们已经了解了组件的使用,就是把组件名当做自定义标签使用,但是这种使用方法有的时候也会出现问题,


4.1. 组件标签解析错误

通过下面的实例了解组件标签解析时发生的问题.

例如:实例代码如下

<div id="app"><!-- 3. 使用组件 --><table><tbody><row></row><row></row><row></row></tbody></table>
</div><!-- 组件模板 -->
<template id="myComponent"><tr><td>内容</td><td>123</td></tr>
</template><script>// 1. 组件选项对象let MyComponent = {template: "#myComponent",}// 2. 注册组件const vm = new Vue({el:"#app",components: {"row": MyComponent}})</script>

显示结果:

template模板编译问题.png

实例中的写法,在查看代码结构时,发现子组件的tr标签并不在tbody里,

原因在与浏览器规范中tbody标签里面必须放tr标签,但是我们放的是row自定义标签,所以在解析的时候就会出问题


诸如此类的还有ul,ol标签里只能放li标签, select标签中只能放option,这些都是需要注意的事项

那么我们怎么解决这类问题呢?


4.2 通过is属性使用组件

vue允许我们使用is属性来使用组件, is属性的值是组件名

所以可以采用is属性来制定组件

示例代码如下:

<div id="app"><!-- 3. 使用组件 --><table><tbody><tr is="row"></tr><tr is="row"/><tr is="row"/></tbody></table>
</div>

显示结果:

通过is属性使用组件.png

此时我们就会发现不仅结果没问题, 编译后标签的嵌套也没有什么问题,

因为我们是按照标准在tbody标签中使用的是tr标签,只不过通过is属性将tr标签替换为了组件row的模板标签.

此时tr标签中没有嵌套内容,所有使用单标签,双标签都可以

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/17802.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2、PHP 8.1.0-dev 后门远程命令执行漏洞复现

1、青少年ctf&#xff0c;题目PHP后门 2、页面 3、bp抓包发现PHP版本为8.1.0-dev 4、尝试使用以前爆出过的漏洞&#xff08;网上查相关案例&#xff09; User-Agentt: zerodiumvar_dump(5*5); User-Agentt: zerodiumsystem("cat /flag"); 5、查找flag User-Agentt: z…

双向带头链表实现

目录 一. 逻辑结构图解 1. 节点中存储的值 2.逻辑实现 二. 各种功能实现 1. 创建节点函数 2. 初始化哨兵位 3. 尾插 4. 头插 5. 尾删 6. 头删 7. 打印链表值 8. 查找数据&#xff0c;返回节点地址 9. 指定地址后插入节点 10. 删除指定地址节点 11. 销毁链表 三.…

JAVA云HIS医院系统源码 云HIS运维平台源码 融合B/S版电子病历系统,支持电子病历四级,saas模式

JAVA云HIS医院系统源码 云HIS运维平台源码 融合B/S版电子病历系统&#xff0c;支持电子病历四级&#xff0c;saas模式 HIS系统就是医院信息管理系统&#xff0c;HIS系统是整个医院信息化的核心&#xff0c;门诊、住院、药房、药库等都是由HIS系统来承载起来的&#xff0c;所以…

【Spring】深入学习AOP编程思想的实现原理和优势

【切面编程】深入学习AOP编程思想的实现原理和优势 前言AOP的由来及AOP与代理的关系AOP的实现方式详解静态代理动态代理 AOP的应用场景记录日志权限控制数据库事务控制缓存处理异常处理监控管理 AOP的底层实现全流程解析Spring AOP的简介动态代理的实现原理Spring AOP的实现原理…

rockeylinux 搭建k8s 1.28.10

1.关闭防火墙 systemctl stop firewalld systemctl disable firewalld 2.关闭selinux # 临时禁用selinux # 将 SELinux 设置为 permissive 模式&#xff08;相当于将其禁用&#xff09; setenforce 0 sed -i s/^SELINUXenforcing$/SELINUXpermissive/ /etc/selinux/config 3.网…

tinyrenderer-渲染器着色

整理了代码&#xff0c;创建了一个相机类&#xff0c;控制镜头 class Camera { public:Camera(Vec3f cameraPos, Vec3f target, Vec3f up):cameraPos_(cameraPos), target_(target), up_(up) {}Matrix getView();Matrix getProjection(); private:Vec3f cameraPos_;Vec3f targ…

2024年区块链,物联网与信息技术国际会议(ICBITIT 2024)

2024年区块链&#xff0c;物联网与信息技术国际会议&#xff08;ICBITIT 2024&#xff09; 2024 International Conference on Blockchain, Internet of Things, and Information Technology 会议简介&#xff1a; 2024年区块链&#xff0c;物联网与信息技术国际会议&#xff…

css样式,点击 箭头方向上下转换

实现效果&#xff1a; 点击切换箭头方向 实现代码 <divclass"modelPart"click"showClick"><div class"modelPart_left"><img:srcaidefalutIconclass"sNodeIcon"><div>{{ selectModel }}</div><div …

Scala的简单认识

Scala编程基础 小白的Scala学习笔记 2024/5/21 上午某一时刻 文章目录 Scala编程基础spark是用Scala开发出来的Scala的优点 打开idea 搜索scala&#xff0c;安装 如果不小心点了取消&#xff0c;或者没有上图的提示&#xff0c;就在依赖里面添加 spark是用Scala开发出来的 类比…

英语学习笔记21+23——Which book?/Which glasses?

Which book?/Which glasses? 哪本书&#xff1f;/哪些杯子&#xff1f; 词汇 Vocabulary give v. 给 搭配&#xff1a;Give me five! 击掌庆祝 用法&#xff1a;give 人 东西     give 东西 to 人    把……东西给某人 例句&#xff1a;把这些苹果给 Bobby.   …

元宇宙vr美术虚拟展馆激发更多文化认同和互鉴

科技引领创新潮流&#xff0c;借助前沿的Web3D技术&#xff0c;我们为用户打造了一个沉浸式的纯3D虚拟空间体验平台&#xff1a;元宇宙线上互动展厅。您只需通过网页即可轻松访问这个充满未来感的互动平台。 在这个独特的虚拟环境中&#xff0c;用户可以轻松创建个性化角色&…

缓存三问与缓存预热-如何预防缓存崩溃

一、缓存三剑客 &#xff08;图片来源&#xff1a;什么是缓存雪崩、击穿、穿透&#xff1f; | 小林coding&#xff09; 缓存穿透 (Cache Penetration) 又称"空缓存"指用户请求的数据在缓存和数据库中都不存在,导致每次请求都去查询数据库,给数据库带来巨大压力。解…

【深度学习】【NLP】词表,分词,嵌入

from transformers import AutoTokenizertokenizer AutoTokenizer.from_pretrained("prajjwal1/bert-tiny") tokenizer.save_pretrained("./bert-tiny/")input_string "Your input string here 我是中文" token_ids tokenizer.encode(input_s…

【PID算法详解】

PID算法 PID算法介绍用途pid数学表达式及其含义P算法D算法I算法 PID总结数学公式转换代码设计实际运用PID代码实现 PID算法介绍 PID控制器是一种广泛应用于工业控制系统的反馈控制器&#xff0c;它通过比例&#xff08;Proportional&#xff09;、积分&#xff08;Integral&am…

快写猪好用吗 #知识分享#笔记#学习方法

快写猪是一个非常好用的论文写作工具&#xff0c;它提供了强大的查重降重功能&#xff0c;帮助用户轻松完成论文写作任务。无论是在学术研究还是日常写作中&#xff0c;快写猪都能提供高效、准确的检测&#xff0c;确保文本的原创性和质量。 首先&#xff0c;快写猪的查重降重功…

c 系统宏有多少

在C语言中&#xff0c;系统宏&#xff08;也称为预定义宏或内置宏&#xff09;的数量并不是固定的&#xff0c;因为它们取决于C标准、编译器以及可能的其他因素。然而&#xff0c;有一些常见的预定义宏是几乎所有C编译器都支持的。 以下是一些常见的C预定义宏&#xff1a; __…

利用预测大模型完成办公室饮水机剩余热水量

背景 在每天上班的时候&#xff0c;很多同事都有喝热水的习惯&#xff0c;但是饮水机内的热水量总是比较少的&#xff0c;如何避免等待&#xff0c;高效的接到热水是我接下来要做的事情的动机。 理论基础 在大量真实数据的情况下&#xff0c;可以分析出用水紧张的时间段和用水…

【css3】01-css3新特性样式篇

目录 1 背景 1.1 设置背景图片的定位 1.2 背景裁切-规定背景的绘制区域 1.3 设置背景图片尺寸 2 边框 2.1 盒子阴影box-shadow 2.2 边框图片border-image 3 文本 -文字阴影text-shadow 1 背景 1.1 设置背景图片的定位 background-origin&#xff1a;规定背景图片的定位…

科技守护,河流水文监测保障水资源安全!

中小河流是城乡水资源的补给&#xff0c;又是不可或缺的排放渠道&#xff0c;维系着城乡水资源的平衡与生态的健康。然而&#xff0c;随着工业化、城市化的快速推进&#xff0c;河流生态环境面临着越来越大的压力。为了有效保护和合理利用河流资源&#xff0c;河流水文监测成为…

2024年新算法-红嘴蓝鹊优化器(RBMO)优化BP神经网络回归预测

2024年新算法-红嘴蓝鹊优化器(RBMO)优化BP神经网络回归预测 亮点&#xff1a; 输出多个评价指标&#xff1a;R2&#xff0c;RMSE&#xff0c;MSE&#xff0c;MAPE和MAE 满足需求&#xff0c;分开运行和对比的都有对应的主函数&#xff1a;main_BP, main_RBMO, main_BPvsBP_R…