当前位置: 首页 > news >正文

Rust中Tracing 应用指南

欢迎来到这篇全面的Rust跟踪入门指南。Rust 的tracing是一个用于应用程序级别的诊断和调试的库。它提供了一种结构化的、异步感知的方式来记录日志和跟踪事件。与传统的日志记录相比,tracing能够更好地处理复杂的异步系统和分布式系统中的事件跟踪,帮助开发者理解程序的执行流程和状态变化。

在本文中,我们将探讨跟踪的概念,它在Rust生态系统中的重要性,以及如何利用它来改进Rust应用程序。无论你是一个经验丰富的Rust爱好者,希望深入研究性能优化,还是一个对学习更多调试工具感兴趣的语言新手,本指南都旨在为你提供在Rust中有效使用跟踪所需的知识。

理解基本概念

在我们深入研究Rust中的跟踪细节之前,了解跟踪是什么以及为什么它是开发人员的关键工具是很重要的。

在软件开发的上下文中,跟踪是一种用于监视程序执行的方法。它包括记录有关程序执行的信息,例如函数调用、变量值,甚至整个调用堆栈。这些信息通常被称为“跟踪数据”,然后可以对其进行分析,以深入了解程序的行为。
在这里插入图片描述

跟踪在调试和性能优化中起着关键作用。通过提供程序执行的详细视图,跟踪允许开发人员识别瓶颈,发现低效率,并了解bug的根本原因。这使得它成为提高代码性能和可靠性的宝贵工具。下面是tracing 的几个关键概念:

  • Span(跨度):Span 代表程序执行中的一个时间段,可以嵌套。例如,函数的执行时间范围可以是一个 Span,在这个函数内部调用的其他函数的执行范围可以是嵌套在该 Span 中的子 Span。它可以用于记录函数调用的开始和结束时间、相关的变量值等信息。
  • Event(事件):事件是在程序执行过程中发生的离散的、值得记录的点。比如,某个重要的条件被满足、一个错误被抛出或者一个网络请求被发送等情况都可以记录为一个事件。

在下一节中,我们将探讨如何在Rust应用程序中利用跟踪功能。

tracing包简介

在Rust中,跟踪是由名为tracing 包的强大库提供的。这个包提供了实现框架,用于检测Rust程序,以收集结构化的、基于事件的诊断信息。与传统的日志记录不同,tracing旨在了解系统中一个事件或一系列事件的上下文,使其成为诊断复杂系统的强大工具。

要开始在Rust中进行跟踪,首先需要将tracing 包添加到项目中。这可以通过在Cargo中添加以下行来完成Cargo.toml文件:

[dependencies]
tracing = "0.1.37"
tracing - subscriber = "0.3.17"

一旦tracing 包被添加到你的项目中,你就可以通过在你的主Rust文件中添加以下一行来开始使用它:

use tracing::{info, trace, warn, error};

tracing包为不同级别的诊断信息提供了几个宏,包括 debug!,info!, warn!error!这些宏对应于不同级别的事件,可用于在适当级别记录信息。

在下一节中,我们将深入研究如何在Rust项目中实现跟踪,并提供代码示例和解释。

简单日志记录

我们已经在项目中设置了tracing包,让我们深入了解如何在Rust项目中实现跟踪。

use tracing::{info, trace, warn, error};
use tracing_subscriber::FmtSubscriber;fn main() {tracing::subscriber::set_global_default(tracing_subscriber::FmtSubscriber::new()).expect("setting default subscriber failed");let number = 5;info!("The number is {}", number);let result = compute(number);info!("The result is {}", result);
}fn compute(n: i32) -> i32 {trace!("Computing the value...");if n > 10 {warn!("The number is greater than 10");} else if n < 1 {error!("The number is less than 1");}n * 2
}

在上面的代码中,我们首先为跟踪事件设置默认订阅者。然后,我们使用这些info!在信息级别记录事件的宏。在compute函数中,我们使用trace!warn!,还有error!基于n的值在不同级别记录事件的宏。

这是简单的示例,但它说明了tracing包的基本用法。你可以根据需要向代码中添加更复杂的跟踪逻辑。

