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

硬件接线:

显示屏的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.mzph.cn/news/774709.shtml

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

相关文章

【嵌入式DIY实例】-火焰报警系统

DIY火焰报警系统 文章目录 DIY火焰报警系统1、硬件准备2、硬件接线3、代码实现火灾报警器有时也称为烟雾报警器,是一种检测火灾、烟雾和热量并发出警告信号的设备。 它们可以固定在天花板或墙上,让您有更多时间离开那个地方。 如果您的房屋没有安装火灾报警器,您可能不会意识…

速看!MC-CCPIT第二十二届中国国际冶金工业展览会

METALLURGY CHINA 2024 MC-CCPIT第二十二届中国国际冶金工业展览会 ——冶金装备品牌展示区 主办单位&#xff1a;中国钢铁工业协会 中国国际贸易促进委员会冶金行业分会 承办单位&#xff1a;冶金工业国际交流合作中心 地 点&#xff1a;上海新国际博览中心 时 间&am…

第5章 数据建模和设计

思维导图 5.1 引言 最常见的6种模式&#xff1a;关系模式、多维模式、面向对象模式、 事实模式、时间序列模式和NoSQL模式 每种模式分为三层模型&#xff1a;概念模型、逻辑模型和物理模型 每种模型都包含一系列组件&#xff1a;如实体、关系、事实、键和属性。 5.1.1 业务驱…

修改Jupyter Notebook的默认路径,以及在PowerShell中自定义其启动路径

修改Jupyter Notebook的默认路径&#xff0c;以及在PowerShell中自定义其启动路径 设置 Jupyter Notebook 配置文件&#xff0c;修改默认路径要在PowerShell中设置自定义的启动脚本&#xff0c;以确保Jupyter Notebook能够自动定位到当前路径设置后的效果 在使用Jupyter Notebo…

设置asp.net core WebApi函数输入和返回类型中的属性名称开头大小写格式

以下列类型定义为例创建简单的ASP.NET Core的WebApi函数&#xff0c;此时输入参数和返回结果的属性名称开头默认为小写&#xff0c;如下图所示。 public class UserInfo { public string UserName { get; set; }public string UserSex { get; set; }public string UserP…

机器翻译.

0机器翻译 - 蓝桥云课 (lanqiao.cn) 题目描述 小晨的电脑上安装了一个机器翻译软件&#xff0c;他经常用这个软件来翻译英语文章。 这个翻译软件的原理很简单&#xff0c;它只是从头到尾&#xff0c;依次将每个英文单词用对应的中文含义来替换。对于每个英文单词&#xff0c;软…

如何写出防御性代码(屎山代码)

一、想说的 这两年各大公司纷纷实现广进计划&#xff0c;开猿节流&#xff0c;吾辈程序员家人们深受其害。大厂程序员被裁&#xff0c;向下挤压中小厂老铁们的岗位&#xff0c;内卷的不行。 为了咱们保住咱们的饭碗&#xff0c;形成护城河&#xff0c;形成核心竞争力&#xff…

STM32的SPI通信介绍

SPI简介 SPI:串行外设接口,与IIC一样都是通用数据总线。四根通信线&#xff1a;SCK&#xff0c;MOSI&#xff08;DO&#xff09;&#xff0c;MISO&#xff08;DI&#xff09;&#xff0c;SS。同步&#xff08;共用一根时钟线&#xff09;&#xff0c;全双工&#xff08;数据发…

阿里云服务器多少钱一个月?低至5元1个月

阿里云服务器一个月多少钱&#xff1f;最便宜5元1个月。阿里云轻量应用服务器2核2G3M配置61元一年&#xff0c;折合5元一个月&#xff0c;2核4G服务器30元3个月&#xff0c;2核2G3M带宽服务器99元12个月&#xff0c;轻量应用服务器2核4G4M带宽165元12个月&#xff0c;4核16G服务…

搜维尔科技:【应急演练】【工业仿真】救援模拟演练可视化仿真项目实施

