32USART串口

news2025/1/16 18:00:04

目录

一.通信接口

 二.时序

三.USART简介

​编辑四.数据帧

五.起始位侦测和采样位置对齐 &波特率计算

六.相关函数

七.编码格式设置

(1) UTF-8编码(有的软件兼容性不好)​编辑

(2)GB2312编码

 八.代码实现

1.串口发送数据

2.串口数据发送和接收 

(1)函数解决

(2)中断解决


一.通信接口

 二.时序

三.USART简介

 S:同步,它只支持时钟输出,不支持时钟输入,更多是为了兼容别的协议或者特殊用途而设计的

并不支持两个USART之间的同步通信

硬件流控制(防止接收数据的设备处理慢而导致数据丢失):有A发送设备和B接收设备,在硬件电路上多出一根线,如果B没准备好就置高电平,准备好就置低电平,A通过线路接收到B反馈的准备信号,就只会在B准备好时发送数据

LIN:局域网的通信协议

USART1时APB2总线上的设备,USART2,USART3为APB1总线上的设备

USART框图

四.数据帧

五.起始位侦测和采样位置对齐 &波特率计算

六.相关函数

 

七.编码格式设置

(1) UTF-8编码(有的软件兼容性不好)

写入参数--no-multubyte-chars 

(2)GB2312编码

不用写入参数--no-multubyte-chars 

 八.代码实现

1.串口发送数据

Serial.c 

#include "stm32f10x.h"                  // Device header
#include <stdio.h>
#include <stdarg.h>
void Serial_Init(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,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);
	
	USART_InitStructure.USART_BaudRate=9600;//自动计算写入BRR寄存器
	USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None;//流控配置
	USART_InitStructure.USART_Mode=USART_Mode_Tx;//发送模式
	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_Cmd(USART1,ENABLE);
	
}
void Serial_SendByte(uint8_t Byte)
{
	USART_SendData(USART1,Byte);//将数据写入DR
	while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);
}
void Serial_SendArray(uint8_t *Array,uint16_t Size)
{
	uint8_t i;
	for(i=0;i<Size;i++)
	{
		
		Serial_SendByte(Array[i]);
	}
}
void Serial_SendString(char *My_String)
{
	uint8_t i;
	for(i=0;My_String[i] != '\0';i++)
	{
		Serial_SendByte(My_String[i]);
	}
	
}
uint32_t Serial_Pow(uint8_t Number,uint8_t Num)//次方,返回值定义要足够大
{
	uint32_t result=1;
	while(Num--)
	{
		result *= Number;
	}
	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');//加字符消除偏移
		
	}
}
int fputc(int ch,FILE *f)//fputc是printf的底层
{
	Serial_SendByte(ch);//将fputc函数重定向到串口
	return ch;
}

	/*
		封装可变参数的格式
	
	*/
void Serial_Printf(char *format,...)
{
	char String[100];
	va_list arg;
	va_start(arg,format);
	vsprintf(String,format,arg);//对于封装格式,要用vsprintf
	va_end(arg);//释放参数列表
	Serial_SendString(String);
	
	
}

 main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"  
#include "OLED.h" 
#include "Serial.h" 
int main(void)
{
	char String[100];//局部变量都要定义在开头
	uint8_t arr1[]={0x41,0x42,0x43,0x44,0x45};
	OLED_Init();
	Serial_Init();
	Serial_SendByte(0x41);
	Serial_SendArray(arr1,5);
	Serial_SendString("Akebi\r\n");//换行是“\r\n”
	Serial_SendNumber(12345,5);
	/*printf移植*/
	printf("Num=%d\r\n",666);//1.在Serial.c中将fputc函数重定向到串口
	 
	/*	
		sprintf可以指定打印位置,不涉及重定向
		这样可以确保每个串口都可以使用sprintf
	*/
	sprintf(String,"Num=%d\r\n",666);//可以将格式化字符输出到一个字符串里
	Serial_SendString(String);
	
	/*
		封装可变参数的格式
	
	*/
	Serial_Printf("Num=%d\r\n");
	
	
	Serial_Printf("你好,世界");
	
	
	
	while(1)
	{
		
		
	}	
}

2.串口数据发送和接收 

(1)函数解决

(2)中断解决

Serial.c 

#include "stm32f10x.h"                  // Device header
#include <stdio.h>
#include <stdarg.h>
uint8_t ReData,ReFlag;
void Serial_Init(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,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=9600;//自动计算写入BRR寄存器
	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_ITConfig(USART1,USART_IT_RXNE,ENABLE);
	
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	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);//将数据写入DR
	while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);
}
void Serial_SendArray(uint8_t *Array,uint16_t Size)
{
	uint8_t i;
	for(i=0;i<Size;i++)
	{
		
		Serial_SendByte(Array[i]);
	}
}
void Serial_SendString(char *My_String)
{
	uint8_t i;
	for(i=0;My_String[i] != '\0';i++)
	{
		Serial_SendByte(My_String[i]);
	}
	
}
uint32_t Serial_Pow(uint8_t Number,uint8_t Num)//次方,返回值定义要足够大
{
	uint32_t result=1;
	while(Num--)
	{
		result *= Number;
	}
	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');//加字符消除偏移
		
	}
}
int fputc(int ch,FILE *f)//fputc是printf的底层
{
	Serial_SendByte(ch);//将fputc函数重定向到串口
	return ch;
}

