Stm32_串口的帧(不定长)数据接收

目录标题

  • 前言
  • 1、串口中断接收固定帧头帧尾数据
    • 1.1、任务需求
    • 1.2、实现思路
    • 1.3、程序源码:
  • 2、串口中断接收+用定时器来判断帧结束
  • 3、串口中断接收数据+空闲中断
    • 3.1、串口的空闲中断
    • 3.2、实现思路
    • 3.3、程序源码
  • 4、串口的空闲中断+DMA转运
    • 4.1、DMA简介
    • 4.2、DMA模式
    • 4.3、DMA资源
    • 4.4、DMA主要特征
    • 4.5、实现思路
    • 4.6程序源码:
  • 报错及解决:

前言

使用串口传输数据时,因为串口是异步通信协议,所以我们需要去判断哪是一帧完整的数据,并进行数据的处理。


接收不定长的数据以下几个主要有以下方法:

  • 加固定的帧头和帧尾
  • 串口中断接收+用定时器来判断帧结束
  • 串口中断接收+利用串口空闲中断来判断帧结束

1、串口中断接收固定帧头帧尾数据

1.1、任务需求

1、PC端通过串口1发送给单片机一个有固定帧头帧尾的数据包。
2、单片机利用串口中断接收数据并判断帧头帧尾。
3、当识别到对应的帧头帧尾时,将接收到的数据再发送给PC端。
4、清除接收缓存。

1.2、实现思路

  • 1、使能相关时钟
    使能相关GPIO所在APB2总线的时钟
    使能串口所在APB2总线的时钟

  • 2、初始化串口
    配置数据位个数、停止位个数、校验位、波特率等

  • 3、初始化GPIO
    配置RX、TX对应GPIO口。

  • 4、初始化串口中断
    配置中断通道,中断优先级等

  • 4、使能串口中断

  • 5、使能串口

  • 6、编写中断服务程序:

    • ①判断接收到的数据是否是帧头
    • ②若是帧头则将数据写入到缓冲区;若不是则无视,等待下一个数据。
    • ③当有帧头后,不断在后面接收到的数据中找帧尾。找到了则说明这一帧数据接收完成了。
    • ④将这一帧数据再通过printf发送回PC端。
    • ⑤清楚缓冲区数据,将索引清零。

1.3、程序源码:

usart.h

#ifndef __SERIAL_H__
#define __SERIAL_H__#include "stm32f10x.h"                  // Device header
#include <stdarg.h>
#include "stdio.h"
#include "string.h"#define USART1_ENABLE  1          //使能串口1      
#define USART1_BAUDRATE 9600      //串口1波特率
#define USART1_INTERRUPT_ENABLE 1  //使能串口1中断#define RECEIVE_BUF_MAX_SIZE 100  //单次最大接收字节数typedef struct{uint8_t Buffer[RECEIVE_BUF_MAX_SIZE];uint16_t Lenth;
}usart_data;
void Serial_Init(void);//串口初始化
#endif

usart.c

