keep-alive组件的作用与原理

什么是keep-alive

“keep-alive” 是 Vue.js 中的一个特殊组件,用于缓存组件的状态,以提高应用性能。在 Vue.js 中,组件通常是动态创建和销毁的,当切换到另一个页面或组件时,之前的组件会被销毁,再次进入时会重新创建和初始化。这样可能导致组件的状态丢失,需要重新初始化,增加了资源的消耗。

组件解决了这个问题,它可以将组件缓存起来,而不是销毁,使得组件在再次进入时保持之前的状态,以及避免重复的创建和初始化过程。这样可以大幅度提高组件的加载速度和性能。

keep-alive的作用

<keep-alive> 的作用是在 Vue.js 应用中缓存组件的状态,以提高应用性能和用户体验。它可以将组件暂时保存在内存中,而不是每次都重新创建和初始化组件。

主要的作用有以下几点:

  1. 组件状态保持:通过使用 <keep-alive>,在组件被切换时,其状态会被保留。这意味着组件内部的数据、状态以及一些计算结果都会被缓存,不会因为组件的销毁而丢失。当再次进入该组件时,它会恢复到之前的状态,而不需要重新初始化。这对于用户在不同页面或组件间切换时提供了更流畅的体验。

  2. 减少资源消耗:如果没有使用 <keep-alive>,每次切换到一个组件时,都需要重新创建和初始化组件。对于复杂的组件,这可能会导致不必要的资源消耗,例如重新加载数据、执行复杂的计算等。而使用 <keep-alive>,组件被缓存起来,下次再次进入时直接从缓存中恢复,避免了重复的初始化过程,大大减少了资源消耗。

  3. 优化性能:由于避免了重复的创建和初始化过程,使用 <keep-alive> 可以显著提高组件的加载速度,加快页面响应时间,从而提供更好的用户体验。

需要注意的是,<keep-alive> 并不是适用于所有组件的,特别是对于一些动态变化的组件,如果希望每次进入时都重新初始化,或者希望释放组件占用的资源,就不应该使用 <keep-alive>

要使用 <keep-alive>,只需将需要缓存的组件包裹在 <keep-alive> 标签中即可,Vue.js 会自动管理缓存和组件的生命周期。这是一个简单但强大的功能,可在合适的场景下大幅度提升应用性能。

原理

<keep-alive> 的原理主要涉及两个方面:组件缓存和生命周期的管理。

  1. 组件缓存

    • 当一个组件被包裹在 <keep-alive> 标签中时,Vue.js 会将该组件的实例缓存起来,而不是销毁它。
    • 组件的缓存是通过一个名为 cache 的对象来管理的,该对象会保存被缓存的组件实例。
    • 当切换到一个被缓存的组件时,Vue.js 首先检查 cache 对象中是否已经有该组件的缓存实例。如果有,就直接从缓存中取出该实例;如果没有,就创建一个新的组件实例并将其缓存起来。
  2. 生命周期的管理

    • 在切换到一个被缓存的组件时,组件的生命周期钩子函数并不会被触发,而是会触发 <keep-alive> 自己的生命周期钩子函数。
    • <keep-alive> 组件有两个主要的生命周期钩子函数:createddestroyed
    • 在组件第一次被缓存时,created 钩子函数会被触发,表示 <keep-alive> 组件已经创建,此时会创建被缓存组件的实例并将其缓存起来。
    • 在切换到其他组件时,destroyed 钩子函数会被触发,表示 <keep-alive> 组件将被销毁,此时会销毁所有缓存的组件实例。

需要注意的是,被包裹在 <keep-alive> 标签中的组件,必须具有唯一的标识,否则会导致缓存冲突。默认情况下,Vue.js 使用组件的名称作为缓存的标识,但也可以通过 key 属性来指定唯一的标识。

