USART串口协议

USART串口协议

文章目录

  • USART串口协议
    • 1. 通信接口
    • 2.串口通信
      • 2.1硬件电路
      • 2.2电平标准
      • 2.3串口参数及时序(软件部分)
    • 3.USART串口外设
      • 3.1串口外设
      • 3.2USART框图
      • 3.3USART基本结构
      • 3.4数据帧
    • 4.输入电路
      • 4.1起始位侦测
      • 4.2数据采样
    • 5.波特率发生器
    • 6.相关函数介绍
    • 7.串口发送
      • 7.1接线图
      • 7.2代码编写
        • 7.2.1主程序代码main.c
        • 7.2.2函数定义Serial.c
        • 7.2.3函数定义Serial.h
    • 8.串口发送+接收
      • 8.1接线图
      • 8.2代码编写
        • 8.2.1主程序代码
        • 8.2.2函数定义
        • 8.2.3函数定义
    • n.总结
      • n.1USART实现步骤

1. 通信接口

  • 通信的目的:将一个设备的数据传送到另一个设备,扩展硬件系统
  • 通信协议:制定通信的规则,通信双方按照协议规则进行数据收发

stm32中的通信协议(只列出最典型的参数)

名称引脚双工时钟电平设备说明
USARTTX、RX全双工异步单端点对点TX是数据发送引脚,TP是数据接收引脚
I2CSCL、SDA半双工同步单端多设备SCL是时钟,SDA是数据
SPISCLK、MOSI、MISO、CS全双工同步单端多设备MOSI主机输出数据脚,MISO主机输入数据脚,CS片选,用于指定通信的对象
CANCAN_H、CAN_L半双工异步差分多设备差分数据脚,用两个数据表示一个差分数据
USBDP、DM半双工异步差分点对点D+,D-,也是一对差分数据脚

全双工:就是指通信双方能够同时进行双向通信,一般来说全双工的通信都有两根通信线,表中的两个是合成一根的

单工:是指数据只能从一个设备到另一个设备,而不能反着来

同步:有单独的时钟线,可以在时钟信号的指引下进行采样

异步:没有时钟线,需要双方约定一个采样频率,且还要加一些帧头帧率等,进行采样位置对齐

单端:引脚的高低电平都是对GND的电压差,所以单端通信的双方需要共地,就是把GND接在一起

差分:靠两个引脚的电压差来传输信号的,通信的时候可以不需要GND,USB有些地方需要单端信号。优点:使用差分信号可以极大地提高抗干扰特性,所以差分信号一般传输信号和距离都会非常高

2.串口通信

  • 串口是一种应用十分广泛的通讯接口,串口成本低、容易使用、通信线路简单,可实现两个设备的互相通信
  • 单片机的串口可以使单片机与单片机、单片机与电脑、单片机与各式各样的模块互相通信,极大地扩展了单片机的应用范围,增强了单片机系统的硬件实力

串口转USB为例:

串口USB

2.1硬件电路

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

硬件电路

2.2电平标准

  • 电平标准是数据1和数据0的表达方式,是传输线缆中人为规定的电压与数据的对应关系,串口常用的电平标准有如下三种:

  • TTL电平:+3.3V或+5V表示1,0V表示0

  • RS232电平:-3-15V表示1,+3+15V表示0

  • RS485电平:两线压差+2+6V表示1,-2-6V表示0(差分信号),该电平可实现远距离通信

2.3串口参数及时序(软件部分)

  • 波特率:串口通信的速率,每秒传码元的个数,单位是码元每秒/s,或者直接叫波特(Baud)(发送和接收必须约定好速率)
  • 起始位:标志一个数据帧的开始,固定为低电平
  • 数据位:数据帧的有效载荷,1为高电平,0为低电平,低位先行
  • 校验位:用于数据验证,根据数据位计算得来
  • 停止位:用于数据帧间隔,固定为高电平

串口数据

串口时序

串口时序

总结:TX引脚定时输出翻转的高低电平,RX引脚定时读取引脚的高低电平