void Serial_Init(void)
{GPIO_InitTypeDef GPIO_InitStructure;USART_InitTypeDef USART_InitStructure;NVIC_InitTypeDef NVIC_Structure;#if USART1_ENABLE  RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//串口1在APB2总线上,串口2、3在APB1总线上RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;     //复用推挽输出GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;     //读。上拉输入GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);USART_InitStructure.USART_BaudRate= USART1_BAUDRATE;USART_InitStructure.USART_HardwareFlowControl= USART_HardwareFlowControl_None;USART_InitStructure.USART_Mode= USART_Mode_Tx|USART_Mode_Rx;USART_InitStructure.USART_Parity= USART_Parity_No;USART_InitStructure.USART_StopBits= USART_StopBits_1;USART_InitStructure.USART_WordLength= USART_WordLength_8b;USART_Init(USART1,&USART_InitStructure);
#if USART1_INTERRUPT_ENABLE	USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//使能串口1接受中断NVIC_Structure.NVIC_IRQChannel=USART1_IRQn;NVIC_Structure.NVIC_IRQChannelCmd=ENABLE;NVIC_Structure.NVIC_IRQChannelPreemptionPriority=1;//抢占优先级NVIC_Structure.NVIC_IRQChannelSubPriority=3;//响应优先级NVIC_Init(&NVIC_Structure);
#endifUSART_Cmd(USART1,ENABLE);
#endif
}int fputc(int ch,FILE *f)        //重构定向,printf直接打印到串口1
{USART_SendData(USART1,ch);	while(USART_GetFlagStatus(USARTx,USART_FLAG_TXE)==RESET);return ch;
}//串口一中断处理函数
usart_data usart1_rxdata;uint16_t data;
void USART1_IRQHandler(void)
{if(USART_GetITStatus(USART1,USART_IT_RXNE)==SET) //若发生串口中断{usart1_rxdata.Buffer[usart1_rxdata.Lenth++] = USART_ReceiveData(USART1);if(usart1_rxdata.Buffer[0]!='h') //判断帧头usart1_rxdata.Lenth =0;if((usart1_rxdata.Buffer[0]=='h')&&(usart1_rxdata.Buffer[usart1_rxdata.Lenth-1]=='e'))//判断帧头帧尾{printf("rx_data:%s\r\n",usart1_rxdata.Buffer);//将接收到的一帧数据再发送回去,做验证memset(usart1_rxdata.Buffer,'\0',usart1_rxdata.Lenth);usart1_rxdata.Lenth = 0;}	USART_ClearITPendingBit(USART1,USART_IT_RXNE);//清中断标志位}
}

2、串口中断接收+用定时器来判断帧结束

暂时不写

3、串口中断接收数据+空闲中断

3.1、串口的空闲中断

空闲中断USART_IT_IDLE,俗称帧中断,即第一帧数据接收完毕到第二帧数据开始接收期间存在一个空闲状态(就是没数据接收的状态),检测到此空闲状态后即执行中断程序。进入中断程序即意味着已经接收到一组完整帧数据,仅需及时对数据处理或将数据转移出缓冲区即可。
相较于上面的利用固定帧头和帧尾来判断完整数据帧,这种方法更为使用,通过利用空闲中断,对于没有固定帧头和帧尾的数据我们也能准确接收了。

3.2、实现思路

  • 1、使能相关时钟
    使能相关GPIO所在APB2总线的时钟
    使能串口所在APB2总线的时钟
  • 2、初始化串口
    配置数据位个数、停止位个数、校验位、波特率等
  • 3、初始化GPIO
    配置RX、TX对应GPIO口。
  • 4、初始化串口中断
    配置中断通道,中断优先级等
  • 4、使能串口中断
  • 5、使能串口的空闲中断
  • 5、使能串口
  • 6、编写中断服务程序:
    • ①判断中断类型:串口中断、串口的空闲中断
    • ②若是串口中断,就将接收的数据存入缓冲区。然后清除中断标志
    • ③若是串口的空闲中断,则说明这一帧数据接收完了后面就是数据处理了。
    • ④数据处理完后,就将数据缓冲区清除,将索引号清零。
    • ⑤清除串口空闲中断标志位(通过读串口的SR和DR寄存器)

3.3、程序源码

usart.h

#ifndef __SERIAL_H__
#define __SERIAL_H__#include "stm32f10x.h"                  // Device header
#include <stdarg.h>
#include "stdio.h"
#include "string.h"#define USART1_ENABLE  1                
#define USART1_BAUDRATE 9600
#define USART1_INTERRUPT_ENABLE 1
#define USART1_IDLE_INTERRUPT_ENABLE 1#define RECEIVE_BUF_MAX_SIZE 100  //单次最大接收字节数typedef struct{uint8_t Buffer[RECEIVE_BUF_MAX_SIZE];uint16_t Lenth;
}usart_data;void Serial_Init(void);//串口初始化
#endif

usart.c