使用span示例

下面示例使用span跟踪函数执行:

use tracing::{info, Level, span, Instrument};
use tracing_subscriber::FmtSubscriber;fn main() {let subscriber = FmtSubscriber::builder().with_max_level(Level::INFO).finish();tracing::subscriber::set_global_default(subscriber).expect("设置全局订阅者失败");let result = add_numbers(3, 5).instrument(span!(Level::INFO, "add_numbers_span"));info!("计算结果为: {}", result);
}fn add_numbers(a: i32, b: i32) -> i32 {let sum = a + b;sum
}

使用span!(Level::INFO, "add_numbers_span")创建了一个INFO级别的 Span,名称为add_numbers_span

然后通过instrument方法将add_numbers函数的执行与这个 Span 关联起来。当函数执行时,tracing会记录这个 Span 的开始和结束时间等信息,帮助开发者更好地了解函数执行的上下文。运行这个程序会输出包含 Span 相关信息的日志,如 Span 的进入和退出记录。

在下一节中,我们将讨论如何分析程序生成的跟踪数据。

分析跟踪数据

一旦在Rust应用程序中实现了跟踪并生成了跟踪数据,下一步就是分析这些数据,以深入了解应用程序的行为。分析跟踪数据包括检查记录的事件,并使用它们来理解程序的执行流。这有帮于识别模式、发现异常,并了解应用程序的性能特征。Rust中有几个工具可用于分析跟踪数据。其中最流行的是 tracing-subscriber,它提供了实现和配置订阅者的实用程序。

下面是如何使用跟踪订阅者来分析跟踪数据的一个基本示例:

use tracing_subscriber::FmtSubscriber;fn main() {let subscriber = FmtSubscriber::builder().with_max_level(tracing::Level::TRACE).finish();tracing::subscriber::set_global_default(subscriber).expect("setting default subscriber failed");// Your application code goes here...
}

在本例中,我们创建了FmtSubscriber,并将其设置为全局默认值。with_max_level函数用于设置订阅者将记录的事件的最大级别。在本例中,设置了最大日志级别为TRACE。这意味着只有TRACE级别及以上的日志消息才会被打印。一旦收集了跟踪数据,就可以使用各种工具对其进行可视化和解释。这可以帮助你了解应用程序的性能特征、识别瓶颈并发现潜在问题。

在下一节中,我们将探索Rust中的一些高级跟踪技术。

高级跟踪技术

随着您对Rust中的基本跟踪越来越熟悉,您可能会发现自己需要更高级的技术来诊断复杂问题或优化性能。Rust生态系统为高级跟踪提供了几个强大的工具和库。

其中一个这样的工具是tracking-futures,它是跟踪tracing包的扩展,提供了支持用诊断信息检测的Future。这在异步Rust程序中特别有用,在异步Rust程序中,理解future的行为对于调试和性能优化至关重要。

另一个有用的工具是trace-serde,它提供了一个序列化器实现,用于将tracing的Id、Metadata、Event、Record和Span类型作为Serde Serializable类型。当你需要序列化跟踪数据以进行分析或传输时,这非常有用。

以下是一个更复杂一些的Rust tracing示例,展示了在异步环境下如何使用tracing以及如何对不同的模块或组件进行更细致的跟踪:

首先,在Cargo.toml中添加必要的依赖:

[dependencies]
tracing = "0.1.37"
tracing-subscriber = "0.3.17"
tokio = { version = "1.34.0", features = ["full"] }

异步跟踪示例代码:

use tracing::{debug, error, info, instrument, span, Level};
use tracing_subscriber::{fmt, prelude::*, EnvFilter};
use tokio::task;// 模拟一个异步操作,这里只是简单地休眠一段时间
async fn async_operation(name: &str) -> i32 {let sleep_duration = tokio::time::Duration::from_secs(2);tokio::time::sleep(sleep_duration).await;debug!("{} 异步操作完成", name);42
}// 一个函数,内部调用了异步操作,并进行跟踪
#[instrument(skip(inner_operation_name))]
async fn perform_operation(inner_operation_name: &str) -> i32 {let outer_span = span!(Level::INFO, "perform_operation_span");let _outer_guard = outer_span.enter();info!("开始执行 perform_operation");// 调用异步操作,并使用instrument进行跟踪let result = async_operation(inner_operation_name).instrument(span!(Level::DEBUG, "async_operation_span")).await;info!("perform_operation 执行结束");result
}#[tokio::main]
async fn main() {// 设置日志输出格式和过滤条件let subscriber = fmt::Subscriber::builder().with_env_filter(EnvFilter::from_default_env()).with_max_level(Level::DEBUG).finish();tracing::subscriber::set_global_default(subscriber).expect("设置全局订阅者失败");let operation_name = "重要操作";let result = perform_operation(operation_name).await;info!("最终结果: {}", result);// 模拟一个错误情况let error_span = span!(Level::ERROR, "error_span");let _error_guard = error_span.enter();error!("发生了一个错误");
}
  1. 异步操作的跟踪

    async_operation函数模拟简单的异步操作,这里只是休眠一段时间然后返回固定值。在函数内部使用debug!宏记录了异步操作完成的信息。

    当在perform_operation函数中调用async_operation时,通过instrument方法将其与一个名为async_operation_spanDEBUG级别的 Span 相关联。这样在执行异步操作时,tracing会记录关于这个 Span 的相关信息,比如它的开始和结束时间等,有助于在异步环境下准确了解这个操作的执行情况。

  2. 函数执行的跟踪

    perform_operation函数本身也被instrument标记,创建了名为perform_operation_spanINFO级别的 Span。在函数内部,先记录了开始执行的信息,然后调用异步操作并等待其结果,最后记录了执行结束的信息。通过这种方式,可以清晰地看到整个函数从开始到结束的执行流程以及其中包含的异步操作的情况。

    Rust中,instrument宏主要用于为异步操作添加跟踪功能。它不会改变被包装函数(在这里是async_operation)的返回值类型和实际返回的值。当async_operation函数执行完成后,它原本返回的i32类型的值(在这个例子中是42)会被正常返回,然后赋值给result变量。instrument只是在异步操作执行的过程中,围绕这个操作创建一个Span,用于记录诸如操作开始、结束等相关的跟踪信息。例如,在async_operation函数内部返回42这个值,通过instrument包装后,这个42依然会被正确地传递给result变量,就好像instrument不存在一样,从返回值的角度看,它是透明的。

  3. 主函数中的设置和操作

    main函数中,首先设置了日志订阅者的格式和过滤条件。这里使用EnvFilter根据环境变量来确定要过滤的日志级别,并且设置了最大日志级别为DEBUG,这样可以看到更多详细的信息。然后调用perform_operation函数并等待其结果,记录最终结果的信息。

    最后,为了演示错误情况的记录,创建了ERROR级别的error_span,并在其中使用error!宏记录了发生错误的信息。

运行这个程序,你会看到类似如下的输出(具体输出可能因环境和执行情况略有不同):

INFO  [perform_operation_span] 开始执行 perform_operation
DEBUG [async_operation_span] 重要操作 异步操作完成
INFO  [perform_operation_span] perform_operation 执行结束
INFO  最终结果: 42
ERROR [error_span] 发生了一个错误

通过这些输出,可以清晰地跟踪程序各个部分的执行情况,包括函数的调用、异步操作的执行以及错误的发生等,这对于调试和理解复杂的异步程序非常有帮助。

这些只是Rust中可用的高级跟踪技术的几个例子。随着你继续探索Rust生态系统,你将发现更多旨在帮助您充分利用跟踪的工具和库。编码快乐!🦀

相关文章:

Rust中Tracing 应用指南

欢迎来到这篇全面的Rust跟踪入门指南。Rust 的tracing是一个用于应用程序级别的诊断和调试的库。它提供了一种结构化的、异步感知的方式来记录日志和跟踪事件。与传统的日志记录相比&#xff0c;tracing能够更好地处理复杂的异步系统和分布式系统中的事件跟踪&#xff0c;帮助开…...