使用 <keep-alive> 时,要注意以下几点:

  • 不是所有组件都适合使用 <keep-alive>,对于一些动态变化的组件,或者需要每次进入时重新初始化的组件,应该避免使用 <keep-alive>
  • 缓存的组件仍然会触发 activateddeactivated 生命周期钩子函数,可以在这两个钩子函数中处理一些特定的逻辑。
  • 如果被缓存的组件包含了一些依赖于外部状态(如路由参数、Vuex 状态等)的逻辑,需要特别注意在重新进入组件时是否需要重新更新这些状态。

总的来说,<keep-alive> 提供了一种简单且强大的机制来优化 Vue.js 应用的性能,特别是在频繁切换组件的场景下。

使用

当您使用 <keep-alive> 组件时,通常需要将需要缓存的组件包裹在 <keep-alive> 标签中,并为每个被缓存的组件设置一个唯一的 key 属性,以确保缓存的正确性。下面是一个使用 <keep-alive> 组件的示例:

假设我们有两个组件,一个是用于显示用户信息的组件 <UserProfile>,另一个是用于显示用户订单信息的组件 <UserOrders>。我们希望在用户切换到 <UserProfile> 组件时,保持该组件的状态,并且在用户切换到 <UserOrders> 组件后再切换回来时,不重新初始化 <UserProfile> 组件。

