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

STM32--USART串口

文章目录

  • 通信接口
  • 串口通信
    • 硬件电路
    • 电平标准
    • 参数
    • 时序
  • USART
    • 主要特性
    • 框图
  • 数据帧
    • 发送器
  • 波特率发生器
  • SWART串口发送与接收工程
  • 串口收发数据包

通信接口

通信接口是指连接中央处理器(CPU)和标准通信子系统之间的接口,用于实现数据和控制信息在不同设备之间的传输和交换。通信接口可以是硬件或软件实现,其目的是使不同设备之间能够进行有效地通信。

在这里插入图片描述
上图是常见的通用通信类型。

双工指的是接口能够实现双向数据传输,即可以同时进行发送和接收数据的功能

半双工:数据在同一个通信信道上交替地进行双向传输,但不能同时进行发送和接收。发送方和接收方需要在不同时间段进行数据传输。

全双工:数据可以同时进行双向数据传输,发送方和接收方可以在同一时间段进行数据发送和接收。通信双方可以独立地进行发送和接收工作。

时钟是指在传输数据是否能达到同步传输。

同步传输:发送方和接收方之间使用一个共同的时钟信号来同步数据的传输。发送方根据时钟信号将数据按照固定的速率发送,接收方也按照相同的时钟信号来接收数据。
异步传输:数据的传输不依赖于共同的时钟信号,而是使用起始位、停止位和数据位之间的固定时间间隔进行同步。发送方会在发送数据之前先发送一个起始位作为同步信号,接收方通过检测起始位来确定数据的开始和结束。

电平是指传输信号时的电压信号形式。

单端传输:信号传输使用单个导线进行,信号的电压相对于某个参考电平变化来表示。这意味着信号只有一个极性,数据通过高低电平表示。在单端传输中,由于信号线未与其他导线耦合,容易受到电磁干扰的影响,可能导致信号质量的下降。
差分传输:信号的传输使用两个导线进行,信号通过两个相互补充的电压信号的差异来表示。差分传输具有很强的抗干扰能力,可以减少电磁干扰对信号的影响。

串口通信

串口是一种应用十分广泛的通讯接口,串口成本低、容易使用、通信线路简单,可实现两个设备的互相通信;适用于近距离的通信。
在这里插入图片描述

单片机的串口可以使单片机与单片机、单片机与电脑、单片机与各式各样的模块互相通信,极大地扩展了单片机的应用范围,增强了单片机系统的硬件实力。

硬件电路

在这里插入图片描述

简单双向串口通信有两根通信线(发送端TX和接收端RX)
TX与RX要交叉连接
当只需单向的数据传输时,可以只接一根通信线
当电平标准不一致时,需要加电平转换芯片

电平标准

电平标准是数据1和数据0的表达方式,是传输线缆中人为规定的电压与数据的对应关系,串口常用的电平标准有如下三种:
TTL电平:+3.3V或+5V表示1,0V表示0
RS232电平:-3~ -15V表示1,+3~ +15V表示0
RS485电平:两线压差+2~+6V表示1,-2 ~-6V表示0(差分信号)

参数

波特率:串口通信的速率,可以指定每秒传输的位数

起始位:标志一个数据帧的开始,固定为低电平

数据位:数据帧的有效载荷,1为高电平,0为低电平,低位先行

校验位:用于数据验证,根据数据位计算得来

常用奇偶校验位:通过在数据中添加一个附加位(校验位),以确保接收端可以检测到传输过程中的错误
奇校验:如果数据位中1的个数为偶数,则校验位设置为1,使得数据位和校验位中的1的总和保持奇数。如果数据位中1的个数为奇数,那么校验位置0。
偶校验:如果数据位中1的个数为偶数,则校验位置0,使得数据位和校验位的总个数为偶数。如果数据位中1的个数为奇数,校验位置1 。

停止位:用于数据帧间隔,固定为高电平

在这里插入图片描述

时序

在这里插入图片描述
输入不同的数据,显示不同的时序。

USART

通用同步异步收发器(USART)提供了一种灵活的方法与使用工业标准NRZ异步串行数据格式的外部设备之间进行全双工数据交换。 USART利用分数波特率发生器提供宽范围的波特率选择。

主要特性

