USART发送单字节数据原理及程序实现

news2024/11/15 10:17:04

硬件接线:

显示屏的SCA接在B11,SCL接在B10,串口的RX连接A9,TX连接A10。

新建Serial.c和Serial.h文件

        在Serial.c文件中,实现初始化函数,等需要的函数,首先对串口进行初始化,只需要发送那么就初始化A9引脚。

初始化步骤:

  • 初始化A9引脚,设置为复用推挽输出,也就是让内部硬件控制引脚
  • 波特率:9600
  • 不使用硬件流控制,也就是不使用RTS,CTS等
  • 串口模式为TX(Transform)表示发送
  • 无校验位,可选择奇校验,偶校验等
  • 1位停止位,可选择0.5 1 1.5 2这几个
  • 8字长,不需要校验选8位,需要选9位

初始化代码:

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);
	
	USART_InitTypeDef USART_InitStructure;
	USART_InitStructure.USART_BaudRate = 9600;//波特率
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//硬件流控制(不使用,CTS,CTS&RTS)
	USART_InitStructure.USART_Mode = USART_Mode_Tx;//串口模式 可以使用(或)|符号实现Tx和Rx同时设置
	USART_InitStructure.USART_Parity = USART_Parity_No;//校验位,无需校验
	USART_InitStructure.USART_StopBits = USART_StopBits_1;//停止位,选择1位
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长
	USART_Init(USART1, &USART_InitStructure);
	
	USART_Cmd(USART1, ENABLE);//开启USART
}

定义发送函数:

void Serial_SendByte(uint8_t Byte) {
	USART_SendData(USART1, Byte);//发送数据
	while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET) {//等待发送寄存器空,
		//TXE就是发送寄存器空的标志位,不需要手动清零,下一次发送数据时候会自动清零
	}
}

发送数据代码原理: 

内部库函数:

        右键SendData函数跳转定义,在内部SendData函数中,会将形参Byte与0x01FF进行&操作,将低9位保留,高9位置1,接着将数据写入DR寄存器中,最终数据会通向TDR(发送数据寄存器),TDR再传递给发送移位寄存器,最后一位一位地把数据从TX引脚移出去。

        如图发送数据执行完后,还需要等待TDR的数据转移到移位寄存器了才可以继续执行程序。那么就需要这个While循环等待标志位,等待发送寄存器空标志位TXE == 1(TXEmpty)。根据手册描述在while函数中,不需要手动将标志位置0,在程序下一次执行SendData时就会自动置0。

发送数组函数:

void Serial_SendArray(uint8_t *Array, uint16_t Length){
	uint16_t i;
	for(int i = 0; i < Length; i++) {
		Serial_SendByte(Array[i]);
	}
}

        发送数组函数很简单,就是循环遍历数组依次发送数组的每一位就好了。

发送字符串函数:

void Serial_SendString(char *String) {//字符串自带结束标志位
	uint8_t i;
	for(int i = 0; String[i] != '\0'; i++) {
		Serial_SendByte(String[i]);
	}
}

        发送字符串时需要注意,字符串实际上就是字符数组,最后一位为结束标志位为'0'。字符串名就是字符数组的首地址,因此可以这样编写代码。与发送数字数组同理,遍历每一个数组元素进行发送就可以了。

发送数字函数:

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 Length) {
	uint8_t i;
	for(int i = 0; i < Length; i++){
		Serial_SendByte((Number / Serial_Pow(10, Length - i - 1)) % 10 + '0');
	}
}

        发送数字时,还是一个字符一个字符进行发送,因此就要把数字的每一位取出来再依次发送,那么对应第p个数字就是原数除以10^(p - 1)再对10取余,例如数字123,除以10^2再对10取余就是3,除以10^1再对10取余就是2,除以10^0再对10取余就是1,因此可以编写出上图程序实现这个操作。

重写Printf函数

#include<stdio.h>
int fputc(int ch, FILE* f){
	Serial_SendByte(ch);//重定向到串口,使得Printf打印到串口
	return ch;
}

        fputc函数就是printf函数的底层,printf函数在打印的时候,就是不断调用fputc函数一个个打印的,将ch传给Serial_SendByte函数,接着这个函数又调用USART_SendData(USART1, Byte)这个函数,就相当于将字符打印到了USART1也就是串口1中。