<template><div><keep-alive><!-- 使用 key 属性确保组件的正确缓存 --><component :is="currentComponent" :key="currentComponent" /></keep-alive><button @click="showUserProfile">Show User Profile</button><button @click="showUserOrders">Show User Orders</button></div>
</template><script>
import UserProfile from './UserProfile.vue';
import UserOrders from './UserOrders.vue';export default {components: {UserProfile,UserOrders,},data() {return {currentComponent: 'UserProfile', // 初始显示用户信息组件};},methods: {showUserProfile() {this.currentComponent = 'UserProfile';},showUserOrders() {this.currentComponent = 'UserOrders';},},
};
</script>

在上面的示例中,使用了动态组件 <component :is="currentComponent"> 来动态地切换显示 <UserProfile><UserOrders> 组件。同时,将 <keep-alive> 标签包裹在动态组件外部,这样 <keep-alive> 会缓存当前被显示的组件。

在切换组件时,使用 key 属性来确保缓存的正确性。当切换到不同的组件时,key 的值会变化,这会触发 <keep-alive> 的重新缓存行为。

注意,key 属性应该是唯一的,以确保每个组件都能被正确地缓存。在实际应用中,可能需要根据组件的具体情况设置不同的 key 值。

理解源码

<keep-alive> 组件的源码相对比较复杂,涉及到 Vue.js 的虚拟 DOM、组件实例管理、生命周期管理等方面。下面简要介绍 <keep-alive> 的关键源码部分,以便了解其基本原理。

在 Vue.js 的源码中,<keep-alive> 组件是由一个特殊的内置组件 KeepAlive 实现的。它的主要作用是处理组件的缓存和管理缓存组件的生命周期。

  1. 组件的缓存实现:

    • KeepAlive 组件内部维护了一个名为 cache 的对象,用于存储缓存的组件实例。
    • 在切换到一个被缓存的组件时,KeepAlive 组件首先会检查 cache 对象,是否已经有该组件的缓存实例。
    • 如果缓存中有该组件实例,KeepAlive 直接返回缓存的组件实例;如果没有,KeepAlive 会创建一个新的组件实例,并将其缓存起来。
  2. 组件生命周期的管理:

    • KeepAlive 组件有两个重要的生命周期钩子函数:createddestroyed
    • created 钩子函数中,KeepAlive 会监听父组件的 includeexclude 属性的变化,以决定是否缓存某个组件。
    • 在切换到被缓存组件时,KeepAlive 会触发 activated 生命周期钩子函数,并从 cache 中取出对应的缓存组件实例。如果没有缓存实例,会触发被缓存组件的 created 生命周期。
    • 在切换到其他组件时,KeepAlive 会触发 deactivated 生命周期钩子函数,并将当前缓存的组件实例暂时从 cache 中移除。如果需要缓存,则缓存的组件实例并不会被销毁。
  3. 组件销毁时的处理:

    • destroyed 钩子函数中,KeepAlive 会销毁所有缓存的组件实例,并清空 cache 对象。

如果想深入了解 <keep-alive> 的源码实现,可以查阅 Vue.js 的 GitHub 仓库并浏览相关代码 keep-alive源码。
这个是从github拿的源码,有兴趣可以研究一下。

import { isRegExp, isArray, remove } from 'shared/util'
import { getFirstComponentChild } from 'core/vdom/helpers/index'
import type VNode from 'core/vdom/vnode'
import type { VNodeComponentOptions } from 'types/vnode'
import type { Component } from 'types/component'
import { getComponentName } from '../vdom/create-component'type CacheEntry = {name?: stringtag?: stringcomponentInstance?: Component
}type CacheEntryMap = Record<string, CacheEntry | null>function _getComponentName(opts?: VNodeComponentOptions): string | null {return opts && (getComponentName(opts.Ctor.options as any) || opts.tag)
}function matches(pattern: string | RegExp | Array<string>,name: string
): boolean {if (isArray(pattern)) {return pattern.indexOf(name) > -1} else if (typeof pattern === 'string') {return pattern.split(',').indexOf(name) > -1} else if (isRegExp(pattern)) {return pattern.test(name)}/* istanbul ignore next */return false
}function pruneCache(keepAliveInstance: { cache: CacheEntryMap; keys: string[]; _vnode: VNode },filter: Function
) {const { cache, keys, _vnode } = keepAliveInstancefor (const key in cache) {const entry = cache[key]if (entry) {const name = entry.nameif (name && !filter(name)) {pruneCacheEntry(cache, key, keys, _vnode)}}}
}function pruneCacheEntry(cache: CacheEntryMap,key: string,keys: Array<string>,current?: VNode
) {const entry = cache[key]if (entry && (!current || entry.tag !== current.tag)) {// @ts-expect-error can be undefinedentry.componentInstance.$destroy()}cache[key] = nullremove(keys, key)
}const patternTypes: Array<Function> = [String, RegExp, Array]// TODO defineComponent
export default {name: 'keep-alive',abstract: true,props: {include: patternTypes,exclude: patternTypes,max: [String, Number]},methods: {cacheVNode() {const { cache, keys, vnodeToCache, keyToCache } = thisif (vnodeToCache) {const { tag, componentInstance, componentOptions } = vnodeToCachecache[keyToCache] = {name: _getComponentName(componentOptions),tag,componentInstance}keys.push(keyToCache)// prune oldest entryif (this.max && keys.length > parseInt(this.max)) {pruneCacheEntry(cache, keys[0], keys, this._vnode)}this.vnodeToCache = null}}},created() {this.cache = Object.create(null)this.keys = []},destroyed() {for (const key in this.cache) {pruneCacheEntry(this.cache, key, this.keys)}},mounted() {this.cacheVNode()this.$watch('include', val => {pruneCache(this, name => matches(val, name))})this.$watch('exclude', val => {pruneCache(this, name => !matches(val, name))})},updated() {this.cacheVNode()},render() {const slot = this.$slots.defaultconst vnode = getFirstComponentChild(slot)const componentOptions = vnode && vnode.componentOptionsif (componentOptions) {// check patternconst name = _getComponentName(componentOptions)const { include, exclude } = thisif (// not included(include && (!name || !matches(include, name))) ||// excluded(exclude && name && matches(exclude, name))) {return vnode}const { cache, keys } = thisconst key =vnode.key == null? // same constructor may get registered as different local components// so cid alone is not enough (#3269)componentOptions.Ctor.cid +(componentOptions.tag ? `::${componentOptions.tag}` : ''): vnode.keyif (cache[key]) {vnode.componentInstance = cache[key].componentInstance// make current key freshestremove(keys, key)keys.push(key)} else {// delay setting the cache until updatethis.vnodeToCache = vnodethis.keyToCache = key}// @ts-expect-error can vnode.data can be undefinedvnode.data.keepAlive = true}return vnode || (slot && slot[0])}
}

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

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

相关文章

【直接收藏】前端JavaScript面试100问 (中篇)

31、http 的理解 ? HTTP 协议是超文本传输协议&#xff0c;是客户端浏览器或其他程序“请求”与 Web 服务器响应之间的应用层通信协议。HTTPS主要是由HTTPSSL构建的可进行加密传输、身份认证的一种安全通信通道。 32、http 和 https 的区别 ? 1、https协议需要到ca申请证书…

ASP.NET Core学习路线图

说明 1. 先决条件 - [C#](https://www.pluralsight.com/paths/csharp) - [Entity Framework](https://www.pluralsight.com/search?qentity%20framework%20core) - [ASP.NET Core](https://www.pluralsight.com/search?qasp.net%20core) - SQL基础知识 2. 通用开发技能 -…

多线程编程4:linux线程同步(信号量和读写锁)

信号量 linux信号量API&#xff1a; #include <semaphore.h> // 定义信号量对象sem_t sem; /*初始化信号量- 第一个参数&#xff1a;信号量地址- 第二个参数&#xff1a;线程同步(0),进程同步(非0)- 第三个参数&#xff1a;初始化信号量资源数(>0)&#xff0c;如果设置…

PostgreSQL:string_agg 多列值聚合成一列

PostgreSQL:string_agg 多列值聚合成一列 string_agg是PostgreSQL中的一个聚合函数&#xff0c;用于将一组值连接为一个字符串。它接受两个参数&#xff1a;要连接的值和连接符。 语法如下&#xff1a; string_agg(expression, delimiter)其中&#xff0c;expression是要连接…

uni-app:实现分页功能,单击行获取此行指定数据,更改行样式

效果&#xff1a; 分段解析代码 分页功能实现&#xff1a; 一、标签 1、搜索栏-模糊查询 <!-- 搜索框--><form action"" submit"search_wip_name"><view class"search_position"><view class"search"><…

SpringBoot项目中的web安全防护

最近这个月公司对项目进行了几次安全性扫描&#xff0c;然后扫描出来了一些安全漏洞&#xff0c;所以最近也一直在修复各种安全漏洞&#xff0c;还有就是最近在备考软考高级系统架构设计师&#xff0c;也刚好复习到了网络安全这一个章节&#xff0c;顺便将最近修复的安全漏洞总…

Debian/Ubuntu 安装 Chrome 和 Chrome Driver 并使用 selenium 自动化测试

截至目前&#xff0c;Chrome 仍是最好用的浏览器&#xff0c;没有之一。Chrome 不仅是日常使用的利器&#xff0c;通过 Chrome Driver 驱动和 selenium 等工具包&#xff0c;在执行自动任务中也是一绝。相信大家对 selenium 在 Windows 的配置使用已经有所了解了&#xff0c;下…

cmd 实现启动mysql时保留窗口

因为mysql启动后, 只有在任务管理器里能看到进程, 关的时候还需要找一下 所以基于 start cmd /k 命令实现了该效果 :: Author: admin :: Date: 2022-08-30 :: Version v1.2 :: ::启动 :: :: echo off::配置变量 set mysqlC:\mysql-5.7.38-winx64\bin\mysqld.exe::打印配置…

Jmeter 压测工具使用手册[详细]

1. jemter 简介 jmeter 是 apache 公司基于 java 开发的一款开源压力测试工具&#xff0c;体积小&#xff0c;功能全&#xff0c;使用方便&#xff0c;是一个比较轻量级的测试工具&#xff0c;使用起来非常简 单。因为 jmeter 是 java 开发的&#xff0c;所以运行的时候必须先…

在 3ds Max 中使用相机映射将静止图像转换为实时素材

推荐&#xff1a; NSDT场景编辑器 助你快速搭建可二次开发的3D应用场景 1. 在 Photoshop 中准备图像 步骤 1 这是我将在教程中使用的静止图像。 这是我的静态相机纸箱的快照。 静止图像 步骤 2 打开 Photoshop。将图像导入 Photoshop。 打开 Photoshop 步骤 3 单击套索工…

windows物理机 上安装centos ,ubuntu,等多个操作系统的要点

一、摘要 一般情况下&#xff0c;我们的笔记本或工作电脑都默认安装windows 分几个区&#xff0c;当下是win7 win8 win 10 win11 等&#xff0c;突然我们有需求需要安装个centos &#xff0c;后面我们应当怎么做&#xff0c;要点是什么&#xff1f;一定要根据网上的贴子一步步来…

Word导出高清PDF

通过word导出pdf清晰度较高的方法_word如何导出高分辨率pdf_Perishell的博客-CSDN博客通过打印机属性设置&#xff0c;让word打印出比较高清的pdf_word如何导出高分辨率pdfhttps://blog.csdn.net/weixin_45390670/article/details/129228568?ops_request_misc%257B%2522reques…

学习笔记|C251|STC32G单片机视频开发教程(冲哥)|第三集:开发环境搭建和程序下载

文章目录 1.STC-ISP软件的下载2.STC32手册下载3.PDF阅读器下载4.学会PDF阅读器查阅手册5.跟着手册搭建C251开发环境Tips:如何同时安装Keil的C51、C251和MDK 6.程序包的下载7.第一个工程的编译和下载 原作者/主讲人&#xff1a;冲哥 原始视频地址 1.STC-ISP软件的下载 STC-ISP …

应急响应-主机后门webshell的排查思路(webshell,启动项,隐藏账户,映像劫持,rootkit后门)

0x00 windows主机后门排查思路 针对主机后门windows&#xff0c;linux&#xff0c;在对方植入webshell后&#xff0c;需要立即响应&#xff0c;排查出后门位置&#xff0c;以及排查对外连接&#xff0c;端口使用情况等等 排查对外连接状态&#xff1a; 借助工具&#xff1a;p…

数据结构 | 线性数据结构——列表

目录 一、无序列表抽象数据类型 二、实现无序列表&#xff1a;链表 2.1 Node类 2.2 UnorderedList类 三、有序列表抽象数据类型 四、实现有序列表 列表是元素的集合&#xff0c;其中每一个元素都有一个相对于其他元素的位置。更具体地说&#xff0c;这种列表成为无序列表…

GDB Debug

使用gdb带着参数启动程序 在gdb中启动程序并传递命令行参数&#xff1a; gdb ./my_program (gdb) run arg1 arg2 arg3 这将在gdb中启动程序"my_program"&#xff0c;并将参数"arg1"、"arg2"和"arg3"传递给程序。 在启动gdb之前&…

后端进阶之路——浅谈Spring Security用户、角色、权限和访问规则(三)

前言 「作者主页」&#xff1a;雪碧有白泡泡 「个人网站」&#xff1a;雪碧的个人网站 「推荐专栏」&#xff1a; ★java一站式服务 ★ ★前端炫酷代码分享 ★ ★ uniapp-从构建到提升★ ★ 从0到英雄&#xff0c;vue成神之路★ ★ 解决算法&#xff0c;一个专栏就够了★ ★ 架…

代码随想录算法训练营day48

文章目录 Day48 打家劫舍题目思路代码 打家劫舍II题目思路代码 打家劫舍 III题目思路代码递归去偷动态规划法&#xff08;状态标记递归&#xff09; Day48 打家劫舍 198. 打家劫舍 - 力扣&#xff08;LeetCode&#xff09; 题目 你是一个专业的小偷&#xff0c;计划偷窃沿街…

两数相加 II

给你两个 非空 链表来代表两个非负整数。数字最高位位于链表开始位置。它们的每个节点只存储一位数字。将这两数相加会返回一个新的链表。 你可以假设除了数字 0 之外&#xff0c;这两个数字都不会以零开头。 示例1&#xff1a; 输入&#xff1a;l1 [7,2,4,3], l2 [5,6,4] 输…

什么是 webpack?

Webpack 介绍 什么是 webpack&#xff1f; :::tip 官方描述 webpack 是一个用于现代 JavaScript 应用程序的静态模块打包工具。当 webpack 处理应用程序时&#xff0c;它会在内部从一个或多个入口点构建一个 依赖图(dependency graph)&#xff0c;然后将你项目中所需的每一个…