1.全双工的,异步通信
2.分数波特率发生器系统
─ 发送和接收共用的可编程波特率,最高达4.5Mbits/s
3.可编程数据字长度(8位或9位)
4.可配置的停止位-支持1或2个停止位
5.校验控制
─ 发送校验位
─ 对接收数据进行校验
6.支持同步模式、硬件流控制、DMA、智能卡、IrDA、LIN

框图

在这里插入图片描述
先看左上角的接口处,TX与RX就是发送和接收的引脚,下面的三个接口是智能卡和IrDA通信的引脚。
发送脚就是通过发送移位寄存器送出去的,接收脚就是将接收数据送到接收移位寄存器的。

灰色部分就是串口的数据寄存器。有两个数据寄存器,一个用来发送数据,一个用来接收数据;在程序上,只表现为一个寄存器,它们寄存器共用地址进行存储
当你进行写入操作时,数据就写到TDR,当你进行读取操作时,数据就从RDR进行读出。
下面是移位寄存器,作用是把一个字节的数据一位一位进行位移。

假设你在某时刻给TDR写入数据0x66,在二进制存储就是01100110,此时硬件会检测到你写入的数据,就会检查移位寄存器是否有数据正在移位,如果没有01100110就会立刻放入移位寄存器中,此时也会置出一个标志位TXE,发送寄存器为空;有了这个标志我们就可以在TDR再写入一个数据了。移位寄存器在发送器控制的情况下,将数据一位一位进行传输到TX引脚进行输出(低位先出),当数据全部移位后,新的数据就会再次从TDR转移到移位寄存器中来,如果移位寄存器还没有完成,那么TDR会等移位寄存器完成移位之后才将数据转移。有了TDR和移位寄存器双重缓存,可以保证数据发送时,数据帧之间没有空闲。
而接收部分也是同样的道理,接收移位寄存器由接收器控制,低位先放到移位寄存器的高位,随着数据的增加而右移,当移位寄存器数据达到一个字节后,传输给接收寄存器RDR,此时也会有一个标志位RXNE置1,意味着接收寄存器有数据了,我们可以对DR寄存器的数据进行读走。

接着看到下面,有一个硬件数据流控,如果发送设备发送太快,接收设备来不及处理,可以通过流控来控制传输的速度。
它有两个引脚,一个是nRTS,另一个是nCTS。
nRTS是请求发送,是输出脚,就是告诉别人,我当前能不能接收;
nCTS是清除发送,是输入脚,用于接收别人nRTS的信号;

接着看右边的SCLK,这是一个产生同步的时钟信号,它是配合发送移位寄存器输出的,发送移位寄存器每发送一次,同步时钟电平就跳变一个周期,时钟会告诉对方我已经移出去一个数据了,你看要不要让我这个时钟信号来指导你接收一下?当然这个时钟只支持输出,不支持输入,所以两个SWART不能实现同步的串口通信。
主要用途是兼容别的协议或者做自适应波特率。

接着看到中间的唤醒单元,这部分的作用是实现串口挂载多个设备。
中断控制,支持对标志位标志的地方进行中断。

最下面的波特率发生器部分,其实就是分频器,对APB时钟进行分频,得到发送和接收移位的时钟。

数据帧

字长可以通过编程USART_CR1寄存器中的M位,选择成8或9位(见图249)。在起始位期间, TX脚处于低电平,在停止位期间处于高电平。
在这里插入图片描述

发送器

发送器根据M位的状态发送8位或9位的数据字。当发送使能位(TE)被设置时,发送移位寄存器中的数据在TX脚上输出,相应的时钟脉冲在CK脚上输出。
USART支持多种停止位的配置: 0.5、 1、 1.5和2个停止位。
在这里插入图片描述

波特率发生器

发送器和接收器的波特率由波特率寄存器BRR里的DIV确定
计算公式:波特率 = fPCLK2/1 / (16 * DIV)
在这里插入图片描述
假设我们要输出波特率为9600,那么通过计算DIV=72M/16/9600=468.75
所以置于波特率寄存器中的值为468.75.

SWART串口发送与接收工程

OLED代码链接入口

接线方式:
在这里插入图片描述
发送时需要一个串口助手在电脑来显示内容,我们将实现STM32与电脑之间的数据传输。以STM32为主机。

Serial.h