Java与C#

Java和C#&#xff08;C Sharp&#xff09;是两种流行的面向对象编程语言&#xff0c;它们在很多方面非常相似&#xff0c;因为它们都受到了类似的编程范式和语言设计理念的影响。然而&#xff0c;它们之间也存在一些重要的区别。 平台依赖性&#xff1a; Java&#xff1a;Java是…...

Docker 部署 MongoDB

&#x1f680; 作者主页&#xff1a; 有来技术 &#x1f525; 开源项目&#xff1a; youlai-mall &#x1f343; vue3-element-admin &#x1f343; youlai-boot &#x1f343; vue-uniapp-template &#x1f33a; 仓库主页&#xff1a; GitCode&#x1f4ab; Gitee &#x1f…...

【C语言】字符串左旋的三种解题方法详细分析

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C语言 文章目录 &#x1f4af;前言&#x1f4af;题目描述&#x1f4af;方法一&#xff1a;逐字符移动法&#x1f4af;方法二&#xff1a;使用辅助空间法&#x1f4af;方法三&#xff1a;三次反转法&#x1f4af;方法对…...

Android导出Excel

poi org.apache.poi:poi-ooxml:4.x&#xff1a; 不支持Android使用&#xff0c; 不支持原因&#xff1a;Android底层库不支持xml所需的bean类&#xff0c;使用即报错only supported starting with Android O (–min-api 26) org.apache.poi:poi-ooxml:5.2.0&#xff1a; 支持A…...

【学术讲座】视觉计算中的深度学习方法 AIGC图像视频生成模型的推理加速

视觉计算中的深度学习方法 发展历程 backbone 强化学习、LLM等&#xff1a;有监督 && 无监督的结合 目标检测 图像分割 网络结构搜索 搜索方法 1&#xff1a;强化学习 2&#xff1a;强化学习 3&#xff1a;梯度算法 结构选择的作用 1&#xff1a;开放环境感知网络…...

华为OD机试真题---智能驾驶

华为OD机试中的“智能驾驶”题目是一道涉及广度优先搜索&#xff08;BFS&#xff09;算法运用的题目。以下是对该题目的详细解析&#xff1a; 一、题目描述 有一辆汽车需要从m * n的地图的左上角&#xff08;起点&#xff09;开往地图的右下角&#xff08;终点&#xff09;&a…...

视频质量评价SimpleVQA

目录 一、研究意义例子 二、介绍三、文章解读3.1 论文动机3.2论文思路3.3方法3.3.1网络框架3.3.2公式解读3.3.3核心创新3.3.4理解 &#xff01;&#xff01;&#xff01;作者对模型的改进本人算法框体视频抽帧美学特征提取网络&#xff1a;3.3.5实验细节&#xff1a; 四、代码复…...

浏览器插件基于nativeMessaging通信

上一篇文章介绍了基于nativeMessaging启动本地程序&#xff0c;使用官方demo是支持双向通信&#xff0c;demo启动的程序是python写的&#xff0c;现在基于Qt程序进行说明。 消息传递协议 Chrome 会在单独的进程中启动每个原生消息传递主机&#xff0c;并使用标准输入 (stdin)…...

解决 MySQL 8.x 身份验证问题的最佳实践20241126

MySQL 8.x 身份验证问题的深入解析与实践解决方案 &#x1f3af; 引言 &#x1f58b;️ MySQL 是全球最受欢迎的开源数据库之一&#xff0c;随着 MySQL 8.x 的发布&#xff0c;引入了更安全的身份验证插件 caching_sha2_password&#xff0c;显著提升了数据库的安全性和性能。…...

对于GC方面,在使用Elasticsearch时要注意什么?

大家好&#xff0c;我是锋哥。今天分享关于【对于GC方面&#xff0c;在使用Elasticsearch时要注意什么&#xff1f;】面试题。希望对大家有帮助&#xff1b; 对于GC方面&#xff0c;在使用Elasticsearch时要注意什么&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java…...

各种排序算法

