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

【STM32+HAL库+CubeMX】UART轮询收发、中断收发、DMA收发方法及空闲中断详解

(转载)原文链接:https://blog.csdn.net/qq_39344192/article/details/131470735

1. 什么是UART?

UART是一种异步串行通信接口,常用于通过串口与外部设备进行通信。它通过发送和接收数据帧来实现数据传输,使用起来相对简单。UART通常包含发送器(Transmitter)和接收器(Receiver),通过两根信号线(传输线)进行双向通信。

2. UART协议内容简介

image

UART协议将一长串数据切成很多固定长度的小段,分别发送。每小段数据前后会加上一些附加数据以保证通信的实时性和准确性,最后形成的每个小段叫做一个数据包——即1帧数据。

  • 起始位:发出1位低电平信号,表示开始传输字符。
  • 数据位:真正发送的数据,一般为8位(1个字节),常采用ASCII编码,从最低位开始发送。
  • 校验位:用于检验接收到的数据是否正确,分为奇校验和偶校验。
  • 停止位:一组数据的结束传输的标志。可以是1位、1.5位、2位的高电平。
  • 空闲位:空闲时数据线为高电平状态,代表无数据传输。
  • 波特率:衡量传输速率的指标。UART通信中波特率等于比特率。

UART通信的两个设备间,以上因素必须完全一致才能实现数据通信。

3. UART轮询收发

UART轮询收发时,CPU会不断检测串口的状态位来判断数据收发的情况。

3.1 UART轮询收发的优缺点

UART轮询收发是一种简单直接的UART通信方式,它具有以下优点和缺点:

优点

  • 简单易实现:相比于中断或DMA方式进行数据收发,UART轮询收发的实现相对简单,不需要额外配置中断或DMA控制器,减少了开发的复杂性。
  • 低延迟:由于没有中断处理程序的介入和数据传输的等待时间,UART轮询收发可以实现较低的延迟,对实时性要求较高的应用场景较为适用。占用CPU资源,效能低:UART轮询收发需要通过不断的轮询来检查发送和接收缓冲区的状态,这会占用CPU的资源,导致CPU无法充分利用来执行其他任务。

缺点

  • 占用CPU资源,效能低:UART轮询收发需要通过不断的轮询来检查发送和接收缓冲区的状态,这会占用CPU的资源,导致CPU无法充分利用来执行其他任务。

UART轮询收发适用于简单的、对实时性要求不高的低速通信场景。但在对实时性、效率和灵活性要求较高的应用中,中断或DMA方式可能更加适合。在选择UART通信方式时,需要根据具体应用需求进行权衡和选择。

3.2 UART轮询收发相关的函数

初始化UART参数:首先,需要对UART进行初始化,包括波特率(Baud rate)、数据位数、校验位、停止位等参数的设置。这些参数决定了数据的传输格式。

初始化函数HAL_UART_Init(UART_HandleTypeDef *huart);
功能根据串口句柄指定的参数进行串口初始化。
入口参数huart:串口句柄的地址指针。
返回值HAL状态值。
说明使用CubeMX配置工程时,初始化代码会自动生成,我们不需要再对串口进行初始化。

发送数据:要发送数据,首先将待发送的数据写入UART发送缓冲区,然后调用轮询发送函数。在轮询方式下,单片机会一直检查是否完成发送,直到超过设定时间或数据发送完成。

轮询发送函数HAL_UART_Transmit(UART_Handle TypeDef *huart, uint 8_t *pData, uint 16_t Size, uint 32_t Timeout);
功能在轮询方式下发送一定数量的数据。
入口参数huart:串口句柄的地址。
pData:待发送数据的首地址。
Size:发送的字节数。
Timeout:超时等待时间, 以毫秒为单位。
返回值HAL状态值。

接收数据:要接收数据,需要从UART接收缓冲区读取数据。同样,在轮询方式下,单片机会一直检查是否完成接收,直到超过设定时间或接收到所有数据。