#ifndef __SERIAL_H__
#define __SERIAL_H__#include <stdio.h>void Serial_Init();
void Serial_SendByte(uint8_t Byte);
void Serial_SendArray(uint8_t* Arr,uint8_t Lenth);
void Serial_SendString(char* String);
void Serial_SendNumber(uint32_t Number,uint8_t Lenth);
uint8_t Serial_GetRxFlag();
uint8_t Serial_GetRxData();#endif 

Serial.c

#include "stm32f10x.h"                  // Device header
#include <stdio.h>
#include <stdarg.h>uint8_t Serial_RxData;
uint8_t Serial_RxFlag;void Serial_Init()
{RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);GPIO_InitTypeDef GPIO_InitStructure;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_InitTypeDef USART_InitStructure;USART_InitStructure.USART_BaudRate=9600; //配置波特率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);//配置usart开启中断USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//中断源RXNENVIC_InitTypeDef NVIC_InitStructure;NVIC_InitStructure.NVIC_IRQChannel=USART1_IRQn;NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;NVIC_InitStructure.NVIC_IRQChannelSubPriority=1;NVIC_Init(&NVIC_InitStructure);USART_Cmd(USART1,ENABLE);
}//发送字节
void Serial_SendByte(uint8_t Byte)
{USART_SendData(USART1,Byte);while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);
}//发送数组
void Serial_SendArray(uint8_t* Arr,uint8_t Lenth)
{uint8_t i;for(i=0;i<Lenth;i++){Serial_SendByte(Arr[i]);}
}//发送字符串
void Serial_SendString(char* String)
{uint8_t i;for(i=0;String[i]!='\0';i++){Serial_SendByte(String[i]);}
}uint32_t Serial_Pow(uint32_t X,uint32_t Y)
{uint32_t Result=1;while(Y--){Result*=X;}return Result;
}void Serial_SendNumber(uint32_t Number,uint8_t Lenth)
{uint8_t i;for(i=0;i<Lenth;i++){Serial_SendByte(Number/Serial_Pow(10,Lenth-1-i)%10+'0');}
}//获取接收状态
uint8_t Serial_GetRxFlag()
{if(Serial_RxFlag==1){Serial_RxFlag=0;return 1;}return 0;}
//获取接收数据
uint8_t Serial_GetRxData()
{return Serial_RxData;
}void USART1_IRQHandler()
{if(USART_GetITStatus(USART1,USART_IT_RXNE)==SET){Serial_RxData=USART_ReceiveData(USART1);Serial_RxFlag=1;USART_ClearITPendingBit(USART1,USART_IT_RXNE);}
}

初始化输出需要配置复用功能,因为SWART为片上外设。
发送字节需要用TXE标志位来表示已经从DR寄存器发送到移位寄存器中。所以用到while来进行等待;
接收时,利用RNXE产生的标志位来产生中断,在中断函数中获取DR寄存器的数据。

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "Buzzer.h"
#include "Serial.h"
#include "OLED.h"
int main()
{uint8_t Rxdata;OLED_Init();Serial_Init();OLED_ShowString(1,1,"RxData:");while(1){if(Serial_GetRxFlag()==1){Rxdata=Serial_GetRxData();Serial_SendByte(Rxdata);OLED_ShowHexNum(1,8,Rxdata,2);}}}

串口收发数据包

连接方式:
在这里插入图片描述

利用连续的数据序列来对LED灯的亮灭进行控制;

Serial.h

#ifndef __SERIAL_H__
#define __SERIAL_H__#include <stdio.h>
#include <string.h>extern uint8_t Serial_RxFlag;
extern char Serial_RxPacket[];void Serial_Init();
void Serial_SendByte(uint8_t Byte);
void Serial_SendArray(uint8_t* Arr,uint8_t Lenth);
void Serial_SendString(char* String);
void Serial_SendNumber(uint32_t Number,uint8_t Lenth);
void Serial_Printf(char* format,...);#endif 

Serial.c