#include "Serial.h"
void Serial_Init(void)
{GPIO_InitTypeDef GPIO_InitStructure;USART_InitTypeDef USART_InitStructure;NVIC_InitTypeDef NVIC_Structure;#if USART1_ENABLE  RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//串口1在APB2总线上,串口2、3在APB1总线上RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;     //复用推挽输出GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;     //读。上拉输入GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);USART_InitStructure.USART_BaudRate= USART1_BAUDRATE;USART_InitStructure.USART_HardwareFlowControl= USART_HardwareFlowControl_None;USART_InitStructure.USART_Mode= USART_Mode_Tx|USART_Mode_Rx;USART_InitStructure.USART_Parity= USART_Parity_No;USART_InitStructure.USART_StopBits= USART_StopBits_1;USART_InitStructure.USART_WordLength= USART_WordLength_8b;USART_Init(USART1,&USART_InitStructure);#if USART1_INTERRUPT_ENABLE	USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//使能串口1接受中断NVIC_Structure.NVIC_IRQChannel=USART1_IRQn;NVIC_Structure.NVIC_IRQChannelCmd=ENABLE;NVIC_Structure.NVIC_IRQChannelPreemptionPriority=1;//抢占优先级NVIC_Structure.NVIC_IRQChannelSubPriority=3;//响应优先级NVIC_Init(&NVIC_Structure);#if USART1_IDLE_INTERRUPT_ENABLE  USART_ITConfig(USART1,USART_IT_IDLE,ENABLE);//使能串口1的空闲中断
#endif#endifUSART_Cmd(USART1,ENABLE);
#endif
}
//=================================================================
int fputc(int ch,FILE *f)        //重构定向,printf直接打印到串口1
{Serial_Sendbyte(USART1,ch);return ch;
}//串口一中断
usart_data usart1_rxdata;uint16_t data;
void USART1_IRQHandler(void)
{uint16_t clear;if(USART_GetITStatus(USART1,USART_IT_RXNE)==SET){usart1_rxdata.Buffer[usart1_rxdata.Lenth++] = USART_ReceiveData(USART1);if(usart1_rxdata.Lenth>RECEIVE_BUF_MAX_SIZE)usart1_rxdata.Lenth =0;USART_ClearITPendingBit(USART1,USART_IT_RXNE);}else if(USART_GetITStatus(USART1,USART_IT_IDLE)==SET)//串口空闲中断{printf("rx_data:%s\r\n",usart1_rxdata.Buffer);memset(usart1_rxdata.Buffer,'\0',usart1_rxdata.Lenth);usart1_rxdata.Lenth = 0;clear = USART1->SR;clear = USART1->DR;}}

4、串口的空闲中断+DMA转运

4.1、DMA简介

直接存储器访问(Direct Memory Access),简称DMA。DMA是CPU一个用于数据从一个地址空间到另一地址空间“搬运”(拷贝)的组件,数据拷贝过程不需CPU干预,数据拷贝结束则通知CPU处理。因此,大量数据拷贝时,使用DMA可以释放CPU资源。

4.2、DMA模式

DMA数据拷贝过程,典型的有:
(1)内存—>内存,内存间拷贝;
(2)外设—>内存,如uart、spi、i2c等总线接收数据过程;
(3)内存—>外设,如uart、spi、i2c等总线发送数据过程。

4.3、DMA资源

STM32F1系列的MCU有两个DMA控制器(DMA2只存在于大容量产品中),DMA1有7个通道,DMA2有5个通道,每个通道专门用来管理来自于一个或者多个外设对存储器的访问请求。还有一个仲裁器来协调各个DMA请求的优先权。
在这里插入图片描述
在这里插入图片描述

4.4、DMA主要特征

  • 每个通道都直接连接专用的硬件 DMA 请求,每个通道都同样支持软件触发。这些功能通过软件来配置。
  • 在同一个 DMA 模块上,多个请求间的优先权可以通过软件编程设置(共有四级:很高、高、中等和低),优先权设置相等时由硬件决定(请求 0 优先于请求 1 ,依此类推,可以参考STM32数据手册)。
  • 独立的源和目标数据区的传输宽度(字节、半字、全字),模拟打包和拆包的过程。源和目标地址必须按数据传输宽度对齐。
  • 支持循环的缓冲器管理(会把原来的数据覆盖)。
  • 每个通道都有 3 个事件标志(DMA 半传输, DMA 传输完成和 DMA 传输出错),这 3 个事件标志逻辑或成为一个单独的中断请求。
  • 存储器和存储器间的传输(仅 DMA2 可以)。
  • 外设和存储器、存储器和外设之间的传输
  • 闪存、SRAM 、外设的 SRAM 、APB1 、APB2 和 AHB 外设均可作为访问的源和目标。
  • 可编程的数据传输数目:最大为65535。

高波特率传输场景下,串口非常有必要使用DMA。

4.5、实现思路

  • 1、使能相关时钟
    使能相关GPIO所在APB2总线的时钟
    使能串口所在APB2总线的时钟
    使能DMA1时钟
  • 2、初始化串口
    配置数据位个数、停止位个数、校验位、波特率等
  • 3、初始化GPIO
    配置RX、TX对应GPIO口。
  • 4、初始化串口空闲中断
    配置中断通道,中断优先级等
  • 5、使能串口的空闲中断
  • 6、配置初始化DMA1。
  • 7、使能串口1的DMA接收;使能DMA1的TX通道(DMA1_Channel4)
  • 8、使能串口
  • 9、编写中断服务程序:
    • ①判断中断类型:串口的空闲中断
    • ②清除串口空闲中断标志位(通过读串口的SR和DR寄存器)
    • ③关闭DMA通道
    • ④进行数据处理(置标志位等)
    • ⑤重新设置 DMA 传输数据长度
    • ⑥使能DMA通道

4.6程序源码:

usart.h

#ifndef __SERIAL_H__
#define __SERIAL_H__#include "stm32f10x.h"                  // Device header
#include <stdarg.h>
#include "stdio.h"
#include "string.h"#define USART1_ENABLE  1                
#define USART1_BAUDRATE 9600
#define USART1_INTERRUPT_ENABLE 1
#define USART1_IDLE_INTERRUPT_ENABLE 1
#define USART1_DMA_TX_ENABLE 0
#define USART1_DMA_RX_ENABLE 1#define USART_BUF_MAX_SIZE 100  //单次最大接收字节数typedef struct{uint8_t Buffer[USART_BUF_MAX_SIZE];uint16_t Lenth;
}usart_data;void Serial_Init(void);//串口初始化
#endif