轮询接收函数HAL_UART_Receive(UART_Handle TypeDef *huart, uint 8_t *pData, uint 16_t Size, uint 32_t Timeout);
功能在轮询方式下接收一定数量的数据。
入口参数huart:串口句柄的地址。
pData:存放数据的首地址。
Size:接收数据的字节数。
Timeout:超时等待时间, 以毫秒为单位。
返回值HAL状态值。

3.3 【实践】使用蓝牙模块发送数据

使用UART协议,向手机循环发送“Hello World”语句。

3.3.1 配置UART

image

  1. 左侧connectivity中点击USART1。
  2. 右侧Mode,选择为Asynchronous(异步通信)
  3. 配置参数:
  • Baud Rate(波特率):9600Bits/s
  • Word Length(字长):8 Bits
  • Parity(奇偶校验位):无校验位
  • Stop Bits(停止位):1位

USART包括UART与USRT。我们使用USART的UART模式,即Asynchronous。

3.3.2 连接单片机与蓝牙模块
单片机蓝牙模块
RXTX
TXRX
5VVCC
GNDGND
3.3.3 代码实现

在main.c中添加如下语句:

/* 添加至引用区 */
#include "string.h" // strlen函数依赖该头文件
/* 添加至变量定义区*/
char txBuffer[] = "Hello World"; // 需要发送的数据
/* 写在main()的while(1)循环内 */
while(1)
{HAL_UART_Transmit(&huart1, (uint8_t *)txBuffer, strlen(txBuffer), 1000);HAL_Delay(1000);
}

4. UART中断收发

4.1 UART中断收发的优缺点

优点

  • 提高系统效率:相比于UART轮询收发方式,UART中断收发可以提高系统的效率。当每完成发送或接收一帧数据时,中断会通知CPU进行相应的处理,而不需要CPU不断地轮询发送或接收缓冲区,释放了CPU资源,使CPU可以同时执行其他任务;
  • 系统响应更快:通过使用中断机制,UART中断收发可以提供更快的系统响应时间。一旦有数据可发送或可接收时,中断立即触发,通知CPU进行相应操作,减少了数据传输的延迟。
  • 灵活性:UART中断收发具有较高的灵活性。中断处理程序可以对数据进行灵活的处理和控制,可以根据实际需求进行相应的操作,如解析数据、执行特定任务等。复杂性增加:相对于UART轮询收发方式,UART中断收发的实现相对复杂一些。需要正确配置中断触发条件和中断优先级,同时编写中断处理程序进行数据的发送和接收操作。

缺点

  • 中断开销:中断处理程序的执行会占用一定的CPU时间和系统资源。频繁的中断触发可能会增加系统的开销。
  • 实时性限制:尽管UART中断收发可以提高系统的响应时间,但它仍然受限于中断处理程序的执行时间和优先级。在高实时性要求的应用中,中断处理程序的执行时间必须保持足够短,以确保数据的及时处理和传输。
  • 占用CPU资源:在传输数据量较大时,如果采用中断方式,每收发一帧的数据,CPU都会被打断,造成CPU无法处理其他事务。因此在批量数据传输,通信波特率较高时,建议采用DMA方式。

UART中断收发相较于UART轮询收发,提高了系统的效率,但是遇到大量、高速的数据传输时仍然会对CPU的性能产生影响。

4.2 UART中断收发时触发中断的流程

image

使能中断后,每收发1帧数据后,UART会触发UART全局中断。此时程序会进入到中断处理函数 USART1_IRQHandler()。在这个函数中,又调用了HAL库的中断处理函数HAL_UART_IRQHandler(&huart1),该函数会通过判断中断类型来决定调用哪个函数。

如果收发完成 -> 进入发送/接收回调函数;
如果产生错误 -> 进入错误回调函数;
如果未收发完 -> 继续发送/接收。

简而言之,每完成一帧数据的收发,都会调用一次中断处理函数,但只有当收发完成时才会调用发送/接收回调函数。在实际操作中,我们一般不需要对中断处理函数HAL_UART_IRQHandler()进行修改。我们将收发完成时的操作逻辑写在回调函数中即可。

4.3 UART中断收发相关的函数