#include "stm32f10x.h"                  // Device header
#include <stdio.h>
#include <stdarg.h>char Serial_RxPacket[100];
uint8_t Serial_RxFlag;void Serial_Init()
{RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);GPIO_InitTypeDef GPIO_InitStructure;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_InitTypeDef USART_InitStructure;USART_InitStructure.USART_BaudRate=9600; //配置波特率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);//配置usart开启中断USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//中断源RXNENVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);NVIC_InitTypeDef NVIC_InitStructure;NVIC_InitStructure.NVIC_IRQChannel=USART1_IRQn;NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;NVIC_InitStructure.NVIC_IRQChannelSubPriority=1;NVIC_Init(&NVIC_InitStructure);USART_Cmd(USART1,ENABLE);
}//发送字节
void Serial_SendByte(uint8_t Byte)
{USART_SendData(USART1,Byte);while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);
}//发送数组
void Serial_SendArray(uint8_t* Arr,uint8_t Lenth)
{uint8_t i;for(i=0;i<Lenth;i++){Serial_SendByte(Arr[i]);}
}//发送字符串
void Serial_SendString(char* String)
{uint8_t i;for(i=0;String[i]!='\0';i++){Serial_SendByte(String[i]);}
}uint32_t Serial_Pow(uint32_t X,uint32_t Y)
{uint32_t Result=1;while(Y--){Result*=X;}return Result;
}
//发送数字
void Serial_SendNumber(uint32_t Number,uint8_t Lenth)
{uint8_t i;for(i=0;i<Lenth;i++){Serial_SendByte(Number/Serial_Pow(10,Lenth-1-i)%10+'0');}
}void Serial_Printf(char* format,...)
{char String[100];va_list arg;va_start(arg,format);vsprintf(String,format,arg);va_end(arg);Serial_SendString(String);
}//串口接收中断(接收一位数据中断一次),接收数据包
void USART1_IRQHandler()
{static uint8_t RxState=0; //获取当前状态static uint8_t pRxPacket=0; //数据包有效位下标if(USART_GetITStatus(USART1,USART_IT_RXNE)==SET){uint8_t RxData=USART_ReceiveData(USART1);if(RxState==0){if(RxData=='&'&& Serial_RxFlag==0){RxState=1;pRxPacket=0;}}else if(RxState==1){            if(RxData=='\r'){RxState=2;}else{Serial_RxPacket[pRxPacket++]=RxData;}}else if(RxState==2){if(RxData=='\n'){RxState=0;Serial_RxPacket[pRxPacket]='\0';Serial_RxFlag=1;}}USART_ClearITPendingBit(USART1,USART_IT_RXNE);}
}

对于接收的数据包,利用了一种状态机的方式进行接收
在这里插入图片描述
通过包头和包尾对数据有效位进行保护,可以让一连串重复的数据可以找到数据位,避免找不到数据位的头的情况。

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "LED.h"
#include "Serial.h"
#include "OLED.h"
int main()
{OLED_Init();Serial_Init();LED_Init();OLED_ShowString(1, 1, "RxPacket:");while(1){if(Serial_RxFlag==1){if(strcmp(Serial_RxPacket,"LED_ON")==0){LED1_ON();OLED_ShowString(2,1,"LED_ON_OK   ");Serial_SendString("LED_ON_OK");}else if(strcmp(Serial_RxPacket,"LED_OFF")==0){LED1_OFF();OLED_ShowString(2,1,"LED_OFF_OK   ");Serial_SendString("LED_OFF_OK");}else{OLED_ShowString(2,1,"ERROR_COMMAND");Serial_SendString("ERROR_COMMAND");}Serial_RxFlag=0;}}}

LED.c

#include "stm32f10x.h"                  // Device headervoid LED_Init(void)
{RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_2;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_SetBits(GPIOA, GPIO_Pin_1 | GPIO_Pin_2);
}void LED1_ON(void)
{GPIO_ResetBits(GPIOA, GPIO_Pin_1);
}void LED1_OFF(void)
{GPIO_SetBits(GPIOA, GPIO_Pin_1);
}void LED1_Turn(void)
{if (GPIO_ReadOutputDataBit(GPIOA, GPIO_Pin_1) == 0){GPIO_SetBits(GPIOA, GPIO_Pin_1);}else{GPIO_ResetBits(GPIOA, GPIO_Pin_1);}
}void LED2_ON(void)
{GPIO_ResetBits(GPIOA, GPIO_Pin_2);
}void LED2_OFF(void)
{GPIO_SetBits(GPIOA, GPIO_Pin_2);
}void LED2_Turn(void)
{if (GPIO_ReadOutputDataBit(GPIOA, GPIO_Pin_2) == 0){GPIO_SetBits(GPIOA, GPIO_Pin_2);}else{GPIO_ResetBits(GPIOA, GPIO_Pin_2);}
}

