SpringBoot源码解析(二):引导上下文DefaultBootstrapContext

SpringBoot源码系列文章

SpringBoot源码解析(一):SpringApplication构造方法

SpringBoot源码解析(二):引导上下文DefaultBootstrapContext


目录

  • 前言
  • 一、入口
  • 二、DefaultBootstrapContext
    • 1、BootstrapRegistry接口
    • 2、BootstrapContext接口
    • 3、DefaultBootstrapContext实现类
  • 三、BootstrapRegistryInitializer
    • 1、作用及触发时机
    • 2、示例
  • 总结

前言

  前文深入解析SpringApplication构造方法,而接下来的几篇文章将重点介绍run方法的执行逻辑。

在这里插入图片描述

SpringBoot版本2.7.18的SpringApplication的run方法的执行逻辑如下,本文将详细介绍第一小节:创建引导上下文

// SpringApplication类方法
public ConfigurableApplicationContext run(String... args) {// 记录应用启动的开始时间long startTime = System.nanoTime();// 1.创建引导上下文,用于管理应用启动时的依赖和资源DefaultBootstrapContext bootstrapContext = createBootstrapContext();ConfigurableApplicationContext context = null;// 配置无头模式属性,以支持在无图形环境下运行// 将系统属性 java.awt.headless 设置为 trueconfigureHeadlessProperty();// 2.获取Spring应用启动监听器,用于在应用启动的各个阶段执行自定义逻辑SpringApplicationRunListeners listeners = getRunListeners(args);// 3.发布开始事件、通知ApplicationListener监听器listeners.starting(bootstrapContext, this.mainApplicationClass);try {// 4.解析应用参数ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);// 5.准备应用环境,包括读取配置文件和设置环境变量ConfigurableEnvironment environment = prepareEnvironment(listeners, bootstrapContext, applicationArguments);// 配置是否忽略 BeanInfo,以加快启动速度configureIgnoreBeanInfo(environment);// 6.打印启动BannerBanner printedBanner = printBanner(environment);// 7.创建应用程序上下文context = createApplicationContext();// 设置应用启动的上下文,用于监控和管理启动过程context.setApplicationStartup(this.applicationStartup);// 8.准备应用上下文,包括加载配置、添加 Bean 等prepareContext(bootstrapContext, context, environment, listeners, applicationArguments, printedBanner);// 9.刷新上下文,完成 Bean 的加载和依赖注入refreshContext(context);// 10.刷新后的一些操作,如事件发布等afterRefresh(context, applicationArguments);// 计算启动应用程序的时间,并记录日志Duration timeTakenToStartup = Duration.ofNanos(System.nanoTime() - startTime);if (this.logStartupInfo) {new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), timeTakenToStartup);}// 11.通知监听器应用启动完成listeners.started(context, timeTakenToStartup);// 12.调用应用程序中的 `CommandLineRunner` 或 `ApplicationRunner`,以便执行自定义的启动逻辑callRunners(context, applicationArguments);}catch (Throwable ex) {// 13.处理启动过程中发生的异常,并通知监听器handleRunFailure(context, ex, listeners);throw new IllegalStateException(ex);}try {// 14.计算应用启动完成至准备就绪的时间,并通知监听器Duration timeTakenToReady = Duration.ofNanos(System.nanoTime() - startTime);listeners.ready(context, timeTakenToReady);}catch (Throwable ex) {// 处理准备就绪过程中发生的异常handleRunFailure(context, ex, null);throw new IllegalStateException(ex);}// 返回已启动并准备就绪的应用上下文return context;
}

一、入口