4.3.1 发送相关函数
中断发送函数HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);
功能在中断方式下发送一定数量的数据。
入口参数huart:串口句柄的地址。
pData:发送数据的首地址。
Size:发送数据的字节数。
返回值HAL状态值。
发送回调函数HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart);
功能本函数会在中断发送完成时被调用。
入口参数huart:串口句柄的地址。
返回值
4.3.2 接收相关函数
中断接收函数HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);
功能在中断方式下接收一定数量的数据。
入口参数huart:串口句柄的地址。
pData:存放数据的首地址。
Size:接收数据的字节数。
返回值HAL状态值。
接收回调函数HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart);
功能本函数会在中断接收完成时被调用。
入口参数huart:串口句柄的地址。
返回值

4.4 回调函数的使用方式

stm32f1xx_hal_uart.c文件中,我们可以找到回调函数的声明方式(下以接收回调函数为例)。

image

关注到函数声明前带有__weak修饰,我们一般将这种函数称之为“弱函数”。含有__weak标识的函数,我们可以自行声明一个同名函数,最终编译器在编译的时候会选择我们自己定义的函数。如果我们没有声明该函数,则会调用有__weak修饰的函数。因此,我们只需要在合适的位置自己声明一个回调函数即可,不必在该文件中进行修改。

4.5 【实践】使用中断方式收发数据

通过蓝牙向单片机发送三个英文字母,单片机将大小写翻转后发回。

4.5.1 配置UART

在上一节配置的基础上,我们打开UART1的全局中断。

image

4.5.2 代码实现

我们在main.c中添加如下代码

/* 添加至宏定义区 */
#define rxDataLen 3 //接收三个字节的数据
/* 添加至变量定义区*/
char rxBuffer[rxDataLen]; // 存储接收到的数据的数组int main()
{/* 添加至while(1)循环前*/HAL_UART_Receive_IT(&huart1, (uint8_t *)rxBuffer, rxDataLen); // 中断方式接收三个字节数据while(1) // 注意不要将中断接收写在while循环内{HAL_Delay(1);}
}

在完成一次接收后,我们需要将接收到的字母大小写翻转并输出。我们需要在接收回调函数中实现这一功能,定义接收回调函数如下。

// 接收回调函数(由于需要覆盖原先的弱函数,本函数名称不能更改)
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) 
{if(huart == &huart1) // 如果是串口1{for(int i = 0; i < rxDataLen; i++){if(rxBuffer[i] >= 'a' && rxBuffer[i] <= 'z'){rxBuffer[i] += 'A' - 'a';}else if(rxBuffer[i] >= 'A' && rxBuffer[i] <= 'Z'){rxBuffer[i] -= 'A' - 'a';}}HAL_UART_Transmit_IT(&huart1, (uint8_t *)rxBuffer, rxDataLen);}// 使用HAL_UART_Receive_IT()时,只会进入一次中断。因此需要在回调函数内再次调用该函数HAL_UART_Receive_IT(&huart1, (uint8_t *)rxBuffer, rxDataLen);
}

5. UART使用DMA进行收发 & 空闲中断

5.1 啥是DMA?

DMA(Direct Memory Access,直接内存访问)是一种计算机系统中的数据传输技术,它允许外设直接访问寄存器,而无需通过CPU的干预。DMA技术可以提高数据传输的效率和系统性能,减轻CPU的负担。

简单点来说,DMA收发与轮询、中断都有所不同。在收发数据时,CPU只需告诉DMA数据的来源以及目的地址等信息即可,而无需参与中间的所有传输过程(例如中断收发时每收发一帧数据都会进入中断处理函数,而DMA只在接收完成等少数时刻触发中断,大大降低了CPU的压力)。

5.2 什么是空闲中断?

空闲中断(Idle Interrupt)是UART通信中的一种中断类型。当UART处于空闲状态(没有接收到数据)且持续时间超过一个帧的传输时间时,空闲中断会触发,调用上节提到的函数USART1_IRQHandler。空闲中断可以用于检测数据帧的结束或接收数据的完成。

5.3 相关函数