LED.h

#ifndef __LED_H__
#define __LED_H__void LED_Init();
void LED1_ON();
void LED1_OFF();
void LED1_Turn();
void LED2_ON();
void LED2_OFF();
void LED2_Turn();#endif

相关文章:

STM32--USART串口

文章目录 通信接口串口通信硬件电路电平标准参数时序 USART主要特性框图 数据帧发送器 波特率发生器SWART串口发送与接收工程串口收发数据包 通信接口 通信接口是指连接中央处理器&#xff08;CPU&#xff09;和标准通信子系统之间的接口&#xff0c;用于实现数据和控制信息在不…...

2023年Java毕业设计题目推荐,怎样选题?500道毕业设计题目推荐

大家好&#xff0c;我是程序员徐师兄&#xff0c;最近有很多同学咨询&#xff0c;说毕业设计了&#xff0c;不知道选怎么题目好&#xff0c;有哪些是想需要注意的。 今天&#xff0c;我整理了一些Java毕业设计的题目,可以参考一下&#xff0c;希望对大家有所帮助 文章目录 一、…...

基于数据湖的多流拼接方案-HUDI概念篇

目录 一、为什么需要HUDI&#xff1f; 1. 传统技术选型存在哪些问题&#xff1f; 2. Hudi有什么优点&#xff1f; 基于 Hudi Payload 机制的多流拼接方案&#xff1a; 二、HUDI的应用场景 1. 什么场景适合使用hudi&#xff1f; 2. 什么场景不适合使用hudi&#xff1f; …...

OpenCV基础知识(5)— 几何变换

前言&#xff1a;Hello大家好&#xff0c;我是小哥谈。OpenCV中的几何变换是指改变图像的几何结构&#xff0c;例如大小、角度和形状等&#xff0c;让图像呈现出缩放、翻转、旋转和透视效果。这些几何变换操作都涉及复杂、精密的计算。OpenCV将这些计算过程都封装成了非常灵活的…...

Linux下源码安装MySQL 8.0

MySQL 8.0源码安装 环境准备步骤 环境准备 Linux环境&#xff0c;本文基于CentOS 8 MySQL安装包&#xff0c;本文基于MySQL 8.1&#xff0c;以下为带boost MySQL 8.1源码下载地址&#xff1a; https://dev.mysql.com/get/Downloads/MySQL-8.1/mysql-boost-8.1.0.tar.gz 步骤…...

大聪明教你学Java | 深入浅出聊 Java 内存模型

前言 🍊作者简介: 不肯过江东丶,一个来自二线城市的程序员,致力于用“猥琐”办法解决繁琐问题,让复杂的问题变得通俗易懂。 🍊支持作者: 点赞👍、关注💖、留言💌~ 在多线程环境下,多个线程同时访问共享数据可能导致一系列问题,如数据不一致、竞态条件和死锁等…...

SAP ABAPG开发屏幕自动生成日期的搜索帮助

代码如下&#xff1a; REPORT z_jason_test_f4 . TABLES: s031. PARAMETER p_spmon TYPE spmon DEFAULT sy-datum0(6) OBLIGATORY. SELECT-OPTIONS s_spmon FOR s031-spmon DEFAULT sy-datum0(6) OBLIGATORY. AT SELECTION-SCREEN ON VALUE-REQUEST…...

leetcode 674. 最长连续递增序列

2023.8.24 与最长递增子序列 类似&#xff0c;不同的是&#xff0c; 本题要求连续序列&#xff0c;所以不需要第二层遍历比较之前所有的元素了&#xff0c;只需要比较上一个元素i-1。 dp[i]的含义为&#xff1a;以nums[i]元素为结尾的序列的最长递增子序列。 注意这里是以i为结…...

Mysql简短又易懂

MySql 连接池:的两个参数 最大连接数&#xff1a;可以同时发起的最大连接数 单次最大数据报文&#xff1a;接受数据报文的最大长度 数据库如何存储数据 存储引擎&#xff1a; InnoDB:通过执行器对内存和磁盘的数据进行写入和读出 优化SQL语句innoDB会把需要写入或者更新的数…...

vue 简单实验 v-model 变量和htm值双向绑定