前置知识 排序: 按照递增或者递减的顺序把数据排列好 稳定性: 值相等的元素在排序之后前后顺序是否发生了改变 内部排序: 数据放在内存上 外部排序: 数据放在磁盘上 内部排序 基于比较的排序 几大排序算法 1. 堆排序 特点: 思想: 1. 创建大根堆,把所有元素放在大根堆里…...

前端-Git

一.基本概念 Git版本控制系统时一个分布式系统&#xff0c;是用来保存工程源代码历史状态的命令行工具 简单来说Git的作用就是版本管理工具。 Git的应用场景&#xff1a;多人开发管理代码&#xff1b;异地开发&#xff0c;版本管理&#xff0c;版本回滚。 Git 的三个区域&a…...

用nextjs开发时遇到的问题

这几天已经基本把node后端的接口全部写完了&#xff0c;在前端开发时考虑时博客视频类型&#xff0c;考虑了ssr&#xff0c;于是选用了nextJs&#xff0c;用的是nextUi,tailwincss,目前碰到两个比较难受的事情。 1.nextUI个别组件无法在服务器段渲染 目前简单的解决方法&…...

Cannot find a valid baseurl for repo: centos-sclo-rh/x86_64

yum install 报错: Cannot find a valid baseurl for repo: centos-sclo-rh/x86_64 CentOS7的SCL源在2024年6月30日停止维护了。 当scl源里面默认使用了centos官方的地址&#xff0c;无法连接&#xff0c;需要替换为阿里云。 cd /etc/yum.repos.d/ 找到 CentOS-SCLo-scl.repo 和…...

HCIA笔记3--TCP-UDP-交换机工作原理

1. tcp协议 可靠的连接 1.1 报文格式 1.2 三次握手 1.3 四次挥手 为什么TIME_WAIT需要2MSL的等待时间&#xff1f; &#xff08;a&#xff09; 为了实现可靠的关闭 &#xff08;b&#xff09;为了让过期的报文在网络上消失 对于(a), 假设host发给server的last ack丢了。 ser…...

RabbitMQ原理架构解析:消息传递的核心机制

文章目录 一、RabbitMQ简介1.1、概述1.2、特性 二、RabbitMQ原理架构三、RabbitMQ应用场景3.1、简单模式3.2、工作模式3.3、发布订阅3.4、路由模式3.5 主题订阅模式 四、同类中间件对比五、RabbitMQ部署5.1、单机部署5.2、集群部署&#xff08;镜像模式&#xff09;5.3、K8s部署…...

C语言指针作业

//8-29 第八章作业 //3.输入10个整数,将其中最小的数第一个数对换,把最大的数与最后一个数对换 //写3个函数:①输人10个数;②进行处理;③输出10个数。 //①输人10个数&#xff0c; //方法一&#xff1a;数组索引 void main() {int arr[10];int *p arr;//printf("请输入…...

区块链:比特币-Binance

写在前面&#xff1a;相对于Tran和BNB而言&#xff0c;Binance不支持智能合约&#xff1b;大约每 10分钟 生成一个新区块 一、认识Binance 官方网站&#xff1a;Blockchain Explorer - Bitcoin Tracker & More | Blockchain.com 官方文档&#xff1a;Authentication – I…...

数据工程流程

** 数据工程流程图** #mermaid-svg-ArT55xCISSfZImy3 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-ArT55xCISSfZImy3 .error-icon{fill:#552222;}#mermaid-svg-ArT55xCISSfZImy3 .error-text{fill:#552222;stroke…...

使用Python和Pybind11调用C++程序(CMake编译)

目录 一、前言二、安装 pybind11三、编写C示例代码四、结合Pybind11和CMake编译C工程五、Python调用动态库六、参考 一、前言 跨语言调用能对不同计算机语言进行互补&#xff0c;本博客主要介绍如何实现Python调用C语言编写的函数。 实验环境&#xff1a; Linux gnuPython3.10…...

基于springboot的县市级土地使用监控系统的设计与实现