字节数据的传递:每个字节的数据+起始位+停止位+可选校验位,打包成数据帧,依次输出在TX引脚,另一端RX引脚依次接收

3.USART串口外设

3.1串口外设

  • USART(Universal Synchronous/Asynchronous Receiver/Transmitter)通用同步/异步收发器
  • USART是STM32内部集成的硬件外设,可根据数据寄存器的一个字节数据自动生成数据帧时序,从TX引脚发送出去,也可自动接收RX引脚的数据帧时序,拼接为一个字节数据,存放在数据寄存器里
  • 自带波特率发生器,最高达4.5Mbits/s
  • 可配置数据位长度(8/9)、停止位长度(0.5/1/1.5/2)
  • 可选校验位(无校验/奇校验/偶校验)
  • 支持同步模式、硬件流控制、DMA、智能卡、IrDA、LIN
  • STM32F103C8T6 USART资源: USART1、 USART2、 USART3

硬件控制流:例A向B发送数据,A发的太快了,如果B没有硬件流控制,那么B只有抛弃新数据或覆盖原数据。如果有硬件流控制,在硬件电路上,会多出一根线。如果B没有准备好,就置高电平,如果B准备好了就置低电平,A在接收到B的反馈后会做出反应

3.2USART框图

USART框图

3.3USART基本结构

USART基本结构

3.4数据帧

数据帧1

4.输入电路

4.1起始位侦测

起始位侦测

4.2数据采样

数据采样

5.波特率发生器

  • 发送器和接收器的波特率由波特率寄存器BRR里的DIV确定
  • 计算公式:波特率 = f PCLK2/1 f_{\text{PCLK2/1}} fPCLK2/1/ (16 * DIV)

波特率发生器

6.相关函数介绍

没解释

void USART_DeInit(USART_TypeDef* USARTx);
void USART_Init(USART_TypeDef* USARTx, USART_InitTypeDef* USART_InitStruct);
void USART_StructInit(USART_InitTypeDef* USART_InitStruct);

用来配置同步时钟输出的,包括时钟是不是要输出,时钟的极性相位等参数

void USART_ClockInit(USART_TypeDef* USARTx, USART_ClockInitTypeDef* USART_ClockInitStruct);
void USART_ClockStructInit(USART_ClockInitTypeDef* USART_ClockInitStruct);
void USART_ITConfig(USART_TypeDef* USARTx, uint16_t USART_IT, FunctionalState NewState);//开启RXNE标志位的通道
void USART_DMACmd(USART_TypeDef* USARTx, uint16_t USART_DMAReq, FunctionalState NewState);//开启USART到DMA的触发通道
void USART_SendData(USART_TypeDef* USARTx, uint16_t Data);//发送数据
uint16_t USART_ReceiveData(USART_TypeDef* USARTx);//接收数据
FlagStatus USART_GetFlagStatus(USART_TypeDef* USARTx, uint16_t USART_FLAG);//获取标志位状态
void USART_ClearFlag(USART_TypeDef* USARTx, uint16_t USART_FLAG);

7.串口发送

7.1接线图

串口发射接线图

7.2代码编写