1.代码 <script src"https://unpkg.com/vuenext" rel"external nofollow" ></script> <div id"two-way-binding"><p>{{ message }}</p><input v-model"message" /> </div> <script>…...

测试框架pytest教程(8)失败重试-pytest-rerunfailures

pytest-rerunfailures是一个pytest插件&#xff0c;用于重新运行失败的测试用例。当测试用例在第一次运行时失败&#xff0c;该插件会自动重新运行指定次数的失败用例&#xff0c;以提高稳定性和减少偶发性错误的影响。 要使用pytest-rerunfailures插件&#xff0c;需要按照以…...

6个主流的工业3D管道设计软件

3D 管道设计软件是大多数行业工程工作的主要部分&#xff0c;例如&#xff1a; 电力、石油和天然气、石化、炼油厂、纸浆和造纸、化学品和加工业。 全球各工程公司使用了近 50 种工厂或管道设计软件。 每个软件都有优点和缺点&#xff0c;包括价格点。 EPC 和业主部门当前的趋势…...

基于微信小程序的垃圾分类系统设计与实现(2.0 版本,附前后端代码)

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝30W、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 1 简介 视频演示地址&#xff1a; 基于微信小程序的智能垃圾分类回收系统&#xff0c;可作为毕业设计 小…...

基础论文学习(4)——CLIP

《Learning Transferable Visual Models From Natural Language Supervision》 CLIP的英文全称是Contrastive Language-Image Pre-training&#xff0c;即一种基于对比文本-图像对的预训练模型。CLIP是一种基于对比学习的多模态模型&#xff0c;与CV中的一些对比学习方法如moc…...

SpringBoot利用ConstraintValidator实现自定义注解校验

一、前言 ConstraintValidator是Java Bean Validation&#xff08;JSR-303&#xff09;规范中的一个接口&#xff0c;用于实现自定义校验注解的校验逻辑。ConstraintValidator定义了两个泛型参数&#xff0c;分别是注解类型和被校验的值类型。在实现ConstraintValidator接口时&…...

十、pikachu之php反序列化

文章目录 1、php反序列化概述2、实战3、关于Magic function4、__wakeup()和destruct() 1、php反序列化概述 在理解这个漏洞前&#xff0c;首先搞清楚php中serialize()&#xff0c;unserialize()这两个函数。 &#xff08;1&#xff09;序列化serialize()&#xff1a;就是把一个…...

PHP“牵手”拼多多商品详情数据获取方法,拼多多API接口批量获取拼多多商品详情数据说明

拼多多商品详情接口 API 是开放平台提供的一种 API 接口&#xff0c;它可以帮助开发者获取拼多多商品的详细信息&#xff0c;包括商品的标题、描述、图片等信息。在拼多多电商平台的开发中&#xff0c;拼多多详情接口 API 是非常常用的 API&#xff0c;因此本文将详细介绍拼多多…...

前端面试:【Redux】状态管理的精髓

嘿&#xff0c;亲爱的Redux探险家&#xff01;在前端开发的旅程中&#xff0c;有一个强大的状态管理工具&#xff0c;那就是Redux。Redux是一个状态容器&#xff0c;它以一种可预测的方式管理应用的状态&#xff0c;通过Store、Action、Reducer、中间件和异步处理等核心概念&am…...

element-ui中的el-table的summary-method(合计)的使用

场景图片&#xff1a; 图片1&#xff1a; 图片2&#xff1a; 一&#xff1a;使用element中的方法 优点&#xff1a; 直接使用summary-method方法&#xff0c;直接&#xff0c;方便 缺点&#xff1a; 只是在表格下面添加了一行&#xff0c;如果想有多行就不行了 1&#xff1a;h…...

“深入探索JVM:解析Java虚拟机的工作原理与性能优化“

标题&#xff1a;深入探索JVM&#xff1a;解析Java虚拟机的工作原理与性能优化 摘要&#xff1a;本文将深入探讨Java虚拟机&#xff08;JVM&#xff09;的工作原理和性能优化。我们将首先介绍JVM的基本组成和工作流程&#xff0c;然后重点讨论JVM内存管理、垃圾回收算法以及性…...

全面掌握ESP WiFi中继器DHCP服务器配置:高效管理嵌入式设备网络