/*
		封装可变参数的格式
	
*/
void Serial_Printf(char *format,...)
{
	char String[100];
	va_list arg;
	va_start(arg,format);
	vsprintf(String,format,arg);//对于封装格式,要用vsprintf
	va_end(arg);//释放参数列表
	Serial_SendString(String);
}
uint32_t Get_Flag(void)
{
	if(ReFlag == 1)
	{
		ReFlag=0;
		return 1;
	}
	return 0;
	
}
uint32_t Get_Byte(void)
{
	
	return ReData;//这里应直接返回Data,不要进行Serial_RxFlag的判断,不然程序执行会跳过
	
	
}
void USART1_IRQHandler(void)
{
	if(USART_GetFlagStatus(USART1,USART_FLAG_RXNE) == SET)
	{
		ReData = USART_ReceiveData(USART1);
		ReFlag = 1;//其实这几行就是将数据进行一次转存,尽量单一化中断函数的内部功能
		USART_ClearFlag(USART1,USART_FLAG_RXNE);
	}
}

main.c 

#include "stm32f10x.h"                  // Device header
#include "Delay.h"  
#include "OLED.h" 
#include "Serial.h" 
uint16_t Data;
int main(void)
{
	OLED_Init();
	Serial_Init();
	OLED_ShowString(1,1,"RxData:");
	while(1)
	{
		if(Get_Flag() == 1)
		{
			Data = Get_Byte();
			Serial_SendByte(Data);//接收数据回传
			OLED_ShowHexNum(1,8,Data,2);
		}	
	}	
}
		
		

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

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

相关文章

【Nicn的刷题日常】之有序序列合并

1.题目描述 描述 输入两个升序排列的序列&#xff0c;将两个序列合并为一个有序序列并输出。 数据范围&#xff1a; 1≤&#xfffd;,&#xfffd;≤1000 1≤n,m≤1000 &#xff0c; 序列中的值满足 0≤&#xfffd;&#xfffd;&#xfffd;≤30000 0≤val≤30000 输入描述…

前端异步相关知识总结

目录 一、同步和异步简介 同步&#xff08;按顺序执行&#xff09; 异步&#xff08;不按顺序执行&#xff09; 异步出现的原因和需求 二、实现异步的方法 回调函数 Promise 生成器Generators/ yield async await 三、promise和 async await 区别 概念 两者的区别 …

07-Java桥接模式 ( Bridge Pattern )

Java桥接模式 摘要实现范例 桥接模式&#xff08;Bridge Pattern&#xff09;是用于把抽象化与实现化解耦&#xff0c;使得二者可以独立变化 桥接模式涉及到一个作为桥接的接口&#xff0c;使得实体类的功能独立于接口实现类&#xff0c;这两种类型的类可被结构化改变而互不影…

实践动物姿态估计,基于最新YOLOv8全系列【n/s/m/l/x】参数模型开发构建公共场景下行人人员姿态估计分析识别系统

姿态估计&#xff08;PoseEstimation&#xff09;在我们前面的相关项目中涉及到的并不多&#xff0c;CV数据场景下主要还是以目标检测、图像识别和分割居多&#xff0c;最近正好项目中在使用YOLO系列最新的模型开发项目&#xff0c;就想着抽时间基于YOLOv8也开发构建实现姿态估…

Open CASCADE学习|创建多段线与圆

使用Open CASCADE Technology (OCCT)库来创建和显示一些2D几何形状。 主要过程如下&#xff1a; 包含头文件&#xff1a;代码首先包含了一些必要的头文件&#xff0c;这些头文件提供了创建和显示几何形状所需的类和函数。 定义变量&#xff1a;在main函数中&#xff0c;定义…

如何查看端口映射?

端口映射是一种用于实现远程访问的技术。通过将外网端口与内网设备的特定端口关联起来&#xff0c;可以使外部网络用户能够通过互联网访问内部网络中的设备和服务。在网络中使用端口映射可以解决远程连接需求&#xff0c;使用户能够远程访问设备或服务&#xff0c;无论是在同一…

JAVA生产使用登录校验模式

背景 目前我们的服务在用户登录时&#xff0c;会先通过登录接口进行密码校验。一旦验证成功&#xff0c;后端会利用UUID生成一个独特的令牌&#xff08;token&#xff09;&#xff0c;并将其存储在Redis缓存中。同时&#xff0c;前端也会将该令牌保存在本地。在后续的接口请求…

常用对象和常用成员函数

常量对象与常量成员函数来防止修改对象&#xff0c;实现最低权限原则。 在Obj被定义为常量对象的情况下&#xff0c;下面这条语句是错误的。 错误的原因是常量对象一旦初始化后&#xff0c;其值就再也不能改变。因此&#xff0c;不能通过常量对象调用普通成员函数&#xff0c;因…

