❤️ 专栏简介:本专栏记录了从零学习单片机的过程,其中包括51单片机和STM32单片机两部分;建议先学习51单片机,其是STM32等高级单片机的基础;这样再学习STM32时才能融会贯通。
☀️ 专栏适用人群 :适用于想要从零基础开始学习入门单片机,且有一定C语言基础的的童鞋。
专栏目标:实现从零基础入门51单片机和STM32单片机,力求在玩好单片机的同时,能够了解一些计算机的基本概念,了解电路及其元器件的基本理论等。⭐️ 专栏主要内容: 主要学习STM32单片机的功能、各个模块、单片机的外设、驱动等,最终玩好单片机和单片机的外设,全程手敲代码,实现我们所要实现的功能。
专栏说明 :如果文章知识点有错误的地方,欢迎大家随时在文章下面评论,我会第一时间改正。让我们一起学习,一起进步。
专栏主页:http://t.csdn.cn/HCD8v
本学习过程参考:https://space.bilibili.com/383400717
STM3单片机安装软件、各种资料以及源码的路径:
链接:https://pan.baidu.com/s/1snD0uuTfMhchFqOMWvAiHA?pwd=asdf#list/path=%2F
提取码:asdf
链接里压缩包的解压密码:32
本大节主要学习USART的相关知识,包含九小节:
第一小节主要学习USART串口协议的理论基础知识
第二小节主要学习USART串口外设的理论基础知识
第三小节是对第一、二小节的内容写一个串口发送程序进行练习
第四小节是对第一、二小节的内容再写一个串口发送+接收的程序进行练习
第五小节主要学习USART串口数据包的理论基础知识
第六小节是对第五小节的内容写一个串口收发HEX数据包程序进行练习
第七小节是对第五小节的内容再写一个串口收发文本数据包程序进行练习
第八小节主要了解MyMcu串口下载的知识
第八小节主要了解STLINK Utility串口下载的知识
最终附上所有的源代码;
本小节主要是对前面学习的串口理论知识进行实战练习:写一个串口发送的程序进行练习。
写一个串口模块,通过串口通信,把一些数据发到电脑上的串口助手来显示,
把STM32的串口引脚,接到电脑上,如下图所示:
之后电脑端打开串口助手软件,按如下图进行配置,并打开串口
按一下复位键,程序刚开始启动时会向串口发送一串数据,如下图所示:
此时切换成文本模式,再次复位,这时,软件就会对刚才的数据进行文本映射,找到每个数据对应的字符,以字符串的形式显示出来;如下图所示:
硬件电路如下所示:
然后打开电脑设备管理器,确认串口端口正常链接没问题:
STM32入门教程资料\程序源码\STM32Project\9-1 串口发送\User
STM32入门教程资料\程序源码\STM32Project\9-1 串口发送\Hardware
STM32入门教程资料\程序源码\STM32Project\9-1 串口发送运\System
整个代码的流程参考以上结构图:
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "OLED.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("\r\nNum1=");
Serial_SendNumber(111, 3);
printf("\r\nNum2=%d", 222);
char String[100];
sprintf(String, "\r\nNum3=%d", 333);
Serial_SendString(String);
Serial_Printf("\r\nNum4=%d", 444);
Serial_Printf("\r\n");
while (1)
{
}
}
Serial.c
:
#include "stm32f10x.h" // Device header
#include
#include
void Serial_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);//开启USART1的时钟;USART1是APB2的外设
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//开启GPIO时钟
/*初始化引脚,把PA9配置为复用推挽输出,供USART1的TX使用*/
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//引脚模式选择复用推挽输出
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;//因为本节只需要发送,所以只初始化Pin_9即可
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIO引脚;
USART_InitTypeDef USART_InitStructure;
USART_InitStructure.USART_BaudRate = 9600;//波特率设置为9600
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;//停止位选择1
USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长选择8位
USART_Init(USART1, &USART_InitStructure);//初始化USART1
USART_Cmd(USART1, ENABLE);
}
/*发送字节函数*/
void Serial_SendByte(uint8_t Byte)
{
USART_SendData(USART1, Byte);//发送数据,就是写DR寄存器
while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);//等待TXE置1;不需要手动清零,下一次发送数据时该标志位会自动清零
}
/*发送数组函数*/
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]);
}
}
/*计算x的y次方*/
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 (i = 0; i < Length; i ++)
{
Serial_SendByte(Number / Serial_Pow(10, Length - i - 1) % 10 + '0');
}
}
/*
重写fputc函数
将fputc函数重定向到串口
重写他的原因是后面我们用printf函数,printf函数底层是不断调用fputc函数的
*/
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);//从format位置开始接收参数表
vsprintf(String, format, arg);
va_end(arg);//释放参数表
Serial_SendString(String);
}