深入解析:Spring Boot 深入剖析:SpringApplicationRunListener
在研究SpringBoot的启动源码的时候我们看到run方法中有一个之前没见过的类SpringApplicationRunListeners 类,那么这个类的主要作用是什么呢?接下来我们来解析一下这个类
public ConfigurableApplicationContext run(String... args) {
、、、、、、
SpringApplicationRunListeners listeners = getRunListeners(args);
listeners.starting(bootstrapContext, this.mainApplicationClass);
、、、、、、
}
SpringApplicationRunListeners
当我们点开SpringApplicationRunListeners查看其源码的时候我们发现,SpringApplicationRunListeners内部含有一个集合用来存储SpringApplicationRunListener类对象。同时大部分方法的模式都是调用doWithListeners方法,区别在于传递的参数不同。其实这些大部分的方法都是事件发布的方法,而他的那个集合SpringApplicationRunListener对象则是事件监听对象,因此这个类整体的作用我们就有了一个大概的了解--当内部的不同方法被调用的时候则会发布事件给SpringApplicationRunListener对象,而SpringApplicationRunListener对象收到事件后则会做出对应的处理
class SpringApplicationRunListeners {
private final Log log;
private final List listeners;
private final ApplicationStartup applicationStartup;
SpringApplicationRunListeners(Log log, List listeners,
ApplicationStartup applicationStartup) {
this.log = log;
this.listeners = List.copyOf(listeners);
this.applicationStartup = applicationStartup;
}
void starting(ConfigurableBootstrapContext bootstrapContext, Class mainApplicationClass) {
doWithListeners("spring.boot.application.starting", (listener) -> listener.starting(bootstrapContext),
(step) -> {
if (mainApplicationClass != null) {
step.tag("mainApplicationClass", mainApplicationClass.getName());
}
});
}
void environmentPrepared(ConfigurableBootstrapContext bootstrapContext, ConfigurableEnvironment environment) {
doWithListeners("spring.boot.application.environment-prepared",
(listener) -> listener.environmentPrepared(bootstrapContext, environment));
}
void contextPrepared(ConfigurableApplicationContext context) {
doWithListeners("spring.boot.application.context-prepared", (listener) -> listener.contextPrepared(context));
}
void contextLoaded(ConfigurableApplicationContext context) {
doWithListeners("spring.boot.application.context-loaded", (listener) -> listener.contextLoaded(context));
}
、、、、、、
private void doWithListeners(String stepName, Consumer listenerAction) {
doWithListeners(stepName, listenerAction, null);
}
private void doWithListeners(String stepName, Consumer listenerAction,
Consumer stepAction) {
StartupStep step = this.applicationStartup.start(stepName);
this.listeners.forEach(listenerAction);
if (stepAction != null) {
stepAction.accept(step);
}
step.end();
}
}
下面我们挑两个主要方法来进行解析
void environmentPrepared(ConfigurableBootstrapContext bootstrapContext, ConfigurableEnvironment environment) {
doWithListeners("spring.boot.application.environment-prepared",
(listener) -> listener.environmentPrepared(bootstrapContext, environment));
}
可以看到大部分的事件发布方法本质都是调用了doWithListeners,同时传递的第一个参数的事件类型不同标志着不同事件的触发,第二个参数则是使用了函数接口生成一个类对象传递给这个方法。
整个类的关键则是在第二个方法,doWithListeners首先则是采用了StartupStep 统计每个步骤的耗时,然后利用了 Java 8 的函数式编程特性遍历了listeners集合随后调用了函数接口listenerAction的方法,目的就是批量触发所有监听器的某个事件方法。而这个方法我们在将解析上一个方法源码的时候已经看到了是调用的则是SpringApplicationRunListener的内部方法。
private void doWithListeners(String stepName, Consumer listenerAction) {
doWithListeners(stepName, listenerAction, null);
}
private void doWithListeners(String stepName, Consumer listenerAction,
Consumer stepAction) {
StartupStep step = this.applicationStartup.start(stepName);
this.listeners.forEach(listenerAction);
if (stepAction != null) {
stepAction.accept(step);
}
step.end();
}
所以本质上SpringApplicationRunListeners的作用则是发布事件,遍历当前类集合中每个SpringApplicationRunListener并且调用其对应这个事件的方法来实现事件的发布。
是 Spring Boot 应用启动过程的“广播站”,专门发布与应用生命周期(如开始启动、环境准备、上下文准备等)紧密相关的固定事件。而相较于ApplicationEventMulticaster
对于SpringApplicationRunListeners来说则更像是一个事件生产者。当我们分析完SpringApplicationRunListener源码便能更加深刻的理解
SpringApplicationRunListener
SpringApplicationRunListener
是一个接口,它定义了用于监听 SpringApplication
的 run()
方法执行过程中各个关键生命周期节点的回调方法。它的实现类通过这些回调方法,在特定的时间点执行必要的逻辑(最主要的就是发布事件)。
为什么说SpringApplicationRunListener
是发布事件呢?我们看看SpringApplicationRunListener
实现类的源码便可知道为啥时发布事件
class EventPublishingRunListener implements SpringApplicationRunListener, Ordered {
private final SpringApplication application;
private final String[] args;
private final SimpleApplicationEventMulticaster initialMulticaster;
EventPublishingRunListener(SpringApplication application, String[] args) {
this.application = application;
this.args = args;
this.initialMulticaster = new SimpleApplicationEventMulticaster();
}
@Override
public int getOrder() {
return 0;
}
@Override
public void starting(ConfigurableBootstrapContext bootstrapContext) {
multicastInitialEvent(new ApplicationStartingEvent(bootstrapContext, this.application, this.args));
}
@Override
public void environmentPrepared(ConfigurableBootstrapContext bootstrapContext,
ConfigurableEnvironment environment) {
multicastInitialEvent(
new ApplicationEnvironmentPreparedEvent(bootstrapContext, this.application, this.args, environment));
}
、、、、、、
private void multicastInitialEvent(ApplicationEvent event) {
refreshApplicationListeners();
this.initialMulticaster.multicastEvent(event);
}
private void refreshApplicationListeners() {
this.application.getListeners().forEach(this.initialMulticaster::addApplicationListener);
}
private static class LoggingErrorHandler implements ErrorHandler {
private static final Log logger = LogFactory.getLog(EventPublishingRunListener.class);
@Override
public void handleError(Throwable throwable) {
logger.warn("Error calling ApplicationEventListener", throwable);
}
}
}
EventPublishingRunListener 是 SpringApplicationRunListener的一个子类实现类,我们可以看到其内部则是有一个SimpleApplicationEventMulticaster 类属性这个属性本质上是ApplicationEventMulticaster类的子类,而ApplicationEventMulticaster类则ApplicationContext内部最为常见的事件广播器主要作用则是监听事件并且广播事件。
当我们我们查看SpringApplicationRunListener不同的方法源码则可以看到不同的方法生成了不同的event事件类型当做参数,最终都是调用的multicastEvent方法,而multicastEvent方法内部的核心则是使用SimpleApplicationEventMulticaster的属性来对当前不同的event事件进行广播处理,最终本质上其实是ApplicationListener类来进行监听并且处理的
特性 | SpringApplicationRunListener | ApplicationListener |
---|---|---|
关注点 | Spring Boot 应用的启动生命周期。 | Spring 框架的应用事件(包括Boot启动事件)。 |
触发时机 | 在 SpringApplication.run() 方法的固定节点被直接调用。 | 在任何事件被发布时,由 ApplicationEventMulticaster 通知。 |
获取方式 | 通过 spring.factories 机制配置。 | 可以是 @Component ,或通过 SpringApplication.addListeners() 添加。 |
作用域 | 仅限于启动过程。 | 贯穿整个应用上下文生命周期。 |
主要职责 | 生产/发布启动事件。 | 消费/处理事件(包括启动事件)。 |
因此我们发现其实SpringApplicationRunListener
是一个时间的发布器,通过ApplicationEventMulticaster来对事件进行广播,最终由ApplicationListener监听到不同的事件来进行处理,而SpringApplicationRunListeners
我们则可以认为他是一个事件生产器,生产完毕之后调用SpringApplicationRunListener
的方法来进行事件广播最终由ApplicationListener来进行事件的处理。
相关文章:
深入解析:Spring Boot 深入剖析:SpringApplicationRunListener
深入解析:Spring Boot 深入剖析:SpringApplicationRunListenerpre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New&qu…...
Hutool 调用第三方接口报错
报错信息 java.lang.reflect.InaccessibleObjectException: Unable to make field protected java.lang.String java.net.HttpURLConnection.method accessible: module java.base does not "opens java.net" to unnamed module @28c4711cat java.base/java.lang.ref…...
丑东西经济学:全面分析
```html丑东西经济学:全面分析<script src="https://cdn.tailwindcss.com"></script>:root { --primary: #8B5A3C; --secondary: #D4A574; --accent: #2E1F15; --neutral: #F5F1EC; --base-100: #FEFCF8; --base-content: #1A1A1A } body { font-family…...
Python turtle 海龟画图入门指南
Python海龟画图入门指南 Python海龟画图(Turtle Graphics)是一个非常适合编程初学者的图形绘制库,它基于Logo编程语言的概念,通过控制一个"海龟"在屏幕上移动来绘制图形。 海龟画图基础知识 1. 基本概念海龟(Turtle):一个可以在画布上移动的箭头(默认形状为箭头,…...
uni-app中v-if使用”异常”
在vue中如果想实现通过某些条件渲染不同内容,其中使用比较多的就是v-if,它会根据你传入的东西是否为真值来决定要不要显示当前内容块,但是最近在uni-app中使用的时候发现了一点小问题,一起看看是什么问题吧。前情 最近又接手一个全新多端项目,包括抖音/快手/微信/支付宝/安…...
二叉树遍历
1.广度优先遍历 2.深度优先遍历 1.前序遍历 根结点--左子树--右子树 void pre(int p){printf("%d\n",p);if(tree[p].l)pre(tree[p].l);if(tree[p].r)pre(tree[p].r); }2.中序遍历 左子树--根结点--右子树 void in(int p){if(tree[p].l)in(tree[p].l);printf("%d…...
Python Socket网络编程(3)
class类里面先执行__new__,然后__init__ 单例模式线程安全单例模式点击查看代码 import threadingclass Singleton:instance = Nonelock = threading.Rlock()def __new__(cls, *args, **kwargs):if cls.instance:return cls.instancewith cls.lock:if cls.instance:return cls…...
实用指南:有关gitlab14.x版本在内网环境下无法添加webhooks的解决方法
实用指南:有关gitlab14.x版本在内网环境下无法添加webhooks的解决方法pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier …...
如何创建可引导的 macOS Tahoe 安装介质
如何创建可引导的 macOS Tahoe 安装介质如何创建可引导的 macOS 安装介质 如何创建可引导的 macOS 安装器 | 如何制作 macOS USB 启动盘 请访问原文链接:https://sysin.org/blog/macos-createinstallmedia/ 查看最新版。原创作品,转载请保留出处。 作者主页:sysin.orgmacOS …...
强类型、类型安全
说 C++ 是类型安全的强类型语言,这句话其实不完全准确。更精确的说法是,C++ 是一门静态类型、强类型但并非完全类型安全的语言。 我们先来理解一下你提到的两个概念,以及它们在C++中的具体体现。 强类型(Strongly-typed) 强类型主要指的是,在语言中,不同类型的数据之间操…...
完整教程:数据结构——逻辑结构物理结构
完整教程:数据结构——逻辑结构&物理结构pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", monospace !i…...
前端面试
题目 JS和TS区别类型JS:动态类型,变量类型在运行时确定,无需显示声明类型。 TS:静态类型,支持类型注解和编译时类型检查,可提前发现错误。编译与运行JS:直接由浏览器或者node.js,无需编译 TS:需要通过编译器(tsc)转换为JS代码后才能运行。语法扩展JS:仅支持基础的ECMAScrip…...
外置Tomcat启动Springboot项目后,请求参数中文乱码的问题解决 - yjry
首先按照常规的排查流程逐个进行配置,都是无效果的: 1、检查了tomcat的conf/server.xml里Connector标签有无配置URIEncoding="UTF-8",添加了配置,无效; 2、tomcat/bin/catalina.bat,添加了JAVA_OPTS="$JAVA_OPTS -Dfile.encoding=UTF-8"配置,无效;…...
JSON字符串转换List对象列表 JSONArray toJavaList
JSON字符串转换List对象列表 JSONArray toJavaListJSON字符串转换List对象列表 JSONArray toJavaListpackage com.example.core.mydemo.java3.jsonDemo;import com.alibaba.fastjson.JSONArray; import com.alibaba.fastjson.JSONObject; import com.example.core.mydemo.json2…...
gradle项目多模块中主模块加载子模块中的sqlmapper文件方法
gradle项目多模块中主模块加载子模块中的sqlmapper文件方法gradle项目多模块中主模块加载子模块中的sqlmapper文件方法 1.子模块 build.gradlesourceSets.main.resources { srcDirs = ["src/main/resources", "src/main/java"] // 包含Java目录下的XML…...
MCP - 使用 fastmcp 编写 Client 调用 MCP Serverr - Streamable HTTP (四)
xx源代码: mcp_server_tool_registry.zip...
全面理解MySQL架构
一条SQL查询执行的流程客户端 连接工具(Navacat,SQLyog,JDBC)都是客户端,主要用于发送执行SQL语句的请求 服务端 服务端可以分为server层和存储引擎层 server层 负责处理SQL语句, 解析, 优化, 缓存等. 负责权限管理, 用户认证等 提供了复制, 备份, 恢复等高级功能 Server层…...
Figma EX 125.7.5 UI原型设计
** 描述** Figma是一款专业的UI原型设计软件,Figma能够帮助团队创建、测试和发布更好的设计,Figma支持Windows和macOS 等多个平台,软件注意自适应屏幕大小、历史记录、导入Sketch文件等功能。它是一个真正为设计师而开发的设计软件,软件可以让网络设计变得简单。它能够创建…...
啥是CPU
啥是CPU?CPU即为中央处理器,一般决定CPU性能的参数有:频率,核心,线程,指令集频率决定了在相同架构下单CPU核心的运行速度,#核心数量一般决定了CPU处理多任务的能力,现在的绝大部分CPU都是多个核心组合成的线程则决定了核心和内存交换数据的速度某些CPU可能为了降低功耗…...
C# Avalonia 15- Animation- CodeAnimation
C# Avalonia 15- Animation- CodeAnimation例子展现了两种设置动画的方法,各位自行选择。CodeAnimation.axaml代码<Window xmlns="https://github.com/avaloniaui"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:d="http://schema…...
vue3 使用 docx-preview 预览 Word文档
引入第三方库npm i docx-previewvue3创建组件<template><div><!-- 用于渲染文档的容器 --><div ref="previewContainer" class="preview-wrapper"></div></div> </template><script setup>import {define…...
数据库原理-第三章——SQL
Structured Query Language数据查询 Select数据操纵 Insert、Delete、Update数据定义 Create、Drop、Alter数据控制 Grant、Revoke事务处理 begin transaction、commit、rollback指针、游标控制(CCL) DECLARE CURSOR、FETCH INTO、UPDATE、WHERE CURRENT数据库的数据类型数值…...
ubuntu 18.04安装mysql 8.0.41
环境Os:ubuntu 18.04 desktop桌面版mysql:mysql:8.0.41 glibc2.17 查看操作系统信息root@db:~# ldd --version ldd (Ubuntu GLIBC 2.27-3ubuntu1) 2.27 Copyright (C) 2018 Free Software Foundation, Inc. This is free software; see the source for copying conditions. Th…...
Topaz Photo AI Pro 4.0.4 AI图片智能降噪(win版)
描述 Topaz Photo AI是一款专业的人工智能图片降噪软件,得益于Topaz公司AI算法,这款照片修复软件可以在修复照片的同时运用人工智能算法AI模型计算图片模糊部分,自动修复图片受损的细节,以增强图片画质。利用未来的技术锐化、消除噪点并提高照片的分辨率。Topaz Photo AI 可…...
阿里云基础设施 AI Tech Day AI 原生,智构未来——AI 原生架构与企业实践专场
本次技术沙龙聚焦企业 AI 应用规模化实战痛点分享 Serverless、AI 网关、可观测、AI 消息队列产品解决方案和企业真实实践,与企业共同探讨如何通过向更先进的架构演进来适应 AI 时代的快速变化,为迎接新的增长机会做好准备。活动简介云为 AI 提供了坚实的基础设施支撑。聚焦阿…...
实用指南:LINUX910 CENTOS8 新建虚拟机;重设root密码/时间同步
实用指南:LINUX910 CENTOS8 新建虚拟机;重设root密码/时间同步pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New&qu…...
零基础学习PYthon记录
变量会变化的 , 描述事物的状态 变量接收某个值定义变量 变量名 赋值符号 变量值 name = "david" print(name)不能使用关键字定义变量 定义的变量要有某种意义(看到变量名就要知道是干什么用的) 用下划线区分两个单词david_height = 180 print(david_height)常量固定…...
C++ std::unordered_set
std::unordered_set 是 C++ STL 中的无序关联容器,其核心特性是存储唯一元素且不保证元素顺序,底层基于哈希表实现,因此插入、查找、删除操作的平均时间复杂度为 O(1)。与 std::set(红黑树实现)相比,它更适合对查找速度有高要求且无需元素有序的场景。 1、底层数据结构与…...
如何将一个项目同时提交到GitHub和Gitee(码云)上
要将本地的一个项目同时提交到 GitHub 和 Gitee,可以通过配置多个远程仓库来实现,以下是具体步骤: 前提条件 确保你已经在 GitHub 和 Gitee 上分别创建了空的远程仓库,并且本地已经安装配置好了 Git,能够正常使用 Git 命令行进行操作。 操作步骤初始化本地仓库(如果还未初…...
基于Matlab的LeNet-5车牌字符识别系统实现
一、网络结构改进方案输入层调整输入尺寸:将LeNet-5原始输入尺寸(3232)调整为车牌字符标准尺寸(2828或4848) 通道扩展:增加颜色通道(RGB→灰度需归一化,或保留RGB通道)卷积层优化 layers = [imageInputLayer([28 28 1]) % 输入层convolution2dLayer(5, 6, Padding, sa…...
MATLAB的交通标志牌识别实现
MATLAB的交通标志牌识别实现,使用了颜色分割、形态学处理、模板匹配等核心技术一、核心识别流程设计图像预处理 灰度化与去噪 色彩空间转换(RGB→HSV/Lab) 颜色阈值分割(红色/蓝色/黄色区域提取) 几何特征提取 边缘检测(Canny算子) 形态学操作(腐蚀+膨胀) 轮廓分析(筛…...
Python常见的数据结构和代码示例
概述 Python 中常见的数据结构可以分为内置数据结构和扩展数据结构(主要来自第三方库)两类,以下是详细介绍: 一、内置数据结构(Python 自带) 这些是 Python 解释器原生支持的数据结构,无需额外导入模块。 1. 列表(List)特点:有序、可变(可修改)、允许重复元素,可存…...
Grafana 中文入门教程 | 构建你的第一个仪表盘
在大厂工作久了,时常对一些工具的存在觉得理所当然。 比如说,需要计算资源的时候,一个配置文件就可以要来两百台虚拟化好的机子。需要试下缓存?点下鼠标就可以要到几十个配置好的 Redis 结点。 最省心的是,这些工具都已经根据工作流配置好了:鉴权、优化、网络连接等等通通…...
Gitee DevOps:中国开发者效率革命的数字引擎
Gitee DevOps:中国开发者效率革命的数字引擎 在数字化转型浪潮席卷全球的当下,中国软件产业正面临前所未有的效率挑战。传统开发模式中的人工干预、碎片化工具链和低效协作已成为制约企业创新速度的关键瓶颈。Gitee DevOps作为本土领先的一站式开发运维平台,通过全流程自动化…...
Topaz Photo AI Pro 4.0.4 AI图片智能降噪
描述 Topaz Photo AI是一款专业的人工智能图片降噪软件,得益于Topaz公司AI算法,这款照片修复软件可以在修复照片的同时运用人工智能算法AI模型计算图片模糊部分,自动修复图片受损的细节,以增强图片画质。利用未来的技术锐化、消除噪点并提高照片的分辨率。Topaz Photo AI 可…...
C++ std::map
std::map 是 C++ STL 中最常用的有序键值对容器,其核心功能是存储唯一键(key)与对应值(value)的映射关系,并自动按键的顺序排序。底层基于红黑树(自平衡二叉搜索树)实现,这使得它在键的查找、插入、删除等操作上保持稳定的高效性。 1、底层数据结构与核心特性 1.1 底层…...
易基因:Nat Genet/IF29:董朝斌团队ChIP-seq等揭示作物株型穗型发育调控新机制 助力表观遗传育种驯化改良(顶刊佳作)
大家好,这里是专注表观组学十余年,领跑多组学科研服务的易基因。 近日,中国农业大学玉米生物育种全国重点实验室董朝斌教授为第一及共同通讯作者,美国加州大学伯克利分校George Chuck研究员为共同通讯作者,在国际遗传学Top期刊《Nature Genetics》发表了题为“A regulator…...
Edge浏览器网页长截图
长截图步骤1.在Edge浏览器中打开目标网页(需确保网页完全加载后再截图,否则可能截取不完整)2.按下Ctrl+Shift+S组合键3.选择截屏模式为捕获整页(自动滚动并截取整个网页,生成一张长图)4.截图后可直接复制或保存为图片文件...
Python TensorFlow的CNN-LSTM-GRU集成模型在边缘物联网数据IoT电动汽车充电站入侵检测应用
全文链接:https://tecdat.cn/?p=43881原文出处:拓端抖音号@拓端tecdat随着物联网(IoT)技术在电动汽车充电站(EVCS)中的普及,充电站不仅成为智能交通的关键节点,更因连接电网、用户设备与管理系统,成为网络攻击的重点目标。传统入侵检测系统(IDS)要么难以处理IoT环境…...
C++多线程编程—线程控制、同步与互斥详解
本文将深入探讨C++多线程编程中的核心概念:线程控制、同步与互斥。 1.线程控制:join 与 detach当我们创建一个线程(std::thread)后,我们必须明确在这个线程对象销毁之前,如何管理它所代表的执行线程。这就是 join 和 detach 的用武之地。join()作用:阻塞当前线程(通常是…...
MySQL启动失败:mysqld.log Permis 报错处理.250916
报错:Could not open file /var/log/mysqld.log for error logging: Permis 解决办法: sudo setenforce 0 systemctl restart mysqld systemctl status mysqld如果好了,就更改selinux策略: sudo semanage fcontext -a -t mysqld_db_t "/home/mysql/data(/.*)?" …...
源码管理—密钥硬编码问题
源码管理—密钥硬编码问题目录密钥硬编码的定义: 指在代码、配置里硬编码密码/明文密码在配置文件中,但是不论是通过 AES 加密过的密码,还是将明文密码存储在远程配置文件中,都属于硬编码密钥。 常见的密钥硬编码场景:密钥放在环境变量 密钥加密存储在代码里 密钥放在服务…...
无速度传感器交流电机的扩展Luenberger观测器
扩展Luenberger观测器是一种用于无速度传感器交流电机控制的重要技术,它能够估计电机的内部状态(如转子磁链)和转速。 理论背景 对于感应电机,在静止α-β坐标系下的模型可以表示为: 状态方程: dx/dt = A(ω)x + Bu y = Cx其中: x = [isα, isβ, ψrα, ψrβ]^T u = […...
AI Ping体验记:终于有人做大模型服务的“性能监控”了
引言 最近几个月,我们公司在开发AI应用平台并集成到现有系统中。作为项目的技术选型负责人,我被MaaS平台API的选择问题折磨得不轻。面对市面上众多的大模型服务商,如何选出最适合我们的那一个,真的是个大难题。 市面上的MaaS平台越来越多,光是国内的就有20多家,这还是我知…...
数据库原理-第二章——关系型数据库
pass...
mac 的任务栏 Windows-Style Taskbar For macOS
https://lawand.io/taskbar/...
快手Java一面
线程池七大参数和作用?有大量执行时间短的任务如何设置线程池参数? Synchronized和ReentrantLock实现上的区别?哪个能尝试获取锁?tryLock方法参数是什么?返回是什么? ReentrantLock的公平锁和非公平锁怎么实现的? JVM堆内存怎么划分的? CMS垃圾回收机制下新生代和老年代…...
详细介绍:Elastic APM 入门指南:快速设置应用性能监控
详细介绍:Elastic APM 入门指南:快速设置应用性能监控pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", mon…...
想找Axure替代?这6个原型设计工具值得一试
引言 在原型设计工具里,Axure可以说是最为经典、老牌的一款了。它在产品圈内有着较高的知名度和地位,但随着工具生态的不断丰富,越来越多产品经理在学习和使用Axure的过程中,逐渐放弃转为寻找Axure替代工具。其实,如今想找到能替代Axure的工具并不难,本文就为大家介绍6款…...
H5游戏性能优化系列-----cpu相关优化
cpu优化主要是优化cpu使用率,帧率平稳性(卡帧,长耗时任务),主要从以下几个方面优化设置合适的帧率。根据游戏类型设置合适的帧率,比如slg,回合制这种类型游戏一般开30帧,mmo等即时战斗的或者对流畅度有很高要求的可以开60帧。 帧同步与状态同步的抉择。一般来说状态同步会…...