usart.c

#include "Serial.h"usart_data usart1_rxdata;
usart_data usart_txdata;void Serial_Init(void)
{GPIO_InitTypeDef GPIO_InitStructure;USART_InitTypeDef USART_InitStructure;NVIC_InitTypeDef NVIC_Structure;DMA_InitTypeDef DMA_InitStructure;
#if USART1_ENABLE  RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//串口1在APB2总线上,串口2、3在APB1总线上RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;     //复用推挽输出GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;     //读。上拉输入GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);USART_InitStructure.USART_BaudRate= USART1_BAUDRATE;USART_InitStructure.USART_HardwareFlowControl= USART_HardwareFlowControl_None;USART_InitStructure.USART_Mode= USART_Mode_Tx|USART_Mode_Rx;USART_InitStructure.USART_Parity= USART_Parity_No;USART_InitStructure.USART_StopBits= USART_StopBits_1;USART_InitStructure.USART_WordLength= USART_WordLength_8b;USART_Init(USART1,&USART_InitStructure);#if USART1_INTERRUPT_ENABLE	|USART1_IDLE_INTERRUPT_ENABLENVIC_Structure.NVIC_IRQChannel=USART1_IRQn;NVIC_Structure.NVIC_IRQChannelCmd=ENABLE;NVIC_Structure.NVIC_IRQChannelPreemptionPriority=1;//抢占优先级NVIC_Structure.NVIC_IRQChannelSubPriority=3;//响应优先级NVIC_Init(&NVIC_Structure);
#if USART1_INTERRUPT_ENABLEUSART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//使能串口1接受中断
#endif #if USART1_IDLE_INTERRUPT_ENABLE  USART_ITConfig(USART1,USART_IT_IDLE,ENABLE);//使能串口1的空闲中断
#endif#endif#if USART1_DMA_TX_ENABLE/*Usart1_TX_DMA_config,从存储区到USART1->DR*/RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);DMA_DeInit(DMA1_Channel4);DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&(USART1->DR);   //外设站点起始地址DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设站点地址不自增DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)usart_txdata.Buffer;//存储器站点起始地址DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //存储器站点自增DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;    //存储器站点到外设站点   //传输方向DMA_InitStructure.DMA_BufferSize = USART_BUF_MAX_SIZE;//缓冲区大小,传输计数器DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  //普通模式,不循环DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //失能(不是存储器到存储器)DMA_InitStructure.DMA_Priority = DMA_Priority_High; //优先级DMA_Init(DMA1_Channel4, &DMA_InitStructure);USART_DMACmd(USART1,USART_DMAReq_Tx,ENABLE); //使能串口1的DMA发送DMA_Cmd(DMA1_Channel4,ENABLE); //使能DMA1的TX通道(DMA1_Channel4)#endif
#if USART1_DMA_RX_ENABLE/*Usart1_RX_DMA_config,从USART1->DR到存储区*/RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);DMA_DeInit(DMA1_Channel5);DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&(USART1->DR);   //外设站点起始地址DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设站点地址不自增DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)usart1_rxdata.Buffer;//存储器站点起始地址DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //存储器站点自增DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;    //外设站点到存储器站点   //传输方向DMA_InitStructure.DMA_BufferSize = USART_BUF_MAX_SIZE;//缓冲区大小,传输计数器DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  //普通模式,不循环DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //失能(不是存储器到存储器)DMA_InitStructure.DMA_Priority = DMA_Priority_High; //优先级DMA_Init(DMA1_Channel5, &DMA_InitStructure);USART_DMACmd(USART1,USART_DMAReq_Rx,ENABLE); //使能串口1的DMA接收DMA_Cmd(DMA1_Channel5,ENABLE); //使能DMA1的TX通道(DMA1_Channel4)#endifUSART_Cmd(USART1,ENABLE);
#endif
}//串口一中断
void USART1_IRQHandler(void)
{uint16_t clear;if(USART_GetITStatus(USART1,USART_IT_IDLE)==SET){clear = USART1->SR;clear = USART1->DR;DMA_Cmd(DMA1_Channel5, DISABLE);  // 关闭 DMA 通道usart1_rxdata.Lenth = sizeof(usart1_rxdata.Buffer) - DMA_GetCurrDataCounter(DMA1_Channel5);  // 计算接收到的数据长度printf("rx_data:%s\r\n",usart1_rxdata.Buffer);memset(usart1_rxdata.Buffer,'\0',usart1_rxdata.Lenth);DMA_SetCurrDataCounter(DMA1_Channel5, sizeof(usart1_rxdata.Buffer));  // 重新设置 DMA 传输数据长度DMA_Cmd(DMA1_Channel5, ENABLE);  // 使能 DMA 通道}}

报错及解决:

int fputc(int ch,FILE *f)        //重构定向,printf直接打印到串口1
{USART_SendData(USART1,ch);	while(USART_GetFlagStatus(USARTx,USART_FLAG_TXE)==RESET);return ch;
}

我们在使用上面的程序进行printf重构定向时,需要注意以下两点:
1、在“魔法棒”—>"Target"中,将”USB Micro LIB“勾选上,否则会出现不能正常打印甚至单片机直接进入硬件错误中断宕机的情况。
在这里插入图片描述

2、需要添加头文件 #include “stdio.h”,否则会出现报错“error: unknown type name “FILE” ”。
在这里插入图片描述

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

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

相关文章

AHB 与 DMA

AHB&#xff08;先进高性能总线&#xff09; 随着深亚微米工艺技术日益成熟&#xff0c;集成电路芯片的规模越来越大。数字IC从基于时序驱动的设计方法&#xff0c;发展到基于IP核复用的设计方法&#xff0c;并在SOC设计中得到了广泛应用。在基于IP核复用的SoC&#xff08;Syst…

【BME2112】w11 notes

下周做老鼠实验 group analysis SPM group analysis 数据地址resting state 可以分析&#xff1a;correlation 计算两个脑区的相关性 静息态实验简单functional 成功的实验能看到激活区不成功的实验&#xff1a;比如被试头动太大&#xff0c;不是健康的被试 Spontaneous brain…

ALPHA开发板烧录工具MfgTool烧写方法

一. 简介 MfgTool 工具是 NXP 提供的专门用于给 I.MX 系列 CPU 烧写系统的软件&#xff0c;可以在 NXP 官网下载到。运行在windows下。可以烧写uboot.imx、zImage、dtb&#xff0c;rootfs。通过 USB口进行烧写。 上一篇文章简单了解了 ALPHA开发板烧录工具MfgTool。文章地址…

数据结构之交换排序

目录 交换排序 冒泡排序 冒泡排序的时间复杂度 快速排序 快速排序单趟排序的时间复杂度 快速排序的时间复杂度 交换排序 在日常生活中交换排序的使用场景是很多的&#xff0c;比如在学校做早操&#xff0c;老师通常会让学生按大小个排队&#xff0c;如果此时来了一个新学…

MySQL系列(一):索引篇

为什么是B树&#xff1f; 我们推导下&#xff0c;首先看下用哈希表做索引&#xff0c;是否可以满足需求。如果我们用哈希建了索引&#xff0c;那么对于如下这种SQL&#xff0c;通过哈希&#xff0c;可以快速检索出数据&#xff1a; select * from t_user_info where id1;但是这…

ThreadX开源助力Microsoft扩大应用范围:对比亚马逊AWS的策略差异

全球超过120亿台设备正在运行ThreadX&#xff0c;这是一款专为资源受限环境设计的实时操作系统。该操作系统在微控制器和小型处理器上表现出色&#xff0c;以极高的可靠性和精确的时间控制处理任务而闻名。 ThreadX曾是英特尔芯片管理引擎的引擎&#xff0c;并且是控制Raspber…

AWS基于x86 vs Graviton(ARM)的RDS MySQL性能对比

概述 这是一个系列。在前面&#xff0c;我们测试了阿里云经济版&#xff08;“ARM”&#xff09;与标准版的性能/价格对比&#xff1b;华为云x86规格与ARM&#xff08;鲲鹏增强&#xff09;版的性能/价格对比。现在&#xff0c;再来看看AWS的ARM版本的RDS情况 在2018年&#…

User: zhangflink is not allowed to impersonate zhangflink

使用hive2连接进行添加数据是报错&#xff1a; [08S01][1] Error while processing statement: FAILED: Execution Error, return code 1 from org.apache.hadoop.hive.ql.exec.mr.MapRedTask. User: zhangflink is not allowed to impersonate zhangflink 有些文章说需要修…

配置OSS后如何将服务器已有文件上传至OSS,推荐使用ossutil使用

1.下载安装ossutil sudo -v ; curl https://gosspublic.alicdn.com/ossutil/install.sh | sudo bash2.交互式配置生成配置文件 ossutil config 根据提示分别设置配置文件路径、设置工具的语言、Endpoint、AccessKey ID、AccessKey Secret和STSToken参数&#xff0c;STSToken留…

【Axure高保真原型】个性化自定义图片显示列表

今天和大家分享个性化自定义图片显示列表的原型模板&#xff0c;鼠标点击多选按钮&#xff0c;可以切换按钮选中或者取消选中&#xff0c;按钮选中时&#xff0c;对应图片会在列表中显示&#xff0c;按钮取消后&#xff0c;对应图片会自动隐藏。那这个模板是用中继器制作的&…

系统设计-缓存介绍

该图说明了我们在典型架构中缓存数据的位置。 沿着流程有多个层次。 客户端应用程序&#xff1a;HTTP 响应可以由浏览器缓存。我们第一次通过 HTTP 请求数据&#xff0c;返回时在 HTTP 标头中包含过期策略&#xff1b;我们再次请求数据&#xff0c;客户端应用程序首先尝试从浏…

前端实现检索文本高亮实现

文章目录 一、前言二、实现三、最后 一、前言 使用搜索引擎时的搜索结果高亮&#xff0c;搜索文本在查询出来的结果内高亮显示&#xff0c;这种在全文检索应该很常见 二、实现 看了下百度检索的实现&#xff0c;是给内容加上了em标签&#xff0c;然后给em标签设置颜色&#x…

机器的深度强化学习算法可以被诱导

设计一个好的奖励函数是机器深度强化学习算法的关键之一。奖励函数用于给予智能体&#xff08;机器&#xff09;在环境中采取不同行动时的反馈信号&#xff0c;以指导其学习过程。一个好的奖励函数应该能够引导智能体朝着期望的行为方向学习&#xff0c;并尽量避免潜在的问题&a…

区块链密码学:基础知识、应用与未来发展

一、引言 区块链技术&#xff0c;作为一种分布式、去中心化的数据管理方式&#xff0c;密码学在其安全性和可靠性方面发挥着至关重要的作用。本文将详细介绍区块链密码学的基础知识、应用以及未来发展趋势。 二、区块链密码学基础知识 区块链密码学是区块链技术的核心组成部分…

【海思SS528 | VO】MPP媒体处理软件V5.0 | VO模块编程总结

&#x1f601;博客主页&#x1f601;&#xff1a;&#x1f680;https://blog.csdn.net/wkd_007&#x1f680; &#x1f911;博客内容&#x1f911;&#xff1a;&#x1f36d;嵌入式开发、Linux、C语言、C、数据结构、音视频&#x1f36d; &#x1f923;本文内容&#x1f923;&a…

mysql5.7安装详细教程

文章目录 1 引言1.1 现有的数据存储方式有哪些&#xff1f;1.2 以上存储方式存在哪些缺点&#xff1f; 2 数据库2.1 概念2.2 数据库的分类 3 数据库管理系统3.1 概念3.2 常见数据库管理系统 4 MySQL4.1 简介4.2 访问与下载4.3 安装4.3.1 解压缩到非中文目录4.3.2 编写配置文件4…

一张图理解接口测试框架

测试框架先向测试数据库中插入测试数据&#xff08;如&#xff1a;name”Tom“&#xff09; 调用被测系统提供的接口&#xff08;传参&#xff1a;name”Tom“&#xff09; 从测试数据库中查到符合参数的数据 将查询到的数据组成Json格式&#xff0c;并返回给测试框架 提供…

【MySQL】:数据库基本认识

数据库基础 一.什么是数据库1.mysql是什么2.为什么要有数据库3.服务器&#xff0c;数据库&#xff0c;表关系4.Mysql架构5.SQL语句分类 二.存储引擎 一.什么是数据库 1.mysql是什么 1.mysql是数据库服务的客户端。 2.mysqld是数据库服务的服务器端。 3.mysql本质&#xff1a;基…

docker安装及配置mysql

docker 安装mysql 下载镜像文件 下载mysql5.7版本 sudo docker pull mysql:5.7检查是否下载成功 sudo docker images2.创建实例并启动 切换到root下避免每次使用sudo 密码&#xff1a;vagrant docker run -p 3306:3306 --name mysql \ -v /mydata/mysql/log:/var/log/my…

解决Error:You‘re using an RSA key with SHA-1, which is no longer allowed

一、问题 在微信开发者工具中&#xff0c;推送代码时发生错误Error:You‘re using an RSA key with SHA-1, which is no longer allowed...... 奇怪的是命令行可以正常push: 原因&#xff1a;因为生成密钥的RSA算法&#xff0c;由于安全性原因&#xff0c;现在已经不允许使用…