// 1.创建引导上下文,用于管理应用启动时的依赖和资源
DefaultBootstrapContext bootstrapContext = createBootstrapContext();
  • bootstrapRegistryInitializers就是上一篇文章中在SpringApplication构造方法中创建的引导注册组件初始化器集合(查询spring.factories文件,没有找到BootstrapRegistryInitializer的实现类)
  • 调用初始化器的initialize方法,参数为bootstrapContext,也就是说每个初始化器都会对bootstrapContext进行必要的设置和准备(启动时需要的资源和依赖
  • 本方法是在run方法最开始调用的,也就是说引导注册组件初始化器组件的执行时机最早

  主要内容就是实例化DefaultBootstrapContext以及遍历BootstrapRegistryInitializer集合调用initialize,下面详细介绍下这两个类的作用。

// SpringApplication类属性方法// 引导注册初始化器
private List<BootstrapRegistryInitializer> bootstrapRegistryInitializers;private DefaultBootstrapContext createBootstrapContext() {// 创建一个 DefaultBootstrapContext 实例,用于管理应用启动时的资源和依赖DefaultBootstrapContext bootstrapContext = new DefaultBootstrapContext();// 遍历 bootstrapRegistryInitializers 集合中的每个 initializer,// 并调用它们的 initialize 方法,将 bootstrapContext 作为参数传入。// 这一步确保每个 initializer 都可以对 bootstrapContext 进行相应的配置,// 为应用程序的启动过程准备所需的资源。this.bootstrapRegistryInitializers.forEach((initializer) -> initializer.initialize(bootstrapContext));// 返回已完成初始化的 DefaultBootstrapContext 对象return bootstrapContext;
}

二、DefaultBootstrapContext

  DefaultBootstrapContext作为SpringBoot启动过程中的核心组件,负责环境配置资源管理生命周期管理,确保应用程序的顺利启动和运行。理解其作用有助于开发者更好地掌握SpringBoot的内部机制。

类图如下:

在这里插入图片描述

1、BootstrapRegistry接口

  一个简单的对象注册表,在启动和处理环境配置期间可用,直到ApplicationContext准备好为止。提供对单例的惰性访问,这些单例的创建成本可能很高,或者需要在ApplicationContext可用之前共享。

  注册表使用Class作为键,这意味着只能存储给定类型的单个实例

  addCloseListener(ApplicationListener)方法可用于添加监听器,当BootstrapContext关闭且ApplicationContext已准备好时,该监听器可以执行某些操作。例如,实例可以选择将自身注册为常规SpringBean,以便可供应用程序使用。

public interface BootstrapRegistry {// 注册特定类型到注册表。如果指定的类型已注册且未以单例形式获取,则将替换。<T> void register(Class<T> type, InstanceSupplier<T> instanceSupplier);// 如果尚未存在,则注册特定类型到注册表。<T> void registerIfAbsent(Class<T> type, InstanceSupplier<T> instanceSupplier);// 返回给定类型是否已经注册。<T> boolean isRegistered(Class<T> type);// 返回给定类型的已注册 {@link InstanceSupplier},如果没有则返回 null。<T> InstanceSupplier<T> getRegisteredInstanceSupplier(Class<T> type);// 添加 {@link ApplicationListener},当 {@link BootstrapContext} 关闭且// {@link ApplicationContext} 准备就绪时,将调用该监听器,并传递 {@link BootstrapContextClosedEvent}。void addCloseListener(ApplicationListener<BootstrapContextClosedEvent> listener);// 提供所需时创建实际实例的供应者。@FunctionalInterfaceinterface InstanceSupplier<T> {// 工厂方法,在需要时创建实例。T get(BootstrapContext context);// 返回所提供实例的作用域。default Scope getScope() {return Scope.SINGLETON;}// 返回一个具有更新 {@link Scope} 的新 {@link InstanceSupplier}。default InstanceSupplier<T> withScope(Scope scope) {Assert.notNull(scope, "Scope must not be null");InstanceSupplier<T> parent = this;return new InstanceSupplier<T>() {@Overridepublic T get(BootstrapContext context) {return parent.get(context);}@Overridepublic Scope getScope() {return scope;}};}// 工厂方法,用于为给定实例创建 {@link InstanceSupplier}。static <T> InstanceSupplier<T> of(T instance) {return (registry) -> instance;}// 工厂方法,用于从 {@link Supplier} 创建 {@link InstanceSupplier}。static <T> InstanceSupplier<T> from(Supplier<T> supplier) {return (registry) -> (supplier != null) ? supplier.get() : null;}}// 实例的作用域。enum Scope {// 单例实例。 {@link InstanceSupplier} 将仅被调用一次,并且每次都将返回相同的实例。SINGLETON,// 原型实例。 {@link InstanceSupplier} 将在每次需要实例时调用。PROTOTYPE}
}

总结:用于注册引导阶段的组件,在应用启动时通过register方法动态添加对象

2、BootstrapContext接口

  一个简单的引导上下文,在启动和处理环境配置期间可用,直到ApplicationContext准备好为止。提供对单例的惰性访问,这些单例的创建成本可能很高,或者需要在ApplicationContext可用之前共享。

public interface BootstrapContext {// 如果类型已注册,则从上下文中返回实例。如果之前未访问过该实例,则会创建该实例<T> T get(Class<T> type) throws IllegalStateException;// 如果类型已注册,则返回上下文中的实例。如果尚未访问该实例,则将创建它。// 如果类型未注册,则返回指定的替代实例。<T> T getOrElse(Class<T> type, T other);// 如果类型已注册,则返回上下文中的实例。如果尚未访问该实例,则将创建它。// 如果类型未注册,则使用指定的供应者提供的实例。<T> T getOrElseSupply(Class<T> type, Supplier<T> other);// 如果类型已注册,则返回上下文中的实例。如果尚未访问该实例,则将创建它。// 如果类型未注册,则抛出由供应者提供的异常。<T, X extends Throwable> T getOrElseThrow(Class<T> type, Supplier<? extends X> exceptionSupplier) throws X;// 返回给定类型是否存在注册<T> boolean isRegistered(Class<T> type);}

总结:用于提供对引导阶段注册组件的只读访问,一旦BootstrapRegistry注册完成并构建成BootstrapContext,所有组件可以通过get方法被安全地访问,直到应用启动完成。

3、DefaultBootstrapContext实现类

  • ConfigurableBootstrapContext是一个空接口,所以直接看核心内容DefaultBootstrapContext
public interface ConfigurableBootstrapContext extends BootstrapRegistry, BootstrapContext {
}

  DefaultBootstrapContext是一个实现了ConfigurableBootstrapContext接口的类,主要用于管理应用启动过程中的实例供应者实例,提供了注册、获取、关闭监听等功能。以下是主要功能的总结:

  1. 实例供应者管理
    • 使用Map<Class<?>, InstanceSupplier<?>> instanceSuppliers来存储类型到实例供应者的映射,可以通过registerregisterIfAbsent方法来注册实例供应者
    • register方法支持覆盖现有的实例供应者,而registerIfAbsent则仅在类型未注册的情况下注册实例供应者
  2. 实例管理
    • 使用Map<Class<?>, Object> instances存储已创建的实例
    • 提供getgetOrElsegetOrElseSupply等方法来获取实例。如果实例尚未创建,则调用相应的实例供应者来创建实例,并在单例作用域下将其存储
  3. 事件管理(后面文章调用时候细讲)
    • 使用ApplicationEventMulticaster来管理事件的发布和监听
    • 提供addCloseListener方法添加关闭监听器,并在close方法中发布BootstrapContextClosedEvent事件
public class DefaultBootstrapContext implements ConfigurableBootstrapContext {// 存储实例供应者的映射private final Map<Class<?>, InstanceSupplier<?>> instanceSuppliers = new HashMap<>();// 存储已创建实例的映射private final Map<Class<?>, Object> instances = new HashMap<>();// 事件广播器,用于发布应用事件private final ApplicationEventMulticaster events = new SimpleApplicationEventMulticaster();@Overridepublic <T> void register(Class<T> type, InstanceSupplier<T> instanceSupplier) {// 注册特定类型的实例供应者register(type, instanceSupplier, true);}@Overridepublic <T> void registerIfAbsent(Class<T> type, InstanceSupplier<T> instanceSupplier) {// 如果尚未注册,则注册特定类型的实例供应者register(type, instanceSupplier, false);}private <T> void register(Class<T> type, InstanceSupplier<T> instanceSupplier, boolean replaceExisting) {// 检查类型和实例供应者是否为空Assert.notNull(type, "Type must not be null");Assert.notNull(instanceSupplier, "InstanceSupplier must not be null");synchronized (this.instanceSuppliers) {// 检查类型是否已注册boolean alreadyRegistered = this.instanceSuppliers.containsKey(type);if (replaceExisting || !alreadyRegistered) {// 确保实例尚未创建Assert.state(!this.instances.containsKey(type), () -> type.getName() + " has already been created");// 注册实例供应者this.instanceSuppliers.put(type, instanceSupplier);}}}@Overridepublic <T> boolean isRegistered(Class<T> type) {// 检查给定类型是否已注册synchronized (this.instanceSuppliers) {return this.instanceSuppliers.containsKey(type);}}@Override@SuppressWarnings("unchecked")public <T> InstanceSupplier<T> getRegisteredInstanceSupplier(Class<T> type) {// 返回已注册的实例供应者synchronized (this.instanceSuppliers) {return (InstanceSupplier<T>) this.instanceSuppliers.get(type);}}@Overridepublic <T> T get(Class<T> type) throws IllegalStateException {// 获取指定类型的实例,如果未注册则抛出异常return getOrElseThrow(type, () -> new IllegalStateException(type.getName() + " has not been registered"));}@Overridepublic <T> T getOrElse(Class<T> type, T other) {// 获取指定类型的实例,如果未注册则返回其他提供的实例return getOrElseSupply(type, () -> other);}@Overridepublic <T> T getOrElseSupply(Class<T> type, Supplier<T> other) {// 尝试获取指定类型的实例,如果未注册则调用其他供应者synchronized (this.instanceSuppliers) {InstanceSupplier<?> instanceSupplier = this.instanceSuppliers.get(type);return (instanceSupplier != null) ? getInstance(type, instanceSupplier) : other.get();}}@Overridepublic <T, X extends Throwable> T getOrElseThrow(Class<T> type, Supplier<? extends X> exceptionSupplier) throws X {// 尝试获取指定类型的实例,如果未注册则抛出由供应者提供的异常synchronized (this.instanceSuppliers) {InstanceSupplier<?> instanceSupplier = this.instanceSuppliers.get(type);if (instanceSupplier == null) {throw exceptionSupplier.get();}return getInstance(type, instanceSupplier);}}@SuppressWarnings("unchecked")private <T> T getInstance(Class<T> type, InstanceSupplier<?> instanceSupplier) {// 获取实例,如果尚未创建则调用实例供应者T instance = (T) this.instances.get(type);if (instance == null) {instance = (T) instanceSupplier.get(this);// 如果作用域为单例,则存储该实例if (instanceSupplier.getScope() == Scope.SINGLETON) {this.instances.put(type, instance);}}return instance;}@Overridepublic void addCloseListener(ApplicationListener<BootstrapContextClosedEvent> listener) {// 添加关闭监听器,当 BootstrapContext 关闭时触发this.events.addApplicationListener(listener);}/*** 当 {@link BootstrapContext} 关闭且 {@link ApplicationContext} 已准备好时调用的方法。* @param applicationContext 已准备好的上下文*/public void close(ConfigurableApplicationContext applicationContext) {// 发布 BootstrapContext 关闭事件this.events.multicastEvent(new BootstrapContextClosedEvent(this, applicationContext));}
}

三、BootstrapRegistryInitializer

1、作用及触发时机

  • 回调接口,用于在BootstrapRegistry(对象注册表)使用之前进行初始化
  • 作用:应用程序启动的早期阶段进行必要的初始化配置
@FunctionalInterface
public interface BootstrapRegistryInitializer {// 此方法在应用启动过程中被调用,允许实现者向注册表注册必要的组件或服务。// 注册的组件随后可以在应用上下文中访问。实现者应仅注册应用所需的类型。void initialize(BootstrapRegistry registry);
}

  引导注册组件初始化器BootstrapRegistryInitializer在SpringApplication的构造方法中通过查找META-INF/spring.factories文件进行加载,然后在引导上下文实例创建完成后,遍历并调用所有BootstrapRegistryInitializer#initialize方法。

在这里插入图片描述

普通SpringBoot项目没有其实现,找了个SpringCloud项目看了下,有两个

在这里插入图片描述

2、示例

自定义BootstrapRegistryInitializer

public class MyBootstrapRegistryInitializer implements BootstrapRegistryInitializer {@Overridepublic void initialize(BootstrapRegistry registry) {// 注册一个自定义服务registry.register(MyCustomService.class, context -> new MyCustomService());System.out.println("MyBootstrapRegistryInitializer已注册");}
}
class MyCustomService {
}

META-INF/spring.factories文件中添加对自定义初始化器的配置

org.springframework.boot.BootstrapRegistryInitializer=com.xc.config.MyBootstrapRegistryInitializer

启动服务

在这里插入图片描述

ps:BootstrapRegistryInitializer是SpringBoot第一个扩展点(注册组件)

总结

  • 引导上下文DefaultBootstrapContext创建:在run方法的最初阶段被实例化,并通过BootstrapRegistryInitializer(第一个注册组件扩展点)进行必要的初始化,确保应用启动时所需的资源和依赖得到妥善管理
  • BootstrapRegistry:该接口作为对象注册表,允许在应用启动早期阶段进行组件的注册和管理,提供了对高成本实例的惰性访问
  • BootstrapContext:作为引导上下文的只读访问接口,它确保注册的组件能够安全、可靠地在应用上下文准备好之前被访问

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

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

相关文章

运维高可用架构设计

一、硬件 1、服务器 2、网络架构 二、软件 1、基础组件 组件名称 高可用方式 最少节点数 负载均衡(Tenginx) corsyncpacemaker互为主备 多组集群通过DNS轮循实现一个大集群 2DNS主从集群2RabbitMQ原生HA镜像集群3Zookeeper原生分布式集群3Kafka原生分布式集群3ES原生分布式集…

C++之vector类的模拟实现

片头 嗨~小伙伴们&#xff0c;今天我们来一起学习关于C的vector类的模拟实现&#xff0c;准备好了吗&#xff1f;咱们开始咯~ 一、基本框架 namespace bit {template<class T>class vector {public:typedef T* iterator;typedef const T* const_iterator;// 针对const修…

MyBatis 返回 Map 或 List<Map>时,时间类型数据,默认为LocalDateTime,响应给前端默认含有‘T‘字符

一、问题 MyBatis 返回 Map 或 List时&#xff0c;时间类型数据&#xff0c;默认为LocalDateTime Springboot 响应给前端的LocalDateTime&#xff0c;默认含有’T’字符&#xff0c;如何统一配置去掉 二、解决方案 1、pom.xml 增加依赖&#xff08;2024.11.6 补充&#xff…

数据结构之二叉树前序,中序,后序习题分析(递归图)

1.比较相同的树 二叉树不能轻易用断言&#xff0c;因为树一定有空 2.找结点值 3.单值二叉树 4.对称二叉树 5.前序遍历

如何使用gewe开发微信机器人

[Gewe](微信管理系统)&#xff0c;个人微信**开源框架&#xff0c;支持二次开发、任意语言都可接入&#xff0c;Restful API接入。 gewe框架优势&#xff1a; - 简单易用&#xff0c;无接入难度&#xff0c;区别于其它开源项目&#xff0c;本框架无需用户安装电脑微信&#x…

vue3 基于element-plus进行的一个可拖动改变导航与内容区域大小的简单方法

1、先上个截图&#xff1a; 说明&#xff1a;拖动上面的分隔栏就可以实现&#xff0c;改变左右区域的大小。 2、上面的例子来自官网的&#xff1a; Container 布局容器 | Element Plus 3、拖动的效果来自&#xff1a; https://juejin.cn/post/7029640316999172104#heading-1…

Excel 无法打开文件

Excel 无法打开文件 ‘新建 Microsoft Excel 工作表.xlsx",因为 文件格式或文件扩展名无效。请确定文件未损坏&#xff0c;并且文件扩展名与文件的格式匹配。

K8S node节点没有相应的pod镜像运行故障处理办法

查看从节点状态 kubectl describe node k8s-node1以下是报错提示 解决办法 需要处理node1节点上的磁盘空间&#xff0c;磁盘空间需要在85%内 处理后的状态 处理正常

使用代理时Stable Diffusion无法正常下载各类模型的解决办法

最近发现了 Stable Diffusion 这个好玩的ai绘画工具&#xff0c;不得不感叹现在ai工具已经进化到这么简单易用的程度&#xff0c;只要下载对应的模型就可以生成各种有意思的图片 就算你没有编程基础&#xff0c;跟着教程也能弄出来 不过使用过程中发现部分功能无法使用 查看日…

GODOT 4 不用scons编译cpp扩展的方法

以terrain3d插件&#xff0c;Godot_v4.3 为例&#xff1a; 下载下来&#xff0c;先用scons编译一遍通过后&#xff0c;整个占用1GB&#xff0c;obj文件都生成在源码旁边&#xff0c;够乱。 scons 是跨平台的构建工具&#xff0c;但是需要需要写python脚本。流程比较莫名其妙…

Python 学习完基础语法知识后,如何进一步提高?

入门Python后&#xff0c;就可以拿些小案例练手了&#xff0c;这时候千万不要傻乎乎地成天啃语法书。 编程是一门实践的手艺&#xff0c;讲究孰能生巧。不管是去手撸算法、或者照葫芦画瓢写几个小游戏都可以让你的Python突飞猛进。 之前看github比较多&#xff0c;推荐给大家…

基于Java的简单图书管理系统的实现(增删改查)

基于Java的简单图书管理系统的实现&#xff08;增删改查&#xff09; package com.situ.lib;public class Book {//对象&#xff1a;书-----定义书的属性:private String name;private String isbn;private String author;private double price;//无参构造方法&#xff1a;pub…

C语言必做30道练习题

C语言练习30题&#xff08;分支循环&#xff0c;数组&#xff0c;函数&#xff0c;递归&#xff0c;操作符&#xff09; 目录 分支循环1.闰年的判断2.阅读代码&#xff0c;计算代码输出的结果3.输入一个1~7的数字&#xff0c;打印对应的星期几4.输入任意一个整数值&#xff0c;…

tp接口 入口文件 500 错误原因

一、描述 二、可能的原因 1、runtime目录没权限 2、关闭了Tp记录日志的功能 3、关闭debug调试模式 4、关闭了debug模式还是报错 一、描述 Thinkphp项目本地正常&#xff0c;上传到线上后静态文件访问正常&#xff0c;访问tp接口报500错误。 经调试发现&#xff0c;在php入…

思源笔记轻松连接本地Ollama大语言模型,开启AI写作新体验!

文章目录 前言1. 下载运行Ollama框架2. Ollama下载大语言模型3. 思源笔记设置连接Ollama4. 测试笔记智能辅助写作5. 安装Cpolar工具6. 配置Ollama公网地址7. 笔记设置远程连接Ollama8. 固定Ollama公网地址 前言 今天我们要聊聊如何通过cpolar内网穿透技术&#xff0c;把国产笔…

CAS 详解

Java 中 CAS 是如何实现的&#xff1f; 在 Java 中&#xff0c;实现 CAS&#xff08;Compare-And-Swap, 比较并交换&#xff09;操作的一个关键类是Unsafe。 Unsafe类位于sun.misc包下&#xff0c;是一个提供低级别、不安全操作的类。由于其强大的功能和潜在的危险性&#xf…

九识智能与徐工汽车达成战略合作,共绘商用车未来新蓝图

近日&#xff0c;九识智能与徐工汽车签署战略合作协议&#xff0c;标志着双方在智能驾驶技术与新能源商用车融合应用、联合生产及市场推广等方面迈入深度合作的新篇章&#xff0c;将共同引领智能驾驶技术商业化浪潮。 近年来&#xff0c;在国家智能化发展战略的引领下&#xff…

【vue2.7.16系列】手把手教你搭建后台系统__登录使用状态管理(15)

使用store进行登录信息管理 其实就是把登录放到vuex的actions中去执行&#xff0c;然后保存用户信息、权限等 在store/modules/account.js中添加如下代码&#xff1a; import { login, logout, getInfo, menusApi } from /api/account; // getExpiresTime import {getToken,s…

sql报错信息将字符串转换为 uniqueidentifier 时失败

报错信息&#xff1a; [42000] [Microsoft][SQL Server Native Client 10.0][SQL Server]将字符串转换为 uniqueidentifier 时失败 出错行如下&#xff1a; 表A.SourceCode 表B.ID 出错原因&#xff1a; SourceCode是nvarchar,但ID是uniqueidentifier 数据库查询字段和类…

「Mac畅玩鸿蒙与硬件22」鸿蒙UI组件篇12 - Canvas 组件的动态进阶应用

在鸿蒙应用中&#xff0c;Canvas 组件可以实现丰富的动态效果&#xff0c;适合用于动画和实时更新的场景。本篇将介绍如何在 Canvas 中实现动画循环、动态进度条、旋转和缩放动画&#xff0c;以及性能优化策略。 关键词 Canvas 组件动态绘制动画效果动态进度条旋转和缩放性能优…