7.2.1主程序代码main.c
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "LED.h"
#include "KEY.h"
#include "OLED.h"
//#include "OLED_Font.h"
#include "Serial.h"int main(void){OLED_Init();Serial_Init();Serial_SendByte(0x41);uint8_t MyArray[] = {0x42, 0x43, 0x44, 0x45};//定义一个数组Serial_SendArray(MyArray, 4);Serial_SendString("helloWorld!\n");Serial_SendNumber(123456,6);//方法1:重定向printf("Num2=%d\r\n", 222);//需要重定向fputc函数,并在工程选项里勾选Use MicroLIB/*方法2:使用sprintf打印到字符数组,再用串口发送字符数组,此方法打印到字符数组,之后想怎么处理都可以,可在多处使用*/char String[100];					//定义字符数组sprintf(String, "\r\nNum3=%d", 333);//使用sprintf,把格式化字符串打印到字符数组Serial_SendString(String);			//串口发送字符数组(字符串)/*方法3:将sprintf函数封装起来,实现专用的printf,此方法就是把方法2封装起来,更加简洁实用,可在多处使用*/Serial_Printf("\r\nNum4=%d", 444);	//串口打印字符串,使用自己封装的函数实现printf的效果Serial_Printf("\r\n");while(1){}
}
7.2.2函数定义Serial.c
#include "stm32f10x.h"                  // Device header
#include <stdio.h>
#include <stdarg.h>void Serial_Init(void){RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA	,ENABLE);//GPIO初始化,输出只需要初始化一个GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;GPIO_Init(GPIOA, &GPIO_InitStructure);USART_InitTypeDef USART_InitStructure;USART_InitStructure.USART_BaudRate = 9600;//波特率(我们给出函数会自己计算)USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//硬件流控制(直接复制参数名称,开启代码提示CTRL+ALT+空格)USART_InitStructure.USART_Mode = USART_Mode_Tx;USART_InitStructure.USART_Parity = USART_Parity_No;//检验位USART_InitStructure.USART_StopBits = USART_StopBits_1;//停止位,0.5、1、1.5、2USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长,可选8或9位USART_Init(USART1,&USART_InitStructure);//开启USART_Cmd(USART1,ENABLE);
}//发送一个Byte
void Serial_SendByte(uint8_t Byte){USART_SendData(USART1,Byte);//发送数据寄存器空标志位,等待TXE置1,标志位不需要手动清除//USART_FLAG_TXE 是发送数据寄存器空标志位(Transmit data register empty)。当该标志位被置 1 时,表示 USART 的发送数据寄存器为空,可以继续写入新的数据;//当该标志位为 0(RESET)时,表示发送数据寄存器仍有数据,需要等待。while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);
}//遍历数组,并发送
void Serial_SendArray(uint8_t *Array,uint16_t Length){uint16_t i;for(i=0;i<Length;i++){Serial_SendByte(Array[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 = Result*x;}return Result;
}//分别发送拆分的数
void Serial_SendNumber(uint32_t Number,uint8_t Length){uint8_t i;for(i=0;i<Length;i++){Serial_SendByte(Number/Serial_Pow(10,Length-i-1)%10+'0');/*在串口通信中,通常需要发送字符的 ASCII 码。数字 0 - 9 的 ASCII 码是连续的,字符 '0' 的 ASCII 码值为 48。将数字加上 '0' 可以将其转换为对应的 ASCII 码字符。例如:数字 1 加上 '0' 得到字符 '1',其 ASCII 码值为 49。数字 2 加上 '0' 得到字符 '2',其 ASCII 码值为 50。数字 3 加上 '0' 得到字符 '3',其 ASCII 码值为 51。*/}
}//printf的底层代码,显示信息需要经过该函数
int fputc(int ch,FILE *f){Serial_SendByte(ch);return ch;
}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);
}	
7.2.3函数定义Serial.h
#ifndef __SERIAL_H
#define __SERIAL_H#include <stdio.h>void Serial_Init(void);
void Serial_SendByte(uint8_t Byte);
void Serial_SendArray(uint8_t *Array,uint16_t Length);
void Serial_SendString(char *String);
uint32_t Serial_Pow(uint32_t x,uint32_t y);
void Serial_SendNumber(uint32_t Number,uint8_t Length);
int fputc(int ch,FILE *f);
void Serial_Printf(char *format,...);#endif

8.串口发送+接收

8.1接线图

串口发送+接收

8.2代码编写