文末获取本系统&#xff08;程序源码数据库调试部署开发环境&#xff09;文末可获取&#xff0c;系统界面在最后面。 摘 要 如今社会上各行各业&#xff0c;都喜欢用自己行业的专属软件工作&#xff0c;互联网发展到这个时候&#xff0c;人们已经发现离不开了互联网。新技术的…...

Hot100 - 最大子数组和

Hot100 - 最大子数组和 最佳思路&#xff1a;动态规划 时间复杂度&#xff1a;O(n) 代码&#xff1a; class Solution {public int maxSubArray(int[] nums) {int sum 0;int ans Integer.MIN_VALUE;for (int i 0; i < nums.length; i) {// 如果 sum > 0&#xff0c…...

Matlab 深度学习 PINN测试与学习

PINN 与传统神经网络的区别 与传统神经网络的不同之处在于&#xff0c;PINN 能够以微分方程形式纳入有关问题的先验专业知识。这些附加信息使 PINN 能够在给定的测量数据之外作出更准确的预测。此外&#xff0c;额外的物理知识还能在存在含噪测量数据的情况下对预测解进行正则…...

[STM32]从零开始的STM32 FreeRTOS移植教程

一、前言 如果能看到这个教程的话&#xff0c;说明大家已经学习嵌入式有一段时间了。还记得嵌入式在大多数时候指的是什么吗&#xff1f;是的&#xff0c;我们所说的学习嵌入式大部分时候都是在学习嵌入式操作系统。从简单的一些任务状态机再到复杂一些的RTOS&#xff0c;再到最…...

软件团队的共担责任

问责制被认为是个人与其社会系统之间的纽带&#xff0c;它创造了一种将个人与其行为和绩效联系起来的身份关系。在入门系列的第一篇文章《超越工具和流程&#xff1a;成功软件开发团队的策略》中&#xff0c;我们介绍了问责制的概念&#xff0c;并提出了以下定义&#xff1a; …...

代码美学:MATLAB制作渐变色

输入颜色个数n&#xff0c;颜色类型&#xff1a; n 2; % 输入颜色个数 colors {[1, 0, 0], [0, 0, 1]}; createGradientHeatmap(n, colors); 调用函数&#xff1a; function createGradientHeatmap(n, colors)% 输入检查if length(colors) ~ nerror(输入的颜色数量与n不一…...

Perforce SAST专家详解:自动驾驶汽车的安全与技术挑战,Klocwork、Helix QAC等静态代码分析成必备合规性工具

自动驾驶汽车安全吗&#xff1f;现代汽车的软件包含1亿多行代码&#xff0c;支持许多不同的功能&#xff0c;如巡航控制、速度辅助和泊车摄像头。而且&#xff0c;这些嵌入式系统中的代码只会越来越复杂。 随着未来汽车的互联程度越来越高&#xff0c;这一趋势还将继续。汽车越…...

大模型中常见的微调方法有哪些?

我整理了1000道算法面试题&#xff1a; 获取 这里说的微调主要是指参数微调&#xff0c;参数微调的方法主要有以下几种&#xff1a; - Adapter 在预训练模型每一层(或某些层)中添加Adapter模块(如上图左侧结构所示)&#xff0c;微调时冻结预训练模型主体&#xff0c;由Ada…...

MATLAB支持的距离度量

距离度量是用于量化两个点或样本之间差异的一种方法。在不同的领域和应用场景中&#xff0c;距离度量的选择可能会有所不同。 欧几里得距离&#xff08;Euclidean Distance&#xff09;&#xff1a;这是最直观的距离定义&#xff0c;适用于n维空间中的两点。对于二维空间中的点…...

c++编程玩转物联网:使用芯片控制8个LED实现流水灯技术分享

在嵌入式系统中&#xff0c;有限的GPIO引脚往往限制了硬件扩展能力。74HC595N芯片是一种常用的移位寄存器&#xff0c;通过串行输入和并行输出扩展GPIO数量。本项目利用树莓派Pico开发板与74HC595N芯片&#xff0c;驱动8个LED实现流水灯效果。本文详细解析项目硬件连接、代码实…...

LSA详情与特殊区域

