vue总结
官网
英文官网:https://vuejs.org/
中文官网:https://cn.vuejs.org/
简介
Vue是一款用于构建用户界面的 JavaScript 框架。它基于标准 HTML、CSS 和 JavaScript 构建,并提供了一套声明式的、组件化的编程模型
单文件组件:Vue 的单文件组件会将一个组件的逻辑 (JavaScript),模板 (HTML) 和样式 (CSS) 封装在同一个文件里。
特点:
- 组件化:
- 响应式数据绑定 :
开发工具
官网推荐的为 Visual Studio Code + Vue - Official 扩展
创建Vue项目
Vue 的官方工具链(如 create-vue、Vue CLI、Vite)均依赖 Node.js 环境运行,所以本地要安装Node.js
在打算创建项目的目录
npm create vue@latest
它是 Vue 官方的项目脚手架工具
如果不确定是否要开启某个功能,你可以直接按下回车键选择 No。在项目被创建后,通过以下步骤安装依赖并启动开发服务器:
cd <your-project-name>
npm install
npm run dev
语法
Vue 使用一种基于 HTML 的模板语法,允许在 HTML 中嵌入动态数据和逻辑。
插值表达式:{{ }}
<template>
<div>
<p>Hello, {{ name }}!</p> <!-- 动态显示 name 的值 -->
</div>
</template>
<script>
export default {
data() {
return { name: 'Vue' };
}
};
</script>
原始 HTML:v-html
双大括号会将数据解释为纯文本,而不是 HTML。若想插入 HTML,你需要使用 v-html 指令:
v-html:用于将数据作为 HTML 代码渲染到元素中,如下p的内容被渲染为rawHtml的html内容
<div id="app">
<p v-html="rawHtml"></p>
</div>
<script>
new Vue({
el: '#app',
data: {
rawHtml: '<span style="color: red">这是红色的文本</span>'
}
})
</script>
页面会显示红色的"这是红色的文本",而不是显示 HTML 标签字符串。
使用 v-html 时要特别注意 XSS(跨站脚本攻击)风险,因为它会直接渲染原始 HTML。
属性绑定:v-bind
双大括号不能在标签属性上使用,需要使用 v-bind 指令
v-bind 或简写 ::动态绑定 HTML 属性
<div id="app">
<!-- 绑定 title 属性 -->
<a v-bind:href="url" v-bind:title="tooltip">Vue 官网</a>
<!-- 使用简写 -->
<a :href="url" :title="tooltip">Vue 官网(简写)</a>
</div>
<script>
new Vue({
el: '#app',
data: {
url: 'https://cn.vuejs.org',
tooltip: '点击访问 Vue.js 中文官网'
}
})
</script>
同名简写,仅支持 3.4 版本及以上
如果 attribute 的名称与绑定的 JavaScript 变量的名称相同,那么可以进一步简化语法,省略 attribute 值:
<!-- 与 :id="id" 相同 -->
<div :id></div>
<!-- 这也同样有效 -->
<div v-bind:id></div>
对于 class 和 style,Vue 做了特殊增强,支持数组、对象等多种格式
<div id="app">
<!-- 对象语法 -->
<div :class="{ active: isActive, 'text-danger': hasError }">
对象语法示例
</div>
<!-- 数组语法 -->
<div :class="[activeClass, errorClass]">
数组语法示例
</div>
<!-- 结合对象和数组 -->
<div :class="[{ active: isActive }, errorClass]">
结合语法示例
</div>
</div>
<script>
new Vue({
el: '#app',
data: {
isActive: true,
hasError: false,
activeClass: 'active-class',
errorClass: 'text-danger'
}
})
</script>
<div id="app">
<!-- 对象语法 -->
<div :style="{ color: activeColor, fontSize: fontSize + 'px' }">
对象语法示例
</div>
<!-- 数组语法 -->
<div :style="[baseStyles, overridingStyles]">
数组语法示例
</div>
</div>
<script>
new Vue({
el: '#app',
data: {
activeColor: 'red',
fontSize: 16,
baseStyles: {
color: 'blue',
fontSize: '13px'
},
overridingStyles: {
fontWeight: 'bold'
}
}
})
</script>
绑定对象(批量绑定)通过不带参数的 v-bind,你可以将指定属性都绑定到单个元素上:
<div id="app">
<!-- 一次性绑定多个属性 -->
<button v-bind="buttonAttrs">按钮</button>
</div>
<script>
new Vue({
el: '#app',
data: {
buttonAttrs: {
id: 'submit-btn',
class: 'btn btn-primary',
disabled: false,
title: '点击提交表单'
}
}
})
</script>
动态绑定属性名,在 Vue 2.6.0+ 中支持
需要包含在一对方括号内
<div id="app">
<div v-bind:[attributeName]="value">动态属性名</div>
<button @click="changeAttribute">切换属性</button>
</div>
<script>
new Vue({
el: '#app',
data: {
attributeName: 'data-status',
value: 'active',
isId: false
},
methods: {
changeAttribute() {
this.attributeName = this.isId ? 'data-status' : 'id';
this.value = this.isId ? 'active' : 'main-content';
this.isId = !this.isId;
}
}
})
</script>
还可以将一个函数绑定到动态的事件名称上
<a v-on:[eventName]="doSomething"> ... </a>
<!-- 简写 -->
<a @[eventName]="doSomething"> ... </a>
使用 JavaScript 表达式
所有的数据绑定中都支持完整的 JavaScript 表达式:
JavaScript 表达式可以被使用在如下场景上:
- 在文本插值中 (双大括号)
- 在任何 Vue 指令 (以 v- 开头的特殊 attribute) attribute 的值中
{{ number + 1 }}
{{ ok ? 'YES' : 'NO' }}
{{ message.split('').reverse().join('') }}
<div :id="`list-${id}`"></div>
调用函数
可以在绑定的表达式中使用一个组件暴露的方法:
<time :title="toTitleDate(date)" :datetime="date">
{{ formatDate(date) }}
</time>
事件绑定:v-on
v-on 或简写 @:监听 DOM 事件并触发方法。
<button @click="handleClick">Click me</button> <!-- 点击触发方法 -->
methods: {
handleClick() {
alert('Button clicked!');
}
}
修饰符
修饰符是以点开头的特殊后缀,表明指令需要以一些特殊的方式被绑定。例如 .prevent 修饰符会告知 v-on 指令对触发的事件调用 event.preventDefault():
<form @submit.prevent="onSubmit">...</form>
条件渲染:v-if / v-show
v-if:根据条件动态添加/移除 DOM 元素。
<p v-if="isVisible">Visible only if isVisible is true</p>
v-show:通过 CSS display 切换显示/隐藏(始终渲染 DOM)。
<p v-show="isVisible">Toggled via CSS</p>
v-if 是“真实的”按条件渲染,因为它确保了在切换时,条件区块内的事件监听器和子组件都会被销毁与重建。
v-if 也是惰性的:如果在初次渲染时条件值为 false,则不会做任何事。条件区块只有当条件首次变为 true 时才被渲染。
相比之下,v-show 简单许多,元素无论初始条件如何,始终会被渲染,只有 CSS display 属性会被切换。
总的来说,v-if 有更高的切换开销,而 v-show 有更高的初始渲染开销。因此,如果需要频繁切换,则使用 v-show 较好;如果在运行时绑定条件很少改变,则 v-if 会更合适。
v-else/v-else-if
可以使用 v-else 为 v-if 添加一个“else 区块”
<button @click="awesome = !awesome">Toggle</button>
<h1 v-if="awesome">Vue is awesome!</h1>
<h1 v-else>Oh no 😢</h1>
v-else-if 提供的是相应于 v-if 的“else if 区块”。它可以连续多次重复使用:
<div v-if="type === 'A'">
A
</div>
<div v-else-if="type === 'B'">
B
</div>
<div v-else-if="type === 'C'">
C
</div>
<div v-else>
Not A/B/C
</div>
<template> 上的 v-if
因为 v-if 是一个指令,他必须依附于某个元素。但如果我们想要切换不止一个元素呢?在这种情况下我们可以在一个 <template> 元素上使用 v-if,这只是一个不可见的包装器元素,最后渲染的结果并不会包含这个 <template> 元素。
<template v-if="ok">
<h1>Title</h1>
<p>Paragraph 1</p>
<p>Paragraph 2</p>
</template>
v-else 和 v-else-if 也可以在 <template> 上使用。
v-show 不支持在 <template> 元素上使用,也不能和 v-else 搭配使用。
列表渲染:v-for
<li v-for="item in items">
{{ item.message }}
</li>
const items = ref([{ message: 'Foo' }, { message: 'Bar' }])
也支持使用可选的第二个参数表示当前项的位置索引。从0开始
<ul>
<li v-for="(item, index) in items" :key="item.id">
{{ index }} - {{ item.name }}
</li>
</ul>
data() {
return {
items: [
{ id: 1, name: 'Apple' },
{ id: 2, name: 'Banana' }
]
};
}
对于多层嵌套的 v-for,作用域的工作方式和函数的作用域很类似。每个 v-for 作用域都可以访问到父级作用域:
<li v-for="item in items">
<span v-for="childItem in item.children">
{{ item.message }} {{ childItem }}
</span>
</li>
你也可以使用 of 作为分隔符来替代 in
<div v-for="item of items"></div>
也可以使用 v-for 来遍历一个对象的所有属性。遍历的顺序会基于对该对象调用 Object.values() 的返回值来决定。
<ul>
<li v-for="value in myObject">
{{ value }}
</li>
</ul>
const myObject = reactive({
title: 'How to do lists in Vue',
author: 'Jane Doe',
publishedAt: '2016-04-10'
})
可以通过提供第二个参数表示属性名 (例如 key):
<li v-for="(value, key) in myObject">
{{ key }}: {{ value }}
</li>
第三个参数表示位置索引:
<li v-for="(value, key, index) in myObject">
{{ index }}. {{ key }}: {{ value }}
</li>
在 v-for 里使用范围值
v-for 可以直接接受一个整数值。在这种用例中,会将该模板基于 1...n 的取值范围重复多次。
<span v-for="n in 10">{{ n }}</span>
<template> 上的 v-for
与模板上的 v-if 类似,你也可以在 <template> 标签上使用 v-for 来渲染一个包含多个元素的块。
<ul>
<template v-for="item in items">
<li>{{ item.msg }}</li>
<li class="divider" role="presentation"></li>
</template>
</ul>
v-for 与 v-if
当它们同时存在于一个节点上时,v-if 比 v-for 的优先级更高。这意味着 v-if 的条件将无法访问到 v-for 作用域内定义的变量别名:
<!--
这会抛出一个错误,因为属性 todo 此时没有在该实例上定义
-->
<li v-for="todo in todos" v-if="!todo.isComplete">
{{ todo.name }}
</li>
在外先包装一层 <template> 再在其上使用 v-for 可以解决这个问题 (这也更加明显易读):
<template v-for="todo in todos">
<li v-if="!todo.isComplete">
{{ todo.name }}
</li>
</template>
v-for通过 key 管理状态
ue 默认按照“就地更新”的策略来更新通过 v-for 渲染的元素列表。当数据项的顺序改变时,Vue 不会随之移动 DOM 元素的顺序,而是就地更新每个元素,确保它们在原本指定的索引位置上渲染。
默认模式是高效的,但只适用于列表渲染输出的结果不依赖子组件状态或者临时 DOM 状态 (例如表单输入值) 的情况。
为了给 Vue 一个提示,以便它可以跟踪每个节点的标识,从而重用和重新排序现有的元素,你需要为每个元素对应的块提供一个唯一的 key attribute:
<div v-for="item in items" :key="item.id">
<!-- 内容 -->
</div>
推荐在任何可行的时候为 v-for 提供一个 key attribute,除非所迭代的 DOM 内容非常简单 (例如:不包含组件或有状态的 DOM 元素),或者你想有意采用默认行为来提高性能。
key 绑定的值期望是一个基础类型的值,例如字符串或 number 类型。不要用对象作为 v-for 的 key。
响应式数据
Vue 通过 数据劫持 和 发布-订阅模式 实现响应式更新,能够自动追踪数据变化并更新相关视图。
声明响应式状态
在组合式 API 中,推荐使用 ref() 函数来声明响应式状态:
import { ref } from 'vue'
const count = ref(0)
ref() 接收参数,并将其包裹在一个带有 .value 属性的 ref 对象中返回
const count = ref(0)
console.log(count) // { value: 0 }
console.log(count.value) // 0
count.value++
console.log(count.value) // 1
要在组件模板中访问 ref,请从组件的 setup() 函数中声明并返回它们:
import { ref } from 'vue'
export default {
// `setup` 是一个特殊的钩子,专门用于组合式 API。
setup() {
const count = ref(0)
// 将 ref 暴露给模板
return {
count
}
}
}
<div>{{ count }}</div>
在模板中使用 ref 时,我们不需要附加 .value。为了方便起见,当在模板中使用时,ref 会自动解包
可以直接在事件监听器中改变一个 ref:
<button @click="count++">
{{ count }}
</button>
可以在同一作用域内声明更改 ref 的函数,并将它们作为方法与状态一起公开:
import { ref } from 'vue'
export default {
setup() {
const count = ref(0)
function increment() {
// 在 JavaScript 中需要 .value
count.value++
}
// 不要忘记同时暴露 increment 函数
return {
count,
increment
}
}
}
暴露的方法可以被用作事件监听器
<button @click="increment">
{{ count }}
</button>
可以通过使用单文件组件 (SFC) 来避免在 setup() 函数中暴露大量的状态和方法
<script setup>
import { ref } from 'vue'
const count = ref(0)
function increment() {
count.value++
}
</script>
<template>
<button @click="increment">
{{ count }}
</button>
</template>
组件
组件是 Vue 实例的封装,允许开发者将 UI 拆分为独立、可复用的部分,每个组件包含自己的模板、逻辑和样式。
可以包含:模板(HTML),数据(data),方法(methods),计算属性(computed),侦听器(watch),生命周期钩子,子组件
组件注册
全局注册:
// 定义组件
Vue.component('my-component', {
template: '<div>A custom component!</div>'
})
// 在根实例中使用
new Vue({ el: '#app' })
局部注册(推荐)
// 定义组件
const ComponentA = {
template: '<div>Component A</div>'
}
// 在父组件中注册
new Vue({
el: '#app',
components: {
'component-a': ComponentA
}
})

浙公网安备 33010602011771号