全面掌握ESP WiFi中继器DHCP服务器配置&#xff1a;高效管理嵌入式设备网络 【免费下载链接】esp_wifi_repeater A full functional WiFi Repeater (correctly: a WiFi NAT Router) 项目地址: https://gitcode.com/gh_mirrors/es/esp_wifi_repeater ESP WiFi中继器是一款…...

孟德尔随机化实战(五)—— 告别报错!Error in if (out == “[]“) 深度解析与TwoSampleMR参数调优全攻略

1. 报错现象深度解析&#xff1a;为什么会出现"参数长度为零"&#xff1f; 最近在孟德尔随机化分析交流群里&#xff0c;这个报错出现的频率简直高得离谱&#xff1a;"Error in if (out "[]") { : argument is of length zero"或者它的中文版&q…...

centos7安装MySQL8.4手册

目录前言一、首先更新插件&#xff0c;并查看当前系统版本二、安装步骤--在线安装1、创建mysql目录2、安装rpm包3、安装 mysql-community-server4、启动MySQL服务5、查看MySQL状态6、设置开机自启动三、查看默认密码四、登录mysql五、修改密码六、开启远程访问1. 修改 MySQL 配…...

各行业开发经验全面解析,本凡科技助你快速提升项目成功率

在当今快速发展的市场中&#xff0c;各行业的开发经验已成为决定项目成败的关键因素。每个行业都面临独特的挑战和需求&#xff0c;了解这些特性有助于企业制定有效的开发策略。例如&#xff0c;科技行业通常需要快速响应市场变化&#xff0c;而食品行业则需关注合规性和安全标…...

AI编程实战:如何用Cursor和Coze在1小时内完成文生图小程序开发

AI编程实战&#xff1a;如何用Cursor和Coze在1小时内完成文生图小程序开发 当产品灵感突然闪现&#xff0c;如何在最短时间内将它变成可交互的原型&#xff1f;传统开发流程中&#xff0c;从UI设计到API对接至少需要数天时间。而现在&#xff0c;借助AI编程工具链&#xff0c;我…...

本地图片检索新方案:ImageSearch完全使用指南

本地图片检索新方案&#xff1a;ImageSearch完全使用指南 【免费下载链接】ImageSearch 基于.NET8的本地硬盘千万级图库以图搜图案例Demo和图片exif信息移除小工具分享 项目地址: https://gitcode.com/gh_mirrors/im/ImageSearch 当你的电脑中存储了成千上万张图片&…...

医学影像与卫星图的救星?深入聊聊JPEG-LS算法在边缘计算设备上的应用优势

JPEG-LS算法&#xff1a;边缘计算时代的医学影像与卫星图像压缩利器 当一台CT扫描仪每秒产生数百张16位深度的医学影像&#xff0c;或一颗遥感卫星每天传回数TB的高清地表数据时&#xff0c;传统的图像压缩方案往往面临两难选择——要么牺牲宝贵的诊断细节&#xff0c;要么耗尽…...

RGBLEDBlender:嵌入式RGB LED色彩混合与动态控制框架

1. RGBLEDBlender 库深度解析&#xff1a;面向嵌入式系统的 RGB 色彩混合与动态控制框架RGBLEDBlender 是一个轻量级、面向硬件的 RGB LED 色彩混合库&#xff0c;专为资源受限的微控制器平台&#xff08;尤其是 Arduino 生态&#xff09;设计。该库由 Erik Sikich 于 2016 年 …...

WiFi热图绘制工具:用Python为你的无线网络做一次“CT扫描“ [特殊字符][特殊字符]

WiFi热图绘制工具&#xff1a;用Python为你的无线网络做一次"CT扫描" &#x1f3e5;&#x1f4f6; 【免费下载链接】wifi-heat-mapper whm also known as wifi-heat-mapper is a Python library for benchmarking Wi-Fi networks and gather useful metrics that can…...

用MediaPipe和Python做个隔空切水果游戏:从手势骨架提取到简单游戏逻辑实现

用MediaPipe和Python打造体感切水果游戏&#xff1a;从手势识别到游戏逻辑全解析 还记得小时候在街机厅玩《水果忍者》的畅快感吗&#xff1f;现在&#xff0c;我们完全可以用Python和MediaPipe技术&#xff0c;在电脑前通过手势隔空切水果&#xff01;本文将带你从零开始&…...