海外云手机的核心优势

随着5G时代的到来&#xff0c;云计算产业正处于高速发展的时期&#xff0c;为海外云手机的问世创造了一个可信任的背景。在资源有限且需求不断增加的时代&#xff0c;将硬件设备集中在云端&#xff0c;降低个人用户的硬件消耗&#xff0c;同时提升性能&#xff0c;这一点单单就…

得物自研API网关实践之路

一、业务背景 老网关使用 Spring Cloud Gateway &#xff08;下称SCG&#xff09;技术框架搭建&#xff0c;SCG基于webflux 编程范式&#xff0c;webflux是一种响应式编程理念&#xff0c;响应式编程对于提升系统吞吐率和性能有很大帮助; webflux 的底层构建在netty之上性能表…

广度优先搜索(BFS)

力扣刷题之旅&#xff1a;进阶篇&#xff08;二&#xff09; 继续我的力扣刷题之旅&#xff0c;我在进阶篇的第一部分中深入探索了BFS&#xff08;广度优先搜索&#xff09;算法&#xff0c;并感受到了它在图形搜索中的强大威力。现在&#xff0c;我进入了进阶篇的第二部分&am…

百卓Smart管理平台 uploadfile.php 文件上传漏洞复现(CVE-2024-0939)

0x01 产品简介 百卓Smart管理平台是北京百卓网络技术有限公司(以下简称百卓网络)的一款安全网关产品,是一家致力于构建下一代安全互联网的高科技企业。 0x02 漏洞概述 百卓Smart管理平台 uploadfile.php 接口存在任意文件上传漏洞。未经身份验证的攻击者可以利用此漏洞上传…

单片机无线发射的原理剖析

目录 一、EV1527编码格式 二、OOK&ASK的简单了解 三、433MHZ 四、单片机的地址ID 五、基于STC15W104单片机实现无线通信 无线发射主要运用到了三个知识点&#xff1a;EV1527格式&#xff1b;OOk&#xff1b;433MHZ。下面我们来分别阐述&#xff1a; EV1527是数据的编…

Stable Diffusion 模型下载:Samaritan 3d Cartoon SDXL(撒玛利亚人 3d 卡通 SDXL)

文章目录 模型介绍生成案例案例一案例二案例三案例四案例五案例六案例七案例八案例九案例十 下载地址 模型介绍 由“PromptSharingSamaritan”创作的撒玛利亚人 3d 卡通类型的大模型&#xff0c;该模型的基础模型为 SDXL 1.0。 条目内容类型大模型基础模型SDXL 1.0来源CIVITA…

IDEA创建Java类时自动添加注释(作者、年份、月份)

目录 IDEA创建Java类时自动添加注释&#xff08;作者、年份、月份&#xff09;如图&#xff1a; IDEA创建Java类时自动添加注释&#xff08;作者、年份、月份&#xff09; 简单记录下&#xff0c;IDEA创建Java类时自动添加注释&#xff08;作者、年份、月份&#xff09;&#…

@PostMapping/ @GetMapping等请求格式

目录 1.只传一个参数的 第一种 第二种 第三种:表单 2.传整个对象的 2.1修改实体类就是传整个对象过来 2.2新增实体类就是传整个对象过来新增 1.只传一个参数的 第一种 PostMapping("/add/{newsId}")public Result addOne(PathVariable Integer newsId) {}pos…

Spring + Tomcat项目中nacos配置中文乱码问题解决

实际工作的时候碰到了nacos中文乱码的问题&#xff0c;一顿排查最终还是调源码解决了。下面为具体的源码流程&#xff0c;有碰到的可以参考下。 对于nacos配置来说&#xff0c;初始主要源码就在NacosConfigService类中。里面有初始化获取配置content以及设置对应监听器的操作。…

windows中的apache改成手动启动的操作步骤

使用cmd解决安装之后开机自启的问题 services.msc 0. 这个命令是打开本地服务找到apache的服务名称 2 .通过服务名称去查看服务的状态 sc query apacheapache3.附加上关掉和启动的命令&#xff08;换成是你的服务名称&#xff09; 关掉命令 sc stop apacheapache启动命令 …

金融行业专题|证券超融合架构转型与场景探索合集(2023版)

更新内容 更新 SmartX 超融合在证券行业的覆盖范围、部署规模与应用场景。新增操作系统信创转型、Nutanix 国产化替代、网络与安全等场景实践。更多超融合金融核心生产业务场景实践&#xff0c;欢迎阅读文末电子书。 在金融行业如火如荼的数字化转型大潮中&#xff0c;传统架…

使用Python语言生成区块链地址

# 单次运行 import binascii import sha3 from ecdsa import SigningKey, SECP256k1priv SigningKey.generate(curveSECP256k1) # 生成私钥 pub priv.get_verifying_key() # 生成公钥keccak sha3.keccak_256() keccak.update(pub.to_string()) # keccak_256哈希运算 addr…