8.2.1主程序代码
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "LED.h"
#include "KEY.h"
#include "OLED.h"
//#include "OLED_Font.h"
#include "Serial.h"uint8_t RxData;int main(void){OLED_Init();Serial_Init();OLED_ShowString(1,1,"RxData:");while(1){//不断的判断RXNE标志位
//		if(USART_GetFlagStatus(USART1,USART_FLAG_RXNE)==SET){
//			RxData = USART_ReceiveData(USART1);
//			OLED_ShowHexNum(1,1,RxData,2);
//		}if(Serial_GetRxFlag() == 1){RxData = Serial_GetRxData();Serial_SendByte(RxData);OLED_ShowHexNum(1,8,RxData,2);}
}}
8.2.2函数定义
#include "stm32f10x.h"                  // Device header
#include <stdio.h>
#include <stdarg.h>uint8_t Serial_RxFlag;
uint8_t Serial_RxData;void Serial_Init(void){RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA	,ENABLE);//GPIO初始化GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;GPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;GPIO_Init(GPIOA, &GPIO_InitStructure);USART_InitTypeDef USART_InitStructure;USART_InitStructure.USART_BaudRate = 9600;//波特率(我们给出函数会自己计算)USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//硬件流控制(直接复制参数名称,开启代码提示CTRL+ALT+空格)USART_InitStructure.USART_Mode = USART_Mode_Tx |USART_Mode_Rx;USART_InitStructure.USART_Parity = USART_Parity_No;//检验位USART_InitStructure.USART_StopBits = USART_StopBits_1;//停止位,0.5、1、1.5、2USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长,可选8或9位USART_Init(USART1,&USART_InitStructure);//开启RXNE到NVIC的通道USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//开启NVIC的通道NVIC_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);
}//标志位
uint8_t Serial_GetRxFlag(void)
{if(Serial_RxFlag == 1){Serial_RxFlag = 0;return 1;//返回1,并自动清零标志位}return 0;
}//Rx封装
uint8_t Serial_GetRxData(void)
{return Serial_RxData;
}//中断函数
void USART1_IRQHandler(void){if(USART_GetITStatus(USART1,USART_IT_RXNE) == SET){Serial_RxData = USART_ReceiveData(USART1);Serial_RxFlag = 1;USART_ClearITPendingBit(USART1,USART_IT_RXNE);}
}//发送一个Byte
void Serial_SendByte(uint8_t Byte){USART_SendData(USART1,Byte);//发送数据寄存器空标志位,等待TXE置1,标志位不需要手动清除//USART_FLAG_TXE 是发送数据寄存器空标志位(Transmit data register empty)。当该标志位被置 1 时,表示 USART 的发送数据寄存器为空,可以继续写入新的数据;//当该标志位为 0(RESET)时,表示发送数据寄存器仍有数据,需要等待。while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);
}//遍历数组,并发送
void Serial_SendArray(uint8_t *Array,uint16_t Length){uint16_t i;for(i=0;i<Length;i++){Serial_SendByte(Array[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 = Result*x;}return Result;
}//分别发送拆分的数
void Serial_SendNumber(uint32_t Number,uint8_t Length){uint8_t i;for(i=0;i<Length;i++){Serial_SendByte(Number/Serial_Pow(10,Length-i-1)%10+'0');/*在串口通信中,通常需要发送字符的 ASCII 码。数字 0 - 9 的 ASCII 码是连续的,字符 '0' 的 ASCII 码值为 48。将数字加上 '0' 可以将其转换为对应的 ASCII 码字符。例如:数字 1 加上 '0' 得到字符 '1',其 ASCII 码值为 49。数字 2 加上 '0' 得到字符 '2',其 ASCII 码值为 50。数字 3 加上 '0' 得到字符 '3',其 ASCII 码值为 51。*/}
}//printf的底层代码,显示信息需要经过该函数
int fputc(int ch,FILE *f){Serial_SendByte(ch);return ch;
}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);
}	
8.2.3函数定义
#ifndef __SERIAL_H
#define __SERIAL_H#include <stdio.h>void Serial_Init(void);
void Serial_SendByte(uint8_t Byte);
void Serial_SendArray(uint8_t *Array,uint16_t Length);
void Serial_SendString(char *String);
uint32_t Serial_Pow(uint32_t x,uint32_t y);
void Serial_SendNumber(uint32_t Number,uint8_t Length);
int fputc(int ch,FILE *f);
void Serial_Printf(char *format,...);
uint8_t Serial_GetRxFlag(void);
uint8_t Serial_GetRxData(void);#endif

n.总结

n.1USART实现步骤

  1. 开启时钟,把需要的USART和GPIO时钟开启
  2. GPIO初始化,把TX配置成复用输出,RX配置成输入
  3. 配置USART,直接使用一个结构体
  4. 如果只需要发送功能,直接开启USART时钟,初始化结束,

(如果需要配置接收的功能那么还需要配置中断,在开启USART之前,再加上ITConfig和NVIC的代码)

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

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

相关文章

【线性代数】1行列式

1. 行列式的概念 行列式的符号表示: 行列式的计算结果:一个数 计算模型1:二阶行列式 二阶行列式: 三阶行列式: n阶行列式: 🍎计算行列式 计算模型2:上三角形行列式 上三角形行列式特征:主对角线下皆为0。 上三角形行列式: 化上三角形通用方法:主对角线下,…

vite让每个scss文件自动导入某段内容

写了如下一个scss函数&#xff0c;希望自动导入到每个scss文件里面 vite.config.ts里面如下配置 import fs from fsconst filePath resolve(__dirname, ./src/assets/css/index.scss);const Minxcss fs.readFileSync(filePath, utf8); css: {preprocessorOptions: {scss: {…

【广州大学主办,发表有保障 | IEEE出版,稳定EI检索,往届见刊后快至1个月检索】第二届电气技术与自动化工程国际学术会议 (ETAE 2025)

第二届电气技术与自动化工程国际学术会议 (ETAE 2025) The 2nd International Conference on Electrical Technology and Automation Engineering 大会官网&#xff1a;http://www.icetae.com/【更多详情】 会议时间&#xff1a;2025年4月25-27日 会议地点&#xff1a…

Java面试第一山!《集合》!

一、引言 在 Java 编程的世界里&#xff0c;数据的存储和处理是非常重要的环节。Java 集合框架就像是一个功能强大的工具箱&#xff0c;为我们提供了各种各样的数据结构来高效地存储和操作数据。今天&#xff0c;跟随小编一起来深入了解 Java 集合框架&#xff0c;这不仅有助于…

APP端弱网模拟与网络测试:如何确保应用在各种网络环境下稳定运行

随着智能手机的普及&#xff0c;APP的网络性能成为用户体验的关键因素之一。尤其是在弱网环境下&#xff0c;应用的表现可能严重影响用户的满意度。因此&#xff0c;APP端的网络测试&#xff0c;尤其是弱网模拟&#xff0c;成为了提升产品质量和用户体验的重要环节。 当前APP网…

使用verilog 实现 cordic 算法 ----- 旋转模式

1-设计流程 ● 了解cordic 算法原理&#xff0c;公式&#xff0c;模式&#xff0c;伸缩因子&#xff0c;旋转方向等&#xff0c;推荐以下链接视频了解 cordic 算法。哔哩哔哩-cordic算法原理讲解 ● 用matlab 或者 c 实现一遍算法 ● 在FPGA中用 verilog 实现&#xff0c;注意…

【Linux】Socket编程—TCP

&#x1f525; 个人主页&#xff1a;大耳朵土土垚 &#x1f525; 所属专栏&#xff1a;Linux系统编程 这里将会不定期更新有关Linux的内容&#xff0c;欢迎大家点赞&#xff0c;收藏&#xff0c;评论&#x1f973;&#x1f973;&#x1f389;&#x1f389;&#x1f389; 文章目…

分布式技术

一、为什么需要分布式技术&#xff1f; 1. 科学技术的发展推动下 应用和系统架构的变迁&#xff1a;单机单一架构迈向多机分布式架构 2. 数据大爆炸&#xff0c;海量数据处理场景面临问题 二、分布式系统概述 三、分布式、集群 四、负载均衡、故障转移、伸缩性 负载均衡&a…

python后端调用Deep Seek API

python后端调用Deep Seek API 需要依次下载 ●Ollama ●Deepseek R1 LLM模型 ●嵌入模型nomic-embed-text / bge-m3 ●AnythingLLM 参考教程&#xff1a; Deepseek R1打造本地化RAG知识库:安装部署使用详细教程 手把手教你&#xff1a;deepseek R1基于 AnythingLLM API 调用本地…

优选驾考小程序

第2章 系统分析 2.1系统使用相关技术分析 2.1.1Java语言介绍 Java语言是一种分布式的简单的 开发语言&#xff0c;有很好的特征&#xff0c;在安全方面、性能方面等。非常适合在Internet环境中使用&#xff0c;也是目前企业级运用中最常用的一个编程语言&#xff0c;具有很大…

02、QLExpress从入门到放弃,相关API和文档

QLExpress从入门到放弃,相关API和文档 一、属性开关 public class ExpressRunner {private boolean isTrace;private boolean isShortCircuit;private boolean isPrecise; }/*** 是否需要高精度计算*/ private boolean isPrecise false;高精度计算在会计财务中非常重要&…

达梦:TPCC 压测

目录 造数1. 脚本启动2. 检查数据库信息3. 删除旧用户和表空间4. 创建新的表空间5. 创建用户和表6. 数据加载7. 创建索引8. 创建存储过程和序列9. 检查数据空间使用情况10. 启用表的快速访问池11. 数据加载完成总结 压测1. 脚本启动2. 检查数据表空间3. 设置表的快速池标志4. 检…

【ClickHouse】Ubuntu下离线安装ClickHouse数据库并使用DBeaver连接

目录 0. 安装前准备1 安装ClickHouse1.1 下载安装包1.2 离线安装1.3 配置密码1.4 启动ClickHouse服务 2 DBeaver连接配置2.1 下载ClickHouse驱动2.2 DBeaver配置2.2.1 配置主要参数2.2.2 配置驱动 2.3 常见问题处理2.3.1 修改远程登录配置2.3.2 更新驱动配置 0. 安装前准备 有…

CCF-GESP 等级考试 2024年9月认证C++二级真题解析

2024年9月真题 一、单选题&#xff08;每题2分&#xff0c;共30分&#xff09; 正确答案&#xff1a;A 考察知识点&#xff1a;计算机存储 解析&#xff1a;磁心存储元件是早期计算机中用于存储数据的部件&#xff0c;它和现代计算机中的内存功能类似&#xff0c;都是用于临时…

nuxt中引入element-ui组件控制台报错问题

在使用element-ui组件的外层加一层 <client-only placeholder"Loading..."><van-button type"primary">主要按钮</van-button> </client-only> 实际使用&#xff1a; <div class"tab"><client-only placehol…

京东 旋转验证码 分析

声明: 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;抓包内容、敏感网址、数据接口等均已做脱敏处理&#xff0c;严禁用于商业用途和非法用途&#xff0c;否则由此产生的一切后果均与作者无关&#xff01; 逆向分析 使用的第三方接码平台识别…

Git 查看修改记录 二

Git 查看修改记录 二 续接 Git 查看一个文件的修改记录 一 一、修改 A.txt 修改 A.txt number6执行命令 git add . git commit -a -m "修改 number6" # git commit -a -m "修改 number6" 执行 输出如下 # $ git commit -a -m "修改 number6"…

微软AutoGen高级功能——Magentic-One

介绍 大家好&#xff0c;博主又来给大家分享知识了&#xff0c;这次给大家分享的内容是微软AutoGen框架的高级功能Magentic-One。那么它是用来做什么的或它又是什么功能呢&#xff0c;我们直接进入正题。 Magentic-One Magnetic-One是一个通用型多智能体系统&#xff0c;用于…

Unity中自定义协程的简单实现

在 Unity 中&#xff0c;协程&#xff08;Coroutine&#xff09;是一种非常强大的工具&#xff0c;它允许我们在不阻塞主线程的情况下&#xff0c;将代码的执行分成多个步骤&#xff0c;在不同的帧中执行。 Unity中协程实现原理 迭代器与状态机&#xff1a;本质上是基于C#的迭…

数值积分:通过复合梯形法计算

在物理学和工程学中&#xff0c;很多问题都可以通过数值积分来求解&#xff0c;特别是当我们无法得到解析解时。数值积分是通过计算积分区间内离散点的函数值来近似积分的结果。在这篇博客中&#xff0c;我将讨论如何使用 复合梯形法 来进行数值积分&#xff0c;并以一个简单的…