Vue2基础知识(二) 计算属性
一 计算属性
1.1 理解
🌈如何来理解计算属性? Vue的计算属性是一个能够基于现有的Vue实例数据进行计算的属性,它们的值是根据依赖的数据动态计算而来的,只有在依赖的数据发生变化时才会重新计算,计算属性常常用于模板表达式中,可以将复杂的逻辑计算封装在计算属性中,使模板更加简洁清晰。 我的理解:通过原有的数据动态计算,才能得一个新的属性 🌈几个概念?
- 计算属性是基于现有的Vue实例数据进行计算的属性,它们的值是根据依赖的数据动态计算而来的。
- 计算属性具有缓存特性,只有在依赖的数据发生变化时才会重新计算。
- 计算属性可以被用作模板表达式中的数据绑定。
- 计算属性支持getter和setter方法,可以通过setter方法实现数据的双向绑定。
<!--
* @Description: 计算属性的理解
* @version: 1.0
* @Author: shu
* @Date: 2023-04-16 16:14:12
* @LastEditors: 修改者填写
* @LastEditTime: 2023-04-16 16:21:04
-->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>计算属性</title>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
<div id="app">
<div>
<p>商品名称:{{ product.name }}</p>
<p>商品价格:{{ salePrice }}</p>
<button @click="product.discount = 0.5">打五折</button>
</div>
</div>
<script>
var app = new Vue({
el: '#app',
data: {
product: {
name: '电视机',
price: 1000,
discount: 0.8
}
},
computed: {
salePrice() {
return this.product.price * this.product.discount
}
}
})
</script>
</html>
上面的例子中,我们定义了一个商品对象,其中包含了商品的名称、价格和折扣,我们通过计算属性salePrice来计算商品的实际售价,然后在模板表达式中使用它来显示商品的价格,由于salePrice是一个计算属性,它的值会随着商品价格和折扣的变化而动态更新,在这个例子中,我们只需要更新商品对象的价格和折扣属性,就可以实现商品价格的实时更新。
1.2 getter和setter
计算属性拥有的两个方法:getter setter 用来实现数据的双向绑定
- 每一个计算属性都包含一个 getter 和一个 setter,我们上面的两个示例都是计算属性的默认用法 , 只是利用了 getter来读取。在你需要时,也可以提供一个 setter 函数 , 当手动修改计算属性的值就像修改一个普通数据那样时,就会触发 setter函数,执行一些自定义的操作
- 计算属性除了上述简单的文本插值外,还经常用于动态地设置元素的样式名称 class 和内联样式 style
<!--
* @Description: get与set方法
* @version: 1.0
* @Author: shu
* @Date: 2023-04-16 16:29:23
* @LastEditors: 修改者填写
* @LastEditTime: 2023-04-16 16:29:23
-->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>get与set方法</title>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
<div id="app">
<div>
<div>{{ formSum.sum }}</div>
<div><button @click="buttonPlusClcik">加点击</button></div>
<div>{{ dataSum.sum }}</div>
<div><button @click="buttonReduceClcik">减点击</button></div>
<!-- 总和展示 -->
<div>{{ theSum }}</div>
</div>
</div>
<script>
var app = new Vue({
el: '#app',
data: {
formSum: {
sum: 0,
sumbutton: 100,
},
dataSum: {
sum: 10,
sumbutton: 1000,
},
},
computed: {
theSum: {
get() {
return this.formSum.sum + this.dataSum.sum
},
set(value) {
this.formSum.sum = value
this.dataSum.sum = value
}
}
},
methods: {
// 加点击
buttonPlusClcik() {
this.theSum++
},
// 减点击
buttonReduceClcik() {
this.theSum--
}
}
})
</script>
</html>
1.3 方法
我们将上面的案例来改写一种方法,定义方法来写,它有是如何写的呢?
<!--
* @Description: 方法
* @version: 1.0
* @Author: shu
* @Date: 2023-04-16 16:29:23
* @LastEditors: 修改者填写
* @LastEditTime: 2023-04-16 16:30:34
-->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>方法</title>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
<div id="app">
<div>
<p>商品名称:{{ product.name }}</p>
<p>商品价格:{{ salePrice() }}</p>
<button @click="product.discount = 0.5">打五折</button>
</div>
</div>
<script>
var app = new Vue({
el: '#app',
data: {
product: {
name: '电视机',
price: 1000,
discount: 0.8
}
},
methods: {
salePrice() {
return this.product.price * this.product.discount
}
}
})
</script>
</html>
我们可以发现效果是一样的,但是这其实是有区别的? ⏺️使用计算属性的好处是,它具有缓存特性,只有在依赖的数据发生变化时才会重新计算,因此效率更高。 ⏺️而使用methods的方法则需要每次调用都重新计算,因此效率较低,此外,使用计算属性还可以将计算逻辑封装在属性中,使代码更加清晰简洁。 ⏺️总之,使用计算属性可以提高代码的可读性和可维护性,同时也可以提高程序的运行效率,但是如果逻辑比较复杂或需要处理异步操作,可能需要使用methods来实现。
1.4 侦听属性
🌈如何理解帧听属性? Vue中的侦听属性(watch)是一个很有用的特性,它可以让开发者监听某个特定的数据属性,并在其发生变化时执行一些特定的操作,侦听属性通常用于处理需要在特定属性发生变化时进行异步操作或复杂逻辑处理的情况。
<!--
* @Description: 侦听属性
* @version: 1.0
* @Author: shu
* @Date: 2023-04-16 16:46:33
* @LastEditors: 修改者填写
* @LastEditTime: 2023-04-16 16:46:33
-->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>侦听属性</title>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
<div id="app">
<div>
<p>商品名称:{{ product.name }}</p>
<p>商品价格:{{ product.price }}</p>
<p>商品折扣:{{ product.discount }}</p>
<p>商品折后价:{{ salePrice }}</p>
<button @click="product.discount = 0.5">打五折</button>
</div>
</div>
<script>
var app = new Vue({
el: '#app',
data: {
product: {
name: '电视机',
price: 1000,
discount: 0.8
}
},
computed: {
salePrice() {
return this.product.price * this.product.discount
}
},
watch: {
'product.discount': function (newVal, oldVal) {
console.log('newVal: ' + newVal)
console.log('oldVal: ' + oldVal)
}
}
})
</script>
</html>
我们可以发现对我们指定的属性进行监听,可以发现一个新值,与旧值,这就方便我们进行一些特定的操作 🌈如何来决定何时使用计算属性与帧听属性? 在Vue中,使用计算属性和侦听属性都可以处理特定的数据变化,但它们的使用场景是不同的,通常来说,当我们需要计算一些值,并将其作为属性暴露给模板时,可以使用计算属性,而当我们需要在某些数据发生变化时,执行一些异步或复杂的操作,可以使用侦听属性。 具体来说,计算属性适用于以下情况:
- 当需要根据某些数据计算出一个值,并在模板中显示时,可以使用计算属性。例如,计算商品的折扣价格、根据用户输入的关键字过滤数据等等。
- 当需要根据其他属性或状态更新某个属性时,可以使用计算属性。例如,在计算属性中根据用户选择的语言切换页面显示的文字。
侦听属性适用于以下情况:
- 当需要在特定数据发生变化时执行异步或复杂操作时,可以使用侦听属性。例如,当用户选择某个城市时,需要通过网络请求获取该城市的天气信息。
- 当需要监听多个数据变化,并在它们发生变化时执行某些操作时,可以使用侦听属性。例如,当商品数量或价格发生变化时,需要重新计算购物车中的总价。
需要注意的是,使用计算属性和侦听属性时需要注意性能问题,计算属性具有缓存机制,只有在依赖的数据发生变化时才会重新计算,因此不会因为频繁计算而降低性能,而侦听属性则不具有缓存机制,每当侦听的属性变化时都会执行一次处理函数,如果处理函数比较耗时,可能会影响页面的性能。 综上所述,使用计算属性和侦听属性时需要根据具体的业务场景和数据变化的情况进行选择,同时也需要注意性能问题。
1.5 完整写法
<!--
* @Description: 侦听属性
* @version: 1.0
* @Author: shu
* @Date: 2023-04-16 16:46:33
* @LastEditors: 修改者填写
* @LastEditTime: 2023-04-16 16:46:33
-->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>侦听属性</title>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
<div id="app">
<div>
<p>商品名称:{{ product.name }}</p>
<p>商品价格:{{ product.price }}</p>
<p>商品折扣:{{ product.discount }}</p>
<p>商品折后价:{{ salePrice }}</p>
<button @click="product.discount = 0.5">打五折</button>
</div>
</div>
<script>
Vue.config.devtools = true;
var app = new Vue({
el: '#app',
data: {
product: {
name: '电视机',
price: 1000,
discount: 0.8
}
},
computed: {
salePrice: {
get() {
return this.product.price * this.product.discount
}
}
},
watch: {
product: {
// // 对复杂数据类型的监听
deep: true,
// 立即执行
immediate: true,
// 侦听属性
handler(newVal, oldVal) {
console.log('newVal: ' + newVal)
console.log('oldVal: ' + oldVal)
},
},
}
})
</script>
</html>
如果对象内有多个属性,并采用以下写法,则对象内每个属性都会被侦听,每个属性的变化都会执行一次侦听操作。
二 生命周期
Vue生命周期是指在一个Vue实例创建、更新和销毁过程中,各个阶段发生的事件和回调函数。Vue的生命周期分为以下8个阶段:
- 创建前(beforeCreate):在实例被创建之前调用,此时实例的属性和方法都没有被初始化。
- 创建后(created):在实例被创建之后立即调用,此时实例已经完成了数据观测(data observer)、属性和方法的初始化(init props、init methods、init data、init computed、init watch)。
- 挂载前(beforeMount):在实例被挂载到DOM之前调用,此时实例的template还没有被渲染成真实的DOM。
- 挂载后(mounted):在实例被挂载到DOM之后立即调用,此时实例的template已经被渲染成真实的DOM,可以对DOM进行操作。
- 更新前(beforeUpdate):在数据更新之前调用,此时实例的数据已经被更新,但是视图还没有被重新渲染。
- 更新后(updated):在数据更新之后立即调用,此时视图已经被重新渲染。
- 销毁前(beforeDestroy):在实例被销毁之前调用,此时实例的数据、方法、监听器都还可用。
- 销毁后(destroyed):在实例被销毁之后调用,此时实例的所有东西都已经被清理,无法再访问到它们。
<!--
* @Description: 生命周期
* @version: 1.0
* @Author: shu
* @Date: 2023-04-16 17:48:16
* @LastEditors: 修改者填写
* @LastEditTime: 2023-04-16 17:50:34
-->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>生命周期</title>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
<div id="app">
<div>
<p>商品名称:{{ product.name }}</p>
<p>商品价格:{{ product.price }}</p>
<p>商品折扣:{{ product.discount }}</p>
<p>商品折后价:{{ salePrice() }}</p>
<button @click="product.discount = 0.5">打五折</button>
</div>
</div>
<script>
var app = new Vue({
el: '#app',
data: {
product: {
name: '电视机',
price: 1000,
discount: 0.8
}
},
computed: {
salePrice() {
return this.product.price * this.product.discount
}
},
methods: {
salePrice() {
return this.product.price * this.product.discount
}
},
beforeCreate() {
console.log('beforeCreate')
},
created() {
console.log('created')
},
beforeMount() {
console.log('beforeMount')
},
mounted() {
console.log('mounted')
},
beforeUpdate() {
console.log('beforeUpdate')
},
updated() {
console.log('updated')
},
beforeDestroy() {
console.log('beforeDestroy')
},
destroyed() {
console.log('destroyed')
}
})
</script>
</html>
总结一下: Vue生命周期钩子函数可以分为以下几类:
- 创建阶段:从创建Vue实例到挂载到DOM上之前的阶段。
- beforeCreate
- created
- 挂载阶段:将Vue实例挂载到DOM上的阶段。
- beforeMount
- mounted
- 更新阶段:当Vue实例的数据发生变化时,触发重新渲染视图的阶段。
- beforeUpdate
- updated
- 销毁阶段:当Vue实例被销毁时的阶段。
- beforeDestroy
- destroyed
下面我们来详细介绍每个钩子函数的作用和应用场景。
- beforeCreate
在实例被创建之初,此钩子函数会被调用。此时,Vue实例的数据和方法都还未初始化,因此无法访问data、computed、methods等选项。 这个阶段的应用场景比较少,一般用于插件开发或者扩展Vue功能。例如,可以在这个钩子函数中添加全局指令或者自定义方法。
- created
在实例初始化完成后,此钩子函数会被调用。此时,Vue实例的数据和方法已经被初始化,可以访问data、computed、methods等选项。 这个阶段的应用场景比较多,可以进行一些初始化操作,例如获取数据、初始化事件等。注意,在这个钩子函数中,尚未完成挂载阶段,因此无法访问到DOM。
- beforeMount
在Vue实例挂载到DOM上之前,此钩子函数会被调用。此时,Vue实例已经完成了模板的编译,并且将要把编译后的模板挂载到指定的DOM元素上。 这个阶段的应用场景比较少,一般用于在DOM元素挂载之前进行一些操作,例如修改数据、添加样式等。
- mounted
在Vue实例挂载到DOM上之后,此钩子函数会被调用。此时,Vue实例已经完成了挂载,可以访问到挂载后的DOM元素。 这个阶段的应用场景比较多,可以进行一些与DOM相关的操作,例如初始化插件、设置定时器、绑定事件等。
- beforeUpdate
在Vue实例数据发生变化之前,此钩子函数会被调用。此时,Vue实例的数据已经更新,但是还未重新渲染视图。 这个阶段的应用场景比较少,一般用于在数据更新之前进行一些操作,例如记录数据更新前的状态、取消更新等。
- updated
在Vue实例数据发生变化之后,此钩子函数会被调用。此时,Vue实例已经完成了数据更新和重新渲染视图。 这个阶段的应用场景比较多,可以进行一些与DOM相关的操作,例如更新插件、重新绑定事件等。 需要注意的是,在这个钩子函数中,如果修改了数据,会导致无限循环更新的问题,因此要避免在这个钩子函数中修改数据。
- beforeDestroy
在Vue实例销毁之前,此钩子函数会被调用。此时,Vue实例还未被销毁,可以访问到Vue实例的数据和方法。 这个阶段的应用场景比较少,一般用于在Vue实例被销毁之前进行一些清理操作,例如取消定时器、解绑事件等。
- destroyed
在Vue实例被销毁之后,此钩子函数会被调用。此时,Vue实例已经被销毁,无法访问到Vue实例的数据和方法。 这个阶段的应用场景比较少,一般用于在Vue实例被销毁之后进行一些清理操作,例如释放内存、取消引用等。
三 数据检测的原理
Vue的数据检测原理是响应式编程的一种实现方式,它通过数据劫持和发布-订阅模式来实现数据的自动更新和视图的响应式更新,具体来说,Vue的数据检测原理主要包括以下几个步骤:
- 数据劫持
当我们使用Vue创建一个实例时,Vue会对其数据对象进行递归地遍历,将每个属性都转换为getter和setter函数,这个过程就是数据劫持。这样,在我们访问或修改数据对象的属性时,Vue就能够捕获到这个操作,并通过getter和setter函数来实现响应式更新。
- 发布-订阅模式
Vue通过一个中央事件总线(即Vue实例的$emit和$on方法)来实现发布-订阅模式。在数据对象被访问或修改时,Vue会自动触发相应的事件,并通知所有订阅了这个事件的Watcher对象。
- Watcher对象
Watcher对象是Vue的核心概念之一,它负责监测数据的变化并更新视图。每个Watcher对象都对应着一个DOM元素或组件实例,当Watcher对象接收到数据变化的事件时,它会重新计算虚拟DOM树并更新对应的视图。在Vue的内部实现中,Watcher对象被组织成一个树状结构,并使用依赖收集的方式来管理依赖关系,这样就能够避免不必要的DOM操作,提高渲染性能。
- 批处理
为了减少不必要的DOM操作,Vue使用了一些优化策略,例如异步更新队列和批处理机制。当数据发生变化时,Vue会将需要更新的Watcher对象添加到一个异步更新队列中,然后通过nextTick方法来异步执行队列中的更新操作。在执行更新操作时,Vue会尽可能地合并多个更新操作,以减少不必要的DOM操作和渲染开销。
<DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>测试</title>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
<div id="app">
<div>
<p>{{ message }}</p>
<button @click="changeMessage">Change Message</button>
</div>
</div>
<script>
var app = new Vue({
el: '#app',
data: {
message: 'Hello, Vue!'
},
methods: {
changeMessage() {
this.message = 'Hello, World!'
}
},
})
</script>
- 在这个案例中,我们有一个数据属性message和一个方法changeMessage,当用户点击按钮时,changeMessage方法会将message的值改为'Hello, World!'。
- 这时,Vue的数据检测机制会自动检测到message的变化,并触发视图的重新渲染,从而将新的值'Hello, World!'显示在页面上。
- 具体来说,当我们创建这个Vue实例时,Vue会对data对象中的message属性进行数据劫持,将其转换为getter和setter函数。
- 当用户点击按钮时,changeMessage方法会修改message的值,这个操作会触发message的setter函数,并向中央事件总线(即Vue实例)发布一个数据变化的事件。
- 此时,与message相关的Watcher对象会接收到这个事件,并将自己添加到异步更新队列中。
- 最后,通过nextTick方法异步执行队列中的更新操作,重新计算虚拟DOM树并更新对应的视图,将新的值'Hello, World!'渲染到页面上。