安全救援综合演练系统是一套面向公共安全事故、预案管理、应急救援模拟演练的虚拟仿真解决方案&#xff0c;它为警察、消防以及专门的应急救援保障部门提供一个综合的应急救援培训和仿真演练平台。平台主要通过设计不同的事故模型和特定的灾难场景&#xff0c;定制不同的应急救…

上班几周了,

过年回来后&#xff0c;时间变得飞快&#xff0c;很多事情都是马上要去干&#xff0c;而且又是很着急的事&#xff0c;呵呵&#xff0c;真的要干趴了 然后——经历了第一次年后的周末连续加班出版本保量产&#xff0c;经历了加班到凌晨3点调试问题&#xff0c;经历我们在疯狂的…

【TypeScript系列】实用工具类型

实用工具类型 TypeScript 提供一些工具类型来帮助常见的类型转换。这些类型是全局可见的。 目录 Partial<T>&#xff0c;TypeScript 2.1Readonly<Type>&#xff0c;TypeScript 2.1Record<Keys, Type>&#xff0c;TypeScript 2.1Pick<Type, Keys>&am…

【字符串算法题记录】反转字符串中的单词(leetcode),右旋字符串(kama)——双指针以及反转的奇思妙用

反转字符串中的单词 题目链接 思考 这题的思路顺序是&#xff1a;移除多余空格&#xff08;双指针法&#xff09;——》反转整个字符串&#xff09;——》反转字符串中每个单词。 移除多余空格&#xff08;双指针法&#xff09; 因为字符串开头也可能有多个字符&#xff0…

深入理解React的setState机制

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

修复ubuntu引导

一、制作ubuntu启动U盘 进入启动盘后&#xff0c;点击Try ubuntu&#xff0c;进入U盘的ubuntu系统。 二、配置和添加源 sudo add-apt-repository ppa:yannubuntu/boot-repair && sudo apt-get update三、运行 Boot Repair重新制作引导 sudo boot-repair注意&#x…

关于HashSet的五个问题

1.HashSet集合的底层数据结构是什么样的? HashSet 集合的底层数据结构是哈希表&#xff0c;它是由一个数组和链表&#xff08;或红黑树&#xff0c;具体取决于 JDK 版本&#xff09;组成的数据结构。 数组&#xff1a;哈希表的主要部分是一个数组&#xff0c;它的每个位置称为…

【java】线程thread

线程池 线程池状态 1.RUNNING 表示线程池正常运行&#xff0c;既能接受新任务&#xff0c;也会正常处理队列中的任务 2. SHUTDOWN 当调用线程池的shutdown(&#xff09;方法时&#xff0c;线程池就进入SHUTDOWN状态&#xff0c;表示线程池处于正在关闭状态&#xff0c;此状…

STEP 格式三维模型读取

STEP是常用的三维模型存储格式&#xff0c;使用Express语言描述几何图形&#xff0c;文件存储方式为BRep&#xff0c;分为STEP203和STEP214&#xff0c;后者多了颜色信息&#xff0c;opencascade中提供了相应算法读取STEP文件。 #include <STEPControl_Reader.hxx>TopoD…

冒泡排序(六大排序)

冒泡排序 冒泡排序的特性总结&#xff1a; 1. 冒泡排序是一种非常容易理解的排序 2. 时间复杂度&#xff1a;O(N^2) 3. 空间复杂度&#xff1a;O(1) 4. 稳定性&#xff1a;稳定 动图分析&#xff1a; 代码实现&#xff1a; Swap(int*p1,int*p2) {int tmp *p1;*p1*p2…

利用Cas中service重定向钓鱼网站问题

前言 今天运维反馈现网有个系统http://aaa-test-env.com/cas/logout?servicehttp://www.evil.com/存在重定向钓鱼网站的安全漏洞。熟悉Cas实现单点登录的都知道&#xff0c;通过service参数&#xff0c;在Cas认证中心登录认证之后重定向到service对应的业务系统。但是Cas本身…