LSA是构成LSDB的重要原材料&#xff0c;在OSPF中发挥很大作用。 报文 通用头部 LS age&#xff1a;LSA寿命&#xff0c;0-3600s Options&#xff1a;可选项 LS type&#xff1a;LSA类型&#xff0c;三要素之一 Link State ID&#xff1a;LSAID 三要素之一 Advertising Ro…...

Leecode刷题C语言之交替组①

执行结果:通过 执行用时和内存消耗如下&#xff1a; 代码如下&#xff1a; int numberOfAlternatingGroups(int* colors, int colorsSize) {int res 0;for (size_t i 0; i < colorsSize; i) {if (colors[i] ! colors[(i - 1 colorsSize) % colorsSize] && col…...

深入解析 Django 中数据删除的最佳实践:以动态管理镜像版本为例

文章目录 引言场景与模型设计场景描述 删除操作详解1. 删除单个 Tag2. 批量删除 Tags3. 删除前确认4. 日志记录 高阶优化与问题分析1. 外键约束与误删保护2. 并发删除的冲突处理3. 使用软删除 结合 Django Admin 的实现总结与实践思考 引言 在现代应用开发中&#xff0c;服务和…...

4457数字示波器 2Gpts的深度存储

4457数字示波器 2Gpts的深度存储 256级灰度等级及四种波形色彩显示 4457M系列数字示波器&#xff0c;带宽从1GHz到4GHz&#xff0c;采样率10GSa/s、20GSa/s&#xff0c;垂直分辨率12bit&#xff0c;存储深度1Gpts&#xff0c;最快波形捕获率70万个波形/秒&#xff0c;独创的…...

【笔记】轻型民用无人驾驶航空器安全操控

《轻型民用无人驾驶航空器安全操控》 理论考试培训材料 法规部分 【民用无人驾驶航空器的分类】 1、如何定义微型、轻型无人驾驶航空器&#xff1f; 微型无人驾驶航空器&#xff0c;是指空机重量小于0.25千克&#xff0c;最大平飞速度不超过40千米/小时&#xff0c;无线电发…...

【leetcode】动态规划

31. 873. 最长的斐波那契子序列的长度 题目&#xff1a; 如果序列 X_1, X_2, ..., X_n 满足下列条件&#xff0c;就说它是 斐波那契式 的&#xff1a; n > 3对于所有 i 2 < n&#xff0c;都有 X_i X_{i1} X_{i2} 给定一个严格递增的正整数数组形成序列 arr &#xff0…...

嵌入式linux系统中图像处理基本方法

目录 2.1 BMP图像处理 2.1.1 BMP文件格式解析 2.1.2 代码实现:将BMP文件解析为RGB格式,在LCD上显示 2.2 JPEG图像处理 2.2.1 JPEG文件格式和libjpeg编译 2.2.2 libjpeg接口函数的解析和使用 2.2.3 使用libjpeg把JPEG文件解析为RGB格式,在LCD上显示 …...

Qt SQL模块概述

Qt SQL支持的数据库 要在项目中使用 Qt SQL 模块&#xff0c;需要在项目配置文件中添加下面一条设置语句&#xff1a; Qt sql在头文件或源文件中使用 Qt SQL 模块中的类&#xff0c;可以使用包含语句&#xff1a; #include <QtSql>这样会将某个 Qt SQL 模块中的所有类…...

PVE相关名词通俗表述方式———多处细节实验(方便理解)

PVE设置初期&#xff0c;对CIDR、 网关、 LinuxBridge、VLAN等很有困惑的朋友一定很需要一篇能够全面通俗易懂的方式去理解PVE 中Linux网桥的工作方式&#xff0c;就像操作一个英雄&#xff0c;多个技能&#xff0c;还是需要一点点去学习理解的&#xff0c;如果你上来就对着别人…...

C语言实现冒泡排序:从基础到优化全解析

一、什么是冒泡排序&#xff1f; 冒泡排序&#xff08;Bubble Sort&#xff09;是一种经典的排序算法&#xff0c;其工作原理非常直观&#xff1a;通过多次比较和交换相邻元素&#xff0c;将较大的元素“冒泡”到数组的末尾。经过多轮迭代&#xff0c;整个数组会变得有序。 二…...

