深入理解 Vue.js(一)
重读文档,同时也是对自己的积累的又一次推倒重建。
什么是 Vue.js
Vue是一个渐进式前端框架,其核心概念是数据的双向绑定、组件化、单向数据流、可复用等。既然是重新阅读官方文档,那我们从一个工程化项目开始。
Vue 工程化项目
1 | npm install -g @vue/cli |
这时,我们得到了一个这样的文件结构:
首先我们来关注 src/main.js:
1 | import Vue from 'vue' |
可见,这里引入了 vue 与 App.vue 文件,同时 new 了一个 Vue 实例并做了节点渲染。
现在我们先封装一个组件 ComponentItem 并抛出。在 component 文件夹下新建 ComponentItem.vue 文件:
1 | <template> |
一个 vue 文件由模板 template、脚本 script 和样式 style三部分构成。
进入 App.vue 在 script 中导入(import) components 中被抛出的组件 ComponentItem 并在 components 中声明使用,最终在 template 中渲染到浏览器显示。
1 | <template> |
注:这里删除掉自动生成的 HelloWorld.vue 文件以及引入的相关多余内容。
这时我们可以在浏览器看到子组件便被渲染到了页面上。
数据与方法
熟悉了基本的组件创建,我们先回顾一下
Vue的数据与方法。
Vue 的初始数据声明在 data 中。当一个实例被创建时,它将 data 对象(组件中为data函数)中的所有属性(property)加入到 Vue 的响应式系统中,通过属性值的变化,自动绑定到视图,称为数据的双向绑定。
注意:在组件中,data 是一个函数而不是对象,这样每个实例便可以维护一份被返回对象的独立的拷贝。
比如在我们新建的 ComponentItem 组件中,我们这样使用 data:
1 | export default { |
注意:只有当实例被创建时就已经存在于 data 中的属性(property)才是 响应式 的。所以如果我们在后续会需要一些属性,但是一开始它为空或不存在,那么需要在 data 中设置一些初始值。
Vue 以自身的 diff 算法遍历计算 Virtual DOM,找到最小差异 DOM 更新,避免了真实的 DOM 渲染引起的整个 DOM 树的重排重绘,减小浏览器消耗。
Vue 自定义实例属性带有 $ 前缀,与用户定义的 property 进行区分,如:
1 | var data = { |
更多 API 查找参照 官网文档
常用的如:vm.$data、vm.$el、vm.$props、vm.$options、vm.$refs、vm.$watch、vm.$set、vm.$on、vm.$emit、vm.$attrs 等。
钩子函数与生命周期
每个 Vue 实例在被创建时都要经过一系列的初始化过程,如设置数据监听、编译模板、挂载实例到 DOM、数据变化更新 DOM 等。在整个过程中,会运行一些生命周期钩子的函数,允许用户在不同阶段添加自己的代码进行处理。
如:
1 | created() {}, |
注意:不要在选项 property 或回调上使用箭头函数,因为箭头函数并没有 this,this 会作为变量一直向上级词法作用域查找,直至找到为止,会导致出现属性未定义、方法不存在的错误。
模板语法
在
Vue.js的使用上,框架提供了基于HTML的模板语法和渲染函数(render) & JSX。
模板语法 允许开发者声明式地将 DOM 绑定至底层 Vue 实例的数据。然后 Vue 将模板编译成虚拟 DOM 渲染函数。结合响应系统,Vue 能够智能地计算出最少需要重新渲染多少组件,并把 DOM 操作次数减到最少。
如果熟悉虚拟 DOM 并偏爱原生的 JavaScript,可使用 渲染函数(render)+JSX语法。
这里介绍模板语法的使用方法。
插值
文本插值
数据绑定最常见的形式就是使用双大括号进行文本插值:
1 | <template> |
原始 HTML
双大括号会将数据解释为普通文本,而非 HTML 代码。使用 v-html 可以输出 HTML:
1 | <template> |
注意:动态渲染的任意 HTML 可能会非常危险,因为它很容易导致 XSS 攻击。只对可信内容使用 HTML 插值,绝不要对用户提供的内容使用插值。
Attribute
使用 v-bind 指令动态绑定到 HTML attribute 上:
1 | <template> |
使用 JavaScript 表达式
同时,所有的数据绑定 Vue.js 还提供了完全的 JavaScript 表达式支持:
1 | <template> |
指令
指令(
Directives) 是带有v-前缀的特殊attribute。
参数
一些指令能够接收一个“参数”,在指令名称之后以冒号表示。如,v-bind 指令可以用于响应式地绑定更新 HTML attribute、v-on 指令可以用于绑定事件监听器:
1 | <template> |
这里 v-bind 指令将该元素的 href attribute 与表达式 url 的值绑定,v-on 指令将该元素的 click 事件与 handleClick 方法进行绑定。
动态参数
可以用方括号括起来的 JavaScript 表达式作为一个指令的参数,将上面的例子修改一下就可写为:
1 | <template> |
虽然效果上好像没有什么不同,但好处是这个参数可以是动态计算出来的,方便后续修改和提供了更多操作的可能。
- 对动态参数的值的约束
动态参数预期会求出一个
字符串,异常情况下值为null。这个特殊的null值可以被显性地用于移除绑定。任何其它非字符串类型的值都将会触发一个警告。语法上的约束,如空格和引号放在
HTML attribute名里是无效的。
修饰符
修饰符(modifier)是以半角句号 . 指明的特殊后缀,用于指出一个指令应该以特殊方式绑定。例如,.prevent 修饰符告诉 v-on 指令对于触发的事件调用 event.preventDefault():
1 | <!-- 提交事件不再重载页面 --> |
修饰符可以分类为事件修饰符、按键修饰符、系统修饰符和自定义修饰符。
简写
简写是 Vue 提供的一种可选的合法字符,用于会频繁使用的两个特定的 attribute,v-bind 和 v-on。
v-bind 缩写
1 | <!-- 完整语法 --> |
v-on 缩写
1 | <!-- 完整语法 --> |
计算属性和侦听器
计算属性(Computed)
用于解耦模板内的复杂逻辑,达到方便维护和简洁清晰的效果。
1 | <template> |
计算属性缓存(computed) vs 方法(methods)
我们发现调用方法也可以实现同样的效果,区别在于 计算属性 是基于它们的响应式依赖进行缓存的,只在相关响应式依赖发生改变时它们才会重新求值。
而方法每当重新渲染时,都会再次执行函数。如果不希望一个值每次渲染都重新计算,那么使用方法在性能上是比较浪费的。
计算属性 vs 监听属性
侦听属性是一个更通用的方式来观察和响应 vue 实例上的数据变动。但 Vue 官方建议,不要滥用 watch,通常更好的做法是使用计算属性。
计算属性的 setter
计算属性默认只有 getter,不过在需要时也可以提供一个 setter:
1 | computed: { |
侦听器(Watch)
虽然计算属性在大多数情况下更合适,但有时也需要一个 自定义的侦听器。这就是为什么 Vue 通过 watch 选项提供了一个更通用的方法,来响应数据的变化。当需要在数据变化时执行 异步 或 开销较大 的操作时,这个方式是最有用的。
1 | <template> |
handler 和 immediate
以上是变化之后,wath 才执行,需要在最初时候 watch 就执行用到 ·、handler 和 immediate 属性。
1 | watch: { |
深度监听 deep
1 | <template> |
同时,官网也给出了监听不同层级的对应方法,效果与上述相同:
1 | watch: { |
监听同时也可以通过方法的形式写在 methods 中:
1 | watch: { |
Class 与 Style 绑定
操作元素的
class列表和内联样式是数据绑定的一个常见需求。一般操作是通过v-bind绑定后处理。
而将 v-bind 用于 class 和 style 时,Vue.js 做了专门的增强。表达式结果的类型除了字符串之外,还可以是对象或数组。
绑定 HTML Class
对象语法
1 | <div :class="{ active: isActive }"></div> |
1 | data() { |
渲染后:
1 | <div class="active"></div> |
同时,也可以绑定在一个返回的计算属性中:
1 | <div :class="classObject"></div> |
1 | data() { |
以上效果是相同的。
数组语法
1 | <div :class="classObject"></div> |
1 | data() { |
渲染后:
1 | <div class="active text"></div> |
绑定内联样式
对象语法
1 | <div :style="styles">Akashi</div> |
1 | data() { |
同样的,对象语法常常结合返回对象的计算属性使用。
数组语法
1 | <div :style="[baseStyles, overridingStyles]">Akashi</div> |
1 | data() { |
条件渲染
v-if指令用于条件性地渲染一块内容。当返回的值为真的时候被渲染。
v-if
1 | <div class="container"> |
1 | data() { |
渲染结果:
1 | <div class="container"> |
用 key 管理可复用元素
默认 Vue 元素是复用的,这样可以高效的渲染元素,例如:
1 | <template v-if="loginType === 'username'"> |
上面的代码中切换 loginType 将不会清除用户已经输入的内容。因为两个模板使用了相同的元素,<input> 不会被替换掉——仅仅是替换了它的 placeholder。
当需求是不需要复用,使用完全独立的两个元素时,可以添加具有唯一值的 key 属性。
1 | <template v-if="loginType === 'username'"> |
v-show
与
v-if相似,根据条件展示元素。
1 | <div v-show="true">Show</div> |
不同的是带有 v-show 的元素始终会被渲染并保留在 DOM 中。v-show 只是简单地切换元素的 CSS property display。
v-if vs v-show
v-if 会动态的渲染和重建节点元素,是属于真正的条件渲染,同时它也是惰性的,只有条件为真时才会渲染。
v-show 无论条件真假都会渲染,只是动态对 CSS 进行切换进而实现显隐。
一般来说,v-if 有更高的切换开销,而 v-show 有更高的初始渲染开销。因此,如果需要非常频繁地切换,则使用 v-show 较好;如果在运行时条件很少改变,则使用 v-if 较好。
同时,不推荐同时使用 v-if 和 v-for。当 v-if 与 v-for 一起使用时,v-for 具有比 v-if 更高的优先级,即在每次重新渲染的时候都会遍历整个列表,即使 v-if 为 false。
列表渲染
v-for指令基于一个数组来渲染一个列表。
v-for
1 | <div class="container"> |
1 | data() { |
注意参数顺序,第一个参数为值,第二个参数为当前项的索引(可省略)。
为了 Vue 能跟踪每个节点的身份,从而重用和重新排序现有元素,需要为每项提供一个唯一的 key 属性。
使用对象
同时 v-for 也支持使用对象进行迭代:
1 | <div class="container"> |
1 | data() { |
注意:这时,参数顺序变为了键值、键名、索引。
同时,也可以用 of 替代 in 作为分隔符,它更接近 JavaScript 迭代器的语法。
而我们在实际中的开发中,更多的是使用数组对象的情况:
1 | <div class="container"> |
1 | data() { |
数组更新检测
Vue 将 JS 的数组方法进行了侦听,数组方法的触发也会动态的更新到视图。
其中包括:
变更方法
push()pop()shift()unshift()splice()sort()reverse()join()
注:此类方法对原数组进行了修改。
替换数组
filter()concat()slice()map()
注:此类方法会返回一个新的数组,而不会改变原始数组。
事件处理
监听事件及处理方法
可以用 v-on 指令监听 DOM 事件(简写为 @),并在触发时运行一些 JavaScript 代码。
1 | <div class="container"> |
1 | data() { |
访问原始的 DOM 事件。可以使用特殊变量 $event,将其传入方法:
1 | <button @click="handleClick($event)">click</button> |
1 | handleClick(e) { |
注:如果函数不需要多余参数值,默认定义的参数同样可以获取 DOM 事件。
1 | <button @click="handleClick">click</button> |
1 | handleClick(e) { |
事件修饰符
在事件处理程序中调用
event.preventDefault()或event.stopPropagation()是非常常见的需求。尽管我们可以在方法中轻松实现这点,但更好的方式是:方法只有纯粹的数据逻辑,而不是去处理DOM事件细节。
为此,Vue.js 为 v-on 提供了事件修饰符:
.stop.prevent.capture.self.once.passive
1 | <!-- 阻止单击事件继续传播 --> |
按键修饰符
1 | <div class="container"> |
1 | onEnter(e) { |
为了在必要的情况下支持旧浏览器,Vue 提供了绝大多数常用的按键码的别名:
.enter.tab.delete(捕获“删除”和“退格”键).esc.space.up.down.left.right
还可以通过全局 config.keyCodes 对象自定义按键修饰符别名:
1 | // 可以使用 `v-on:keyup.f1` |
系统修饰符
可以用如下修饰符来实现仅在按下相应按键时才触发鼠标或键盘事件的监听器。
.ctrl.alt.shift.meta
注:在 Mac 系统键盘上,meta 对应 command 键 (⌘)。在 Windows 系统键盘 meta 对应 Windows 徽标键 (⊞)。在 Sun 操作系统键盘上,meta 对应实心宝石键 (◆)。
1 | <!-- Alt + C --> |
.exact 修饰符
.exact 修饰符允许你控制由精确的系统修饰符组合触发的事件。
1 | <!-- 即使 Alt 或 Shift 被一同按下时也会触发 --> |
鼠标按钮修饰符
.left.right.middle
这些修饰符会限制处理函数仅响应特定的鼠标按钮。
表单输入绑定
使用
v-model指令在表单<input>、<textarea>及<select>元素上创建双向数据绑定。它会根据控件类型自动选取正确的方法来更新元素。
v-model 在内部为不同的输入元素使用不同的 property 并抛出不同的事件:
- text 和 textarea 元素使用 value property 和 input 事件;
- checkbox 和 radio 使用 checked property 和 change 事件;
- select 字段将 value 作为 prop 并将 change 作为事件
基础用法
文本
1 | <div class="container"> |
1 | data() { |
多行文本
1 | <div class="container"> |
1 | data() { |
复选框
1 | <div class="container"> |
1 | data() { |
多个复选框:
1 | <div class="container"> |
1 | data() { |
单选框
1 | <div class="container"> |
1 | data() { |
选择框
1 | <div class="container"> |
1 | data() { |
绑定值
即单选框、复选框、选择框的 value 的绑定。各组件 v-model 绑定状态下会修改 vlue 的值。
修饰符
.lazy
在默认情况下,v-model 在每次 input 事件触发后将输入框的值与数据进行同步。通过添加 lazy 修饰符,从而转为在 change 事件 之后 进行同步。
.number
如果想自动将用户的输入值转为数值类型,可以给 v-model 添加 number 修饰符。
.trim
如果要自动过滤用户输入的首尾空白字符,可以给 v-model 添加 trim 修饰符。
组件基础
基本使用:
ButtonCounter.vue
1 | <template> |
App.vue
1 | <template> |
复用
组件可以被多次复用,但每个组件都会维护一个独立的
data。
一个组件的 data 选项必须是一个函数,因此每个实例可以维护一份被返回对象的独立的拷贝。
App.vue
1 | <template> |
Prop 向子组件传递数据
Prop允许我们在组件上注册一些自定义属性,子组件prop属性通过接收父组件传递过来的值实现数据传递。
1 | <template> |
BlogPost.vue
1 | <template> |
单个根元素
值得注意的是,每个组件必须只有一个
根元素。在开发时,可以使用一个父元素统一将模板内容包裹。prop太多也需要对数据进行重构,以保证内容的清晰和易维护。
一个好的例子:
BlogPost.vue
1 | <template> |
App.vue
1 | <template> |
监听子组件事件
在开发中,不仅仅需要自上而下从父组件向子组件传递数据,同时也会有子组件向父组件进行沟通的需求。
为保证单向数据流,自然是不允许子组件直接修改父组件的数据,造成数据的维护困难。这里就要引出自定义事件对子组件进行监听:
父级组件可以像处理 native DOM 事件一样通过 v-on 监听子组件实例的任意事件:
App.vue
1 | <template> |
同时子组件可以通过调用内建的 $emit 方法并传入事件名称来触发一个事件:
BlogPost.vue
1 | <template> |
使用事件抛出一个值
有时候需要向上传递一个特殊的值,这时可以使用 $emit 的第二个参数来提供这个值:
BlogPost.vue
1 | <template> |
这时,父组件在监听这个事件的时候,可以通过 $event 访问到被抛出的这个值:
App.vue
1 | <template> |
在组件上使用 v-model
自定义事件也可以用于创建支持 v-model 的自定义输入组件。
1 | <template> |
等价于:
1 | <template> |
为了让组件正常工作,这个组件内的 <input> 必须:
- 将其
valueattribute绑定到一个名叫value的prop上 - 在其
input事件被触发时,将新的值通过自定义的input事件抛出
1 | <template> |
这样就可以在组件上使用 v-model。
1 | <template> |
通过插槽分发内容
有时父组件需要动态的向一个子组件传递内容。
这时可以使用 slot 插槽在子组件中进行占位:
ContextBox.vue
1 | <template> |
然后在父组件中可以直接将节点内容传递过去:
App.vue
1 | <template> |
实际上插槽还有具名插槽和作用域插槽等,这里暂时先只介绍一般插槽的使用。
动态组件
在一个需求中,需要在不同的组件之间动态的进行切换,这时可以通过 Vue 的 <component> 元素加一个特殊的 is attribute 来实现:
1 | <template> |
is 属性指明了被绑定的组件,这个 attribute 可以用于常规 HTML 元素,但这些元素也将被视为组件,这意味着所有的 attribute 都会作为 DOM attribute 被绑定。
以上就是 Vue 官方文档中使用 vue 的基础介绍梳理。在重看整理的同时也加入了一些笔者常用内容的记录、对官方的例子进行了具体实现。