使能UART的特定中断(宏)__HAL_UART_ENABLE_IT(__HANDLE__, __INTERRUPT__);
参数__HANDLE__:指向UART句柄的指针;
__INTERRUPT__:要使能的中断标志位。
说明本节会使用到空闲中断,即UART_IT_IDLE
DMA发送函数HAL_UART_Transmit_DMA(UART_Handle TypeDef *huart, uint 8_t *pData, uint 16_t Size);
入口参数huart:串口句柄的地址。
pData:发送数据的首地址。
Size:发送数据的字节数。
DMA接收函数HAL_UART_Receive_DMA(UART_Handle TypeDef *huart, uint 8_t *pData, uint 16_t Size);
入口参数huart:串口句柄的地址。
pData:存放数据的首地址。
Size:接收数据的字节数。
禁用DMA传输(宏)__HAL_DMA_DISABLE(_HANDLE_);
参数__HANDLE__:指向UART句柄的指针
使能DMA传输(宏)__HAL_DMA_ENABLE(_HANDLE_);
参数__HANDLE__:指向UART句柄的指针
设置DMA传输计数器值(宏)__HAL_DMA_SET_COUNTER(__HANDLE__, __COUNTER__);
参数__HANDLE__:指向UART句柄的指针
__COUNTER__:要设置的计数器值。
获取DMA传输的计数器值(宏)__HAL_DMA_GET_COUNTER(__HANDLE__);
参数__HANDLE__:指向UART句柄的指针
宏定义#define __HAL_DMA_GET_COUNTER(__HANDLE__) ((__HANDLE__)->Instance->CNDTR)
说明这个宏需要我们自行定义。
我们可以利用这个宏定义来获取接收到的数据的长度。
接收到的数据长度 = 设置的长度 - 当前DMA计数器的值,即
rxLen = rxBufferLen - __HAL_DMA_GET_COUNTER(__HANDLE__);

5.4 【实践】使用DMA & 空闲中断实现不定长数据的接收

结合DMA以及空闲中断,实现接收不定长度的一段数据,发送这一数据的长度。

5.4.1 配置UART

参考前两节,配置好UART以及全局中断。我们还需要开启UART的DMA接收。

5.4.2 代码实现

在本节中,我们将代码按用途划分为不同函数,逐一实现。先定义如下全局变量以及宏变量。

#define rxBufferLen 40 // 接收数据的最大长度
#define __HAL_DMA_SET_COUNTER(__HANDLE__, __COUNTER__) ((__HANDLE__)->Instance->CNDTR = (uint16_t)(__COUNTER__))
char rxBuffer[rxBufferLen]; // 用于存放接收到的数据

程序运行之初,我们需要先对DMA接收进行初始化,该初始化函数应当在main()函数内调用一次。

void UART_InitDMAReceive() // 初始化函数
{__HAL_UART_CLEAR_IDLEFLAG(&huart1); // 清除空闲中断标志位__HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE); // 开启UART的空闲中断HAL_UART_Receive_DMA(&huart1, (uint8_t*)rxBuffer, rxBufferLen ); // 启动DMA接收
}

每当发生空闲中断或接收到的数据超过缓存区大小时,会产生一次中断。我们首先需要自行编写中断回调函数。

void UART_DMAIdleCallback(UART_HandleTypeDef *huart)
{if(huart == &huart1){// 停止DMA接收__HAL_DMA_DISABLE(huart->hdmarx);// 计算接收到的数据长度uint8_t dataLen = rxBufferLen - __HAL_DMA_GET_COUNTER(huart->hdmarx) + '0'; // 发送数据长度,使用轮询方法HAL_UART_Transmit(&huart1, &dataLen, 1, 10);// 重启DMA接收__HAL_DMA_SET_COUNTER(huart->hdmarx, rxBufferLen); // 重设DMA计数器__HAL_DMA_ENABLE(huart->hdmarx); // 使能DMA接收}
}

我们在中断处理函数内添加如下内容(该函数位于文件stm32g4xx_it.c内)。

void USART1_IRQHandler(void)
{/* USER CODE BEGIN USART1_IRQn 0 *//* USER CODE END USART1_IRQn 0 */HAL_UART_IRQHandler(&huart1);/* USER CODE BEGIN USART1_IRQn 1 */if(__HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE) != RESET) // 判断中断是否为空闲中断{__HAL_UART_CLEAR_IDLEFLAG(&huart1); // 清除空闲中断标志位UART_DMAIdleCallback(&huart1); // 空闲回调函数}/* USER CODE END USART1_IRQn 1 */
}