使用sprintf函数

        上一个重定向的方法有个缺点就是只能指定一个串口进行重定向,串口2需要打印时,就不能使用printf函数了。如果多个串口都需要printf怎么办,这时就可以用sprintf,它可以把格式化字符输出到一个字符串里。

#main函数中	
    char String[100];
	sprintf(String, "Num = %d\r\n", 666);
	Serial_SendString(String);

        注意这个代码在main函数中实现,通过sprintf函数将字符格式化到String字符串中,再通过串口打印出去,如果需要多个串口发送就可以定义另一个串口的SendString函数,例如SendString2函数,再将字符串发送出去。

封装sprintf函数(最常用)

        这个就是将上一个代码封装起来使用,由于sprintf函数的参数比较特殊,是可变参数,因此函数参数传递需要特殊化

添加头文件:#include<stdarg.h>

#include<stdarg.h>
void Serial_Printf(char* format,...){//三个点用来接收后面可变参数列表
	char String[100];
	va_list arg;
	va_start(arg, format);//从format位置开始接收参数表,放在arg里面
	vsprintf(String, format, arg);
	va_end(arg);
	Serial_SendString(String);
}

        如果需要另一个串口的输出,那么将另一个串口编写一个发送字符串函数。例如Serial_SendString2(String), 再写一个Serial_Printf2()函数,这样就能实现多串口输出了。

整体代码:

main:

#include "stm32f10x.h"                  // Device header
#include "DELAY.h"
#include "OLED.h"
#include "Serial.h"
uint8_t KeyNum;
int main() {
	OLED_Init();
	Serial_Init();
	Serial_SendByte(0x41);
	uint8_t MyArray[10] = {0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x50, 0x51};
	Serial_SendArray(MyArray, 10);


	OLED_ShowString(1,1,"Send Data");
	OLED_ShowString(2,1,"Hello World");
	Serial_SendString("HelloWorld\r\n");//编译器会自动补上标志位,因此字符串的存储空间比字符个数多1个

	Serial_SendNumber(123456, 6);
	printf("Num=%d\r\n", 666);
	//使用sprintf让其他的串口也能使用,sprintf可以把格式化字符输出到一个字符串里
	char String[100];
	sprintf(String, "Num = %d\r\n", 666);
	Serial_SendString(String);
	Serial_Printf("数字 = %d\r\n", 666);
	while(1){
		
	}
}

Serial.c

#include "stm32f10x.h"                  // Device header
#include <stdio.h>
#include <stdarg.h>
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);
	
	USART_InitTypeDef USART_InitStructure;
	USART_InitStructure.USART_BaudRate = 9600;//波特率
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//硬件流控制(不使用,CTS,CTS&RTS)
	USART_InitStructure.USART_Mode = USART_Mode_Tx;//串口模式 可以使用(或)|符号实现Tx和Rx同时设置
	USART_InitStructure.USART_Parity = USART_Parity_No;//校验位,无需校验
	USART_InitStructure.USART_StopBits = USART_StopBits_1;//停止位,选择1位
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长
	USART_Init(USART1, &USART_InitStructure);
	
	USART_Cmd(USART1, ENABLE);//开启USART
}
void Serial_SendByte(uint8_t Byte) {
	USART_SendData(USART1, Byte);//发送数据
	while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET) {//等待发送寄存器空,
		//TXE就是发送寄存器空的标志位,不需要手动清零,下一次发送数据时候会自动清零
	}
}
void Serial_SendArray(uint8_t *Array, uint16_t Length){
	uint16_t i;
	for(int i = 0; i < Length; i++) {
		Serial_SendByte(Array[i]);
	}
}
void Serial_SendString(char *String) {//字符串自带结束标志位
	uint8_t i;
	for(int 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 Length) {
	uint8_t i;
	for(int i = 0; i < Length; i++){
		Serial_SendByte((Number / Serial_Pow(10, Length - i - 1)) % 10 + '0');
	}
}
int fputc(int ch, FILE* f){
	Serial_SendByte(ch);//重定向到串口,使得Printf打印到串口
	return ch;
}
//使用sprintf让其他的串口也能使用,sprintf可以把格式化字符输出到一个字符串里
void Serial_Printf(char* format,...){//三个点用来接收后面可变参数列表
	char String[100];
	va_list arg;
	va_start(arg, format);//从format位置开始接收参数表,放在arg里面
	vsprintf(String, format, arg);
	va_end(arg);
	Serial_SendString(String);
}

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 *Array, uint16_t Length);
void Serial_SendString(char *String);
void Serial_SendNumber(uint32_t Number, uint8_t Length);
void Serial_Printf(char* format,...);