ReentrantLock(可重入锁) Semaphore(信号量) CountDownLatch

目录 ReentrantLock(可重入锁) &Semaphore(信号量)&CountDownLatchReentrantLock(可重入锁)既然有了synchronized&#xff0c;为啥还要有ReentrantLock?Semaphore(信号量)如何确保线程安全呢&#xff1f;CountDownLatch ReentrantLock(可重入锁) &Semaphore(信号量…...

Zookeeper选举算法与提案处理概览

共识算法(Consensus Algorithm) 共识算法即在分布式系统中节点达成共识的算法&#xff0c;提高系统在分布式环境下的容错性。 依据系统对故障组件的容错能力可分为&#xff1a; 崩溃容错协议(Crash Fault Tolerant, CFT) : 无恶意行为&#xff0c;如进程崩溃&#xff0c;只要…...

Jmeter中的断言

7&#xff09;断言 1--响应断言 功能特点 数据验证&#xff1a;验证响应数据是否包含或不包含特定的字符串、模式或值。多种匹配类型&#xff1a;支持多种匹配类型&#xff0c;如文本、正则表达式、文档等。灵活配置&#xff1a;可以设置多个断言条件&#xff0c;满足复杂的测…...

【通俗理解】隐变量的变分分布探索——从公式到应用

【通俗理解】隐变量的变分分布探索——从公式到应用 关键词提炼 #隐变量 #变分分布 #概率模型 #公式推导 #期望最大化 #机器学习 #变分贝叶斯 #隐马尔可夫模型 第一节&#xff1a;隐变量的变分分布的类比与核心概念【尽可能通俗】 隐变量的变分分布就像是一场“捉迷藏”游戏…...

Vivado程序固化到Flash

在上板调试FPGA时&#xff0c;通常使用JTAG接口下载程序到FPGA芯片中&#xff0c;FPGA本身是基于RAM工艺的器件&#xff0c;因此掉电后会丢失芯片内的程序&#xff0c;需要重新烧写程序。但是当程序需要投入使用时不能每一次都使用JTAG接口下载程序&#xff0c;一般FPGA的外围会…...

铲屎官进,2024年宠物空气净化器十大排行,看看哪款吸毛最佳?

不知道最近换毛季&#xff0c;铲屎官们还承受的住吗&#xff1f;我家猫咪每天都在表演“天女散花”&#xff0c;家里没有一块干净的地方&#xff0c;空气中也都是堆积的浮毛&#xff0c;幸好有宠物空气净化器这种清理好物。宠物空气净化器针对宠物浮毛设计&#xff0c;可以有效…...

SpringBoot 项目中使用 spring-boot-starter-amqp 依赖实现 RabbitMQ

文章目录 前言1、application.yml2、RabbitMqConfig3、MqMessage4、MqMessageItem5、DirectMode6、StateConsumer&#xff1a;消费者7、InfoConsumer&#xff1a;消费者 前言 本文是工作之余的随手记&#xff0c;记录在工作期间使用 RabbitMQ 的笔记。 1、application.yml 使…...

嵌入式硬件实战提升篇(二)PCB高速板设计 FPGA核心板带DDR3 PCB设计DDR全面解析

引言&#xff1a;设计一款高速板&#xff0c;供读者学习&#xff0c;FPGA核心板&#xff0c;带一颗DDR3内存&#xff0c;FPGA型号&#xff1a;XC6SLX16-2FTG256C。 随着嵌入式硬件技术的快速发展&#xff0c;高速板设计逐渐成为嵌入式系统设计中的核心技术之一。高速板的设计要…...

2044:【例5.12】回文字串

【题目描述】 输入一串字符,字符个数不超过100,且以“.”结束。 判断它们是否构成回文。 【输入】 一行字符串。 【输出】 是否为回文串。是输出“Yes”&#xff0c;否输出“No。” 【输入样例】 abccb 【输出样例】 No 代码实现 #include <stdio.h> /*2044&#x…...