相关文章:

【STM32+HAL库+CubeMX】UART轮询收发、中断收发、DMA收发方法及空闲中断详解

&#xff08;转载&#xff09;原文链接&#xff1a;https://blog.csdn.net/qq_39344192/article/details/131470735 1. 什么是UART&#xff1f; UART是一种异步串行通信接口&#xff0c;常用于通过串口与外部设备进行通信。它通过发送和接收数据帧来实现数据传输&#xff0c;使…...

基于Java医院管理系统设计与实现(源码+部署文档)

博主介绍&#xff1a; ✌至今服务客户已经1000、专注于Java技术领域、项目定制、技术答疑、开发工具、毕业项目实战 ✌ &#x1f345; 文末获取源码联系 &#x1f345; &#x1f447;&#x1f3fb; 精彩专栏 推荐订阅 &#x1f447;&#x1f3fb; 不然下次找不到 Java项目精品实…...

PHP://filter过滤器

今天刷题遇到了php://filter过滤器的知识点考察&#xff1b;不会&#xff0c;看了几篇写的不错的文章&#xff0c;本来想转载的&#xff0c;但是代码复制过来后发现格式很乱&#xff0c;和原文格式差太多了&#xff1b;算了&#xff0c;直接把文章连接拿过来吧&#xff0c;在这…...

蓝桥杯刷题day05——2023

1、题目描述 请求出在12345678 (含) 至 98765432 (含) 中 &#xff0c;有多少个数中完全不包含 2023。 完全不包含 2023是指 无论将这个数的哪些数位移除都不能得到2023。 例如 20322175&#xff0c;33220022 都完全不包含 2023&#xff0c; 而20230415&#xff0c;20193213 …...

【51单片机】开发板和单片机的介绍(2)

前言 大家好吖&#xff0c;欢迎来到 YY 滴单片机系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过单片机的老铁 主要内容含&#xff1a; 欢迎订阅 YY滴C专栏&#xff01;更多干货持续更新&#xff01;以下是传送门&#xff01; YY的《C》专栏YY的《C11》专栏YY的…...

《剑指 Offer》专项突破版 - 面试题 30 和 31:详解如何设计哈希表以及利用哈希表设计更加高级、复杂的数据结构

目录 一、哈希表的基础知识 二、哈希表的设计 2.1 - 插入、删除和随机访问都是 O(1) 的容器 2.2 - 最近最少使用缓存 一、哈希表的基础知识 哈希表是一种常见的数据结构&#xff0c;在解决算法面试题的时候经常需要用到哈希表。哈希表最大的优点是高效&#xff0c;在哈希表…...

回顾2023年及过去五年的成长经历

现在是2024年2月4日&#xff0c;我想回顾下过去两年的经历和感悟。总结下过去五年的成长经历。 最大的感悟就两点。第一&#xff0c;我相比于两年前成长了很多、也成熟了很多&#xff0c;不管是心智上还是心态上。而这些成长来自于读书、思考和结合实践的反思。第二&#xff0…...

99例电气实物接线及52个自动化机械手动图

给大家分享一些流水线设计中常见的一些结构&#xff0c;这些动态图很直观&#xff0c;有助于大家了解其原理&#xff0c;非常好懂。 1.家庭总电箱接线图 2.经典双控灯接线 3.五孔一开接线 4.电动机点动控制接线&#xff08;不安全&#xff09; 5.电动机自锁接线图&#xff08;…...

SQL中聚合函数

SQL中的聚合函数是用于对一组值执行计算&#xff0c;并返回单个值的函数。它们通常在SELECT语句的SELECT列表中使用&#xff0c;并与GROUP BY子句结合使用来汇总数据。聚合函数忽略NULL值&#xff0c;只对非NULL值进行计算。以下是一些最常用的SQL聚合函数&#xff1a; 1. COU…...

深度学习预备知识1——数据操作

所有机器学习方法都涉及从数据中提取信息&#xff0c;因此需要一些关于数据的实用技能&#xff0c;包括存储、操作和预处理数据。 机器学习通常需要处理大型数据集。线性代数和矩阵是计算大量数据的有力工具&#xff0c;需要一些矩阵运算相关的线性代数知识。 深度学习是关于…...

【云原生运维问题记录】kubesphere登录不跳转问题

文章目录 现象问题排查 结论先行&#xff1a;kubesphere-system名称空间下reids宕机重启&#xff0c;会判断是否通过registry-proxy重新拉取镜像&#xff0c;该镜像原本是通过阿里云上拉取&#xff0c;代理上没有出现超时情况&#xff0c;导致失败。解决方案&#xff1a;删除re…...

深入学习Prometheus! 一款开源的监控和警报工具!

深入学习Prometheus! 一款开源的监控和警报工具&#xff01; Prometheus是一个开源的监控和警报工具&#xff0c;它广泛用于记录和收集各种指标&#xff08;如硬件资源使用情况、应用性能等&#xff09;&#xff0c;并提供强大的查询语言以帮助用户分析和查看这些数据。本文将…...

【webrtc】跟webrtc学list遍历

m98 代码:RTT G:\CDN\rtcCli\m98\src\video\call_stats.cc遍历list 进行删除 :remove_if void RemoveOldReports(int64_t now, std::list<CallStats::RttTime>* reports) {static constexpr const <...

网络安全产品之准入控制系统

文章目录 一、什么是准入控制系统二、准入控制系统的主要功能1. 接入设备的身份认证2. 接入设备的安全性检查 三、准入控制系统的工作原理四、准入控制系统的特点五、准入控制系统的部署方式1. 网关模式2. 控制旁路模式 六、准入控制系统的应用场景七、企业如何利用准入控制系统…...

为什么免费ip代理不适用于分布式爬虫?

费IP代理通常是一些公开免费提供的IP地址和端口&#xff0c;供用户免费使用。然而&#xff0c;这些免费IP代理并不适用于分布式爬虫的使用&#xff0c;原因如下&#xff1a; 1. 不稳定性 免费IP代理通常是由个人或组织提供的&#xff0c;没有稳定的维护和管理机制。因此&…...

【HTML 基础】元数据 meta 标签

文章目录 1. 设置字符集2. 描述网页内容3. 设置关键词4. 网页重定向5. 移动端优化注意事项结语 在网页开发中&#xff0c;<meta> 标签是一种十分重要的 HTML 元数据标签。通过巧妙使用 <meta> 标签&#xff0c;我们能够设置各种元数据&#xff0c;从而影响网页在浏…...

考研中常见的算法-逆置

元素逆置 概述&#xff1a;其实就是将 第一个元素和最后一个元素交换&#xff0c;第二个元素和倒数第二个元素交换&#xff0c;依次到中间位置。用途&#xff1a;可用于数组的移动&#xff0c;字符串反转&#xff0c;链表反转操作&#xff0c;栈和队列反转等操作。 逆置图解 …...

docker exec命令流程

背景 在使用docker时&#xff0c;我们经常会使用docker的很多命令&#xff0c;比如docker exec等创建容器并执行命令&#xff0c;那么你知道这条命令背后的原理吗&#xff0c;本文就来解析下这条命令大致的执行流程图 docker exec命令 首先我们按照启动docker之后&#xff0…...

游戏中好胜心的强化作用及其影响

在虚拟与现实交织的数字时代&#xff0c;电子游戏已经发展成为全球数以亿计玩家的日常娱乐和社交活动之一。其中&#xff0c;游戏体验往往激发并放大了参与者的好胜心理&#xff0c;这种现象不仅显著增强了游戏的吸引力&#xff0c;也在一定程度上塑造了玩家的行为模式和性格特…...

备战蓝桥杯---搜索(应用入门)

话不多说&#xff0c;直接看题&#xff1a; 显然&#xff0c;我们可以用BFS&#xff0c;其中&#xff0c;对于判重操作&#xff0c;我们可以把这矩阵化成字符串的形式再用map去存&#xff0c;用a数组去重现字符串&#xff08;相当于map映射的反向操作&#xff09;。移动空格先找…...

C++:std::is_convertible

C++标志库中提供is_convertible,可以测试一种类型是否可以转换为另一只类型: template <class From, class To> struct is_convertible; 使用举例: #include <iostream> #include <string>using namespace std;struct A { }; struct B : A { };int main…...

FFmpeg 低延迟同屏方案

引言 在实时互动需求激增的当下&#xff0c;无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作&#xff0c;还是游戏直播的画面实时传输&#xff0c;低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架&#xff0c;凭借其灵活的编解码、数据…...

tree 树组件大数据卡顿问题优化

问题背景 项目中有用到树组件用来做文件目录&#xff0c;但是由于这个树组件的节点越来越多&#xff0c;导致页面在滚动这个树组件的时候浏览器就很容易卡死。这种问题基本上都是因为dom节点太多&#xff0c;导致的浏览器卡顿&#xff0c;这里很明显就需要用到虚拟列表的技术&…...

大语言模型(LLM)中的KV缓存压缩与动态稀疏注意力机制设计

随着大语言模型&#xff08;LLM&#xff09;参数规模的增长&#xff0c;推理阶段的内存占用和计算复杂度成为核心挑战。传统注意力机制的计算复杂度随序列长度呈二次方增长&#xff0c;而KV缓存的内存消耗可能高达数十GB&#xff08;例如Llama2-7B处理100K token时需50GB内存&a…...

push [特殊字符] present

push &#x1f19a; present 前言present和dismiss特点代码演示 push和pop特点代码演示 前言 在 iOS 开发中&#xff0c;push 和 present 是两种不同的视图控制器切换方式&#xff0c;它们有着显著的区别。 present和dismiss 特点 在当前控制器上方新建视图层级需要手动调用…...

Kafka主题运维全指南:从基础配置到故障处理

#作者&#xff1a;张桐瑞 文章目录 主题日常管理1. 修改主题分区。2. 修改主题级别参数。3. 变更副本数。4. 修改主题限速。5.主题分区迁移。6. 常见主题错误处理常见错误1&#xff1a;主题删除失败。常见错误2&#xff1a;__consumer_offsets占用太多的磁盘。 主题日常管理 …...

9-Oracle 23 ai Vector Search 特性 知识准备

很多小伙伴是不是参加了 免费认证课程&#xff08;限时至2025/5/15&#xff09; Oracle AI Vector Search 1Z0-184-25考试&#xff0c;都顺利拿到certified了没。 各行各业的AI 大模型的到来&#xff0c;传统的数据库中的SQL还能不能打&#xff0c;结构化和非结构的话数据如何和…...

[特殊字符] 手撸 Redis 互斥锁那些坑

&#x1f4d6; 手撸 Redis 互斥锁那些坑 最近搞业务遇到高并发下同一个 key 的互斥操作&#xff0c;想实现分布式环境下的互斥锁。于是私下顺手手撸了个基于 Redis 的简单互斥锁&#xff0c;也顺便跟 Redisson 的 RLock 机制对比了下&#xff0c;记录一波&#xff0c;别踩我踩过…...

Python学习(8) ----- Python的类与对象

Python 中的类&#xff08;Class&#xff09;与对象&#xff08;Object&#xff09;是面向对象编程&#xff08;OOP&#xff09;的核心。我们可以通过“类是模板&#xff0c;对象是实例”来理解它们的关系。 &#x1f9f1; 一句话理解&#xff1a; 类就像“图纸”&#xff0c;对…...

ubuntu清理垃圾

windows和ubuntu 双系统&#xff0c;ubuntu 150GB&#xff0c;开发用&#xff0c;基本不装太多软件。但是磁盘基本用完。 1、查看home目录 sudo du -h -d 1 $HOME | grep -v K 上面的命令查看$HOME一级目录大小&#xff0c;发现 .cache 有26GB&#xff0c;.local 有几个GB&am…...