#endif

程序现象:

        烧录好程序,打开串口助手,选择串口和波特率,点击打开串口按钮,按下STM32的复位按键可以看见串口数据。

文件下载:

程序包:程序打包下载

串口助手:串口助手下载

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

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

相关文章

全国地质灾害点shp崩塌滑坡泥石流空间分布地质灾害详查等数据集

地质灾害是指在自然或者人为因素的作用下形成的&#xff0c;对人类生命财产造成的损失、对环境造成破坏的地质作用或地质现象。地质灾害在时间和空间上的分布变化规律&#xff0c;既受制于自然环境&#xff0c;又与人类活动有关&#xff0c;往往是人类与自然界相互作用的结果。…

easyx查找算法可视化--顺序查找/二分查找/分块查找

&#x1f482; 个人主页:pp不会算法^ v ^ &#x1f91f; 版权: 本文由【pp不会算法v】原创、在CSDN首发、需要转载请联系博主 &#x1f4ac; 如果文章对你有帮助、欢迎关注、点赞、收藏(一键三连)和订阅专栏哦 文章目录 概述演示源码获取 概述 #顺序存储的顺序查找 √ #链式存…

验证码/数组元素的复制.java

1&#xff0c;验证码 题目&#xff1a;定义方法实现随机产生一个5位的验证码&#xff0c;前面四位是大写或小写的英文字母&#xff0c;最后一位是数字 分析&#xff1a;定义一个包含所有大小写字母的数组&#xff0c;然后对数组随机抽取4个索引&#xff0c;将索引对应的字符拼…

Qt C++ | QTimer经验总结

QTimer Class QTimer类提供重复计时器和单次计时器 头文件: #include <QTimer> qmake: QT += core 继承自: QObject 定时器信号 void timeout() 公共函数 Qt::TimerType 枚举定义了 Qt 中不同类型的定时器。它包含以下值: **Qt::PreciseTimer:**高精度定时器,用…

WPF —— DockPanel、ProgressBar 控件详解

ProgressBar 控件详解 1Progress bar简介 ProgressBar&#xff1a;进度条控件。 WPF带有一个方便的控件用于显示进度&#xff0c;称ProgressBar。它的工作原理就是设置最小值和最大值然后通过递增一个值&#xff0c;这样就可以直观的显示当前进度情况。 2 Progress bar常用的…

腾讯VS网易:一场不见终局的游戏未来之战

国内游戏霸主腾讯最近赚足了眼球。 总体上看&#xff0c;腾讯手握“游戏社交”两大王牌&#xff0c;最近发布的财报十分亮眼&#xff0c;其2023年总营收和净利润分别同比增长10%和36%&#xff0c;展现了互联网巨头的强劲活力。 然而巨头亦有焦虑&#xff0c;增值服务营收同比…

AMEYA360代理 | 江苏长晶科技FST2.0高性能 IGBT产品介绍

江苏长晶科技股份有限公司是一家专业从事半导体产品研发、生产和销售的企业。自2019年起&#xff0c;连续4年被中国半导体行业协会评为 “功率器件十强企业”。2021年开始自主研发有着“工业CPU”之称的IGBT&#xff0c;截至2023年Q3在家电/工业/新能源等行业实现8款产品市场应…

Windows python多版本共享方案

1、先安装好python3.11 2、安装好python3.7 这时默认版本是python3.7&#xff0c; A、如果要切换回python3.11则修改环境变量即可 B、 如果想使用3.7&#xff0c;找到python3.7的安装路径 如果想使用3.7 C:\Users\用户\AppData\Local\Programs\Python\Python37 复制python…

Lua热更新(xlua)

发现错误时检查是否:冒号调用 只需要导入asset文件夹下的Plugins和Xlua这两个文件即可,别的不用导入 生成代码 和清空代码 C#调用lua using Xlua; 需要引入命名空间 解析器里面执行lua语法 lua解析器 LuaEnv 单引号是为了避免引号冲突 第二个参数是报错时显示什么提示…

点赋科技教你无人自助咖啡机选什么品牌?不选贵的,只选对的!

随着社会的发展和人们生活水平的提高&#xff0c;咖啡文化在全球范围内越来越受到重视。无人自助咖啡机作为一种便捷、高效的咖啡供应方式&#xff0c;逐渐受到了市场的青睐。在众多品牌中&#xff0c;我们为何应选择智能咖啡机呢&#xff1f;下面就让D咖带大家来一探究竟。 D咖…

Spring Cloud - Openfeign 实现原理分析

OpenFeign简介 OpenFeign 是一个声明式 RESTful 网络请求客户端。OpenFeign 会根据带有注解的函数信息构建出网络请求的模板,在发送网络请求之前,OpenFeign 会将函数的参数值设置到这些请求模板中。虽然 OpenFeign 只能支持基于文本的网络请求,但是它可以极大简化网络请求的…

腾讯云服务器多少钱一年?最新价格4核8G服务器646元15个月

2024年腾讯云4核8G服务器租用优惠价格&#xff1a;轻量应用服务器4核8G12M带宽646元15个月&#xff0c;CVM云服务器S5实例优惠价格1437.24元买一年送3个月&#xff0c;腾讯云4核8G服务器活动页面 txybk.com/go/txy 活动链接打开如下图&#xff1a; 腾讯云4核8G服务器优惠价格 轻…

【C语言进阶篇】编译和链接

【C语言进阶篇】编译和链接 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;C语言&#x1f353; &#x1f33c;文章目录&#x1f33c; 编译环境与运行环境 1. 翻译环境 2. 编译环境&#xff1a;预编译&#xff08;预处理&#xff09;编…

2024年腾讯云4核8G配置服务器一年多少钱?价格超便宜

2024年腾讯云4核8G服务器租用优惠价格&#xff1a;轻量应用服务器4核8G12M带宽646元15个月&#xff0c;CVM云服务器S5实例优惠价格1437.24元买一年送3个月&#xff0c;腾讯云4核8G服务器活动页面 txybk.com/go/txy 活动链接打开如下图&#xff1a; 腾讯云4核8G服务器优惠价格 轻…

代码随想录刷题随记7-字符串1

代码随想录刷题随记7-字符串1 文章目录 代码随想录刷题随记7-字符串1344.反转字符串541. 反转字符串II替换数字151.翻转字符串里的单词右旋字符串 344.反转字符串 leetcode链接 主要的难点在于使用 O(1) 的额外空间解决这一问题 反转字符串依然是使用双指针的方法 swap可以有两…

【算法刷题day4】Leetcode:24. 两两交换链表中的节点、19.删除链表的倒数第N个节点、面试题 02.07. 链表相交、142.环形链表II

24. 两两交换链表中的节点 文档链接&#xff1a;[代码随想录] 题目链接&#xff1a;24. 两两交换链表中的节点 状态&#xff1a;ok 题目&#xff1a; 给定一个链表&#xff0c;两两交换其中相邻的节点&#xff0c;并返回交换后的链表。 你不能只是单纯的改变节点内部的值&#…

老壁灯带你入门动态规划

1. 什么是动态规划 动态规划(dynamic programming)是运筹学的一个分支&#xff0c;是求解决策过程(decision process)最优化的数学方法。 从字面意义上来理解&#xff0c;就是走一步看一步&#xff0c;边解决问题&#xff0c;边对问题进行整体规划。 其实&#xff0c;动态规…

STM32学习笔记(7_2)- ADC模数转换器代码

无人问津也好&#xff0c;技不如人也罢&#xff0c;都应静下心来&#xff0c;去做该做的事。 最近在学STM32&#xff0c;所以也开贴记录一下主要内容&#xff0c;省的过目即忘。视频教程为江科大&#xff08;改名江协科技&#xff09;&#xff0c;网站jiangxiekeji.com 本期开…

MyBatis入门01

MyBatis入门01 文章目录 MyBatis入门01前言一、搭建环境1.新建一个普通的maven项目2.删除src目录3.导入maven依赖&#xff1a;mysql驱动&#xff08;操作jdbc&#xff09;&#xff0c;juint&#xff0c;mybatis注意&#xff1a;要假如builder标签&#xff0c;预防配置文件不可导…

3.27作业

1、完成下面类 #include <iostream> #include <cstring> using namespace std;class myString { private:char *str; //记录c风格的字符串int size; //记录字符串的实际长度 public://无参构造myString():size(10){str new char[size]; …