【單片機與嵌入式】stm32串口通信入門

一、串口通信/協議

(一)串口通信簡介

串口通信是一種通過串行傳輸方式在電子設備之間進行數據交換的通信方式。它通常涉及兩條線(一條用于發送數據,一條用于接收數據),適用于各種設備,從微控制器到計算機等。串口通信的關鍵特點包括:

1.串行傳輸:數據位按照順序一個接一個地傳輸,與并行傳輸相比,節省了引腳和線纜。
2.異步或同步傳輸:串口通信可以是異步的(通過單獨的時鐘信號進行數據同步)或同步的(通過時鐘信號直接同步數據傳輸)。
3.通信速率:串口通信的速率通常以波特率(bps,每秒傳輸的位數)來衡量,典型的速率有9600、19200、38400、115200等。
4.簡單性和廣泛應用:串口通信協議相對簡單,因此在許多嵌入式系統、傳感器和計算機外圍設備中廣泛使用。
5.標準和協議:常見的串口標準包括RS-232、RS-485等,每種標準有特定的電氣特性和通信協議。
6.通信模式:通信可以是單向的(如僅發送或僅接收),也可以是雙向的(可以發送和接收數據)。

(二)通信模式(數據傳輸方式)

1、串行通信與并行通信

串行通信和并行通信是兩種數據傳輸方式,它們有著明顯的區別:

串行通信:

? 定義:串行通信是一種逐位地傳輸數據的方式,數據位按照順序一個接一個地傳輸
??傳輸方式:使用單條線路傳輸數據,一般包括發送線(TX)和接收線(RX)。
? 優點:相比并行通信,串行通信可以減少線路和引腳數量,降低成本和復雜度。
? 典型應用:常見于長距離通信和嵌入式系統,如串口通信(RS-232、RS-485)、USB等。


并行通信:

? 定義:并行通信是同時傳輸多個數據位的方式,每個數據位使用一個獨立的信號線路
? 傳輸方式:通常使用多個并列的線路,每個線路傳輸一個數據位,同時進行。
? 優點:傳輸速度快,適合于需要高速數據傳輸的應用。
? 缺點:需要更多的線路和引腳,因此成本較高且布線復雜。
? 典型應用:在計算機內部的數據總線(如PCI總線)、內存訪問等地方常見并行通信。

可以將兩種數據傳輸方式理解為“串聯”與“并聯”。

2、單工、半雙工、全雙工通信

單工、半雙工和全雙工通信是描述數據在通信中傳輸方向和能力的術語.單工通信適合于單向傳輸、無需反饋的應用;半雙工通信適合于雙向通信但不能同時進行;全雙工通信適合需要同時雙向傳輸數據的場合。

單工通信:

??定義:單工通信是指數據只能在一個方向上傳輸的通信方式。發送方只能發送數據,接收方只能接收數據,不能同時發送和接收。
? 特點:通信是單向的,類似于單行道,信息只能在一個方向上流動。
? 應用場景:常見于廣播和一些簡單的傳感器網絡中,如無線電廣播、鍵盤到計算機的數據傳輸等。

半雙工通信:

? 定義:半雙工通信允許數據在兩個方向上傳輸,但不能同時進行。在某一時刻,通信設備要么發送數據,要么接收數據。
? 特點:通信設備在發送數據時不能同時接收,反之亦然。類似于單行道,但是可以在兩個方向上切換流量。
? 應用場景:廣泛應用于無線電對講機、對講電話以及以太網的半雙工模式。


全雙工通信:

??定義:全雙工通信允許數據在兩個方向上同時進行傳輸,發送和接收可以同時進行。
? 特點:通信設備可以同時發送和接收數據,就像雙車道道路一樣,流量可以在兩個方向上同時流動。
? 應用場景:常見于電話通信、以太網等需要同時雙向傳輸數據的場合,如互聯網視頻會議、數據中心的服務器通信等。

(三)串口協議和RS-232標準介紹

1、串口協議

串口協議通常指的是一種通過串行通信進行數據傳輸的通信協議。串口協議是通過串行通信傳輸數據的一種約定和規范,通常包括以下幾個方面:

(1)數據格式:定義了數據位(通常為7或8位)、校驗位(可選)、停止位(通常為1或2位)等格式。這些位組合在一起形成一個完整的數據幀。
(2)波特率:指數據傳輸速率,單位為波特(bps)。常見的波特率有9600、19200、38400、115200等。
(3)通信協議:指定了數據如何被解釋、傳輸和接收。常見的協議包括ASCII碼、Modbus、SPI等,具體的選擇取決于應用的需求和設備的兼容性。
(4)硬件接口:定義了物理連接和電氣特性,如信號電平、線路配置(如RS-232、RS-485)、引腳定義等。

2、RS-232標準

RS-232是最早的一種串行通信標準,定義了數據傳輸的電氣特性和信號架構。它具有以下特點:

(1)電氣特性:RS-232標準規定了發送和接收設備之間的電平范圍。典型的RS-232電平為正負12V,用于表示邏輯1和邏輯0。
(2)連接方式:RS-232通常使用DB-9或DB-25連接器,分別具有9個或25個引腳,用于連接串口設備。
(3)信號線:RS-232定義了多條信號線,包括發送數據(TX)、接收數據(RX)、數據就緒(DSR)、數據載波檢測(DCD)、請求發送(RTS)和清除發送(CTS)等。
(4)應用范圍:RS-232廣泛應用于計算機和外圍設備之間的通信,如調制解調器、終端設備、打印機等。

盡管RS-232標準已經存在多年,但隨著技術的進步,如USB和以太網的普及,RS-232在某些領域的使用逐漸減少。然而,它仍然是許多傳統設備中不可或缺的通信接口之一,且在工業控制、測量設備等領域仍然廣泛使用。

3、RS232電平與TTL電平的區別

RS-232和TTL電平在電氣特性、應用場景和連接方式上有很大的不同。RS-232適用于長距離和抗干擾要求較高的通信環境,而TTL電平適合于數字電路和低功耗設備之間的短距離通信。選擇合適的電平標準取決于具體的應用需求,包括通信距離、抗干擾能力、功耗和設備兼容性等因素。

RS-232電平

(1)RS-232標準定義了發送和接收設備之間的電平范圍,典型的RS-232電平為正負12V。邏輯1通常對應于負電平(-3V 至 -15V之間),邏輯0對應于正電平(+3V 至 +15V之間)。這種電平范圍使得RS-232在長距離傳輸和抗干擾能力方面表現優秀。

(2)RS-232通常用于需要較長傳輸距離(最長可達50英尺)和較高抗干擾能力的應用,如計算機串口、調制解調器、終端設備等。它適合于工業環境和長距離通信需求,但其電平范圍較廣,需要較多的電氣和電子元件支持。

(3)RS-232通常使用DB-9DB-25連接器,這些連接器包含多個引腳,用于傳輸數據及控制信號。

TTL電平

(1)TTL(Transistor-Transistor Logic)電平是指通常在邏輯電路中使用的電平標準,典型的TTL電平是0V到5V。邏輯1通常為高電平(約3.3V至5V),邏輯0為低電平(約0V至0.8V),這種電平適合數字電路和集成電路之間的直接通信。

(2)TTL電平通常用于短距離通信和數字電路之間的通信,如微控制器與傳感器之間的串行通信、邏輯電路板之間的通信等。由于其電平范圍較窄,適合于低功耗應用和簡化的通信接口

(3)TTL電平通常使用簡單的單針或雙針連接器,如用于Arduino等開發板的數字輸入輸出引腳。

(四)CH340(串口)

CH340模塊是一種USB轉串口芯片,能夠將USB接口轉換為異步串口通信接口,支持RS-232和TTL電平標準,適用于單片機開發、嵌入式系統及消費電子產品中,提供穩定的數據傳輸和低功耗解決方案。

USB/TTL轉232

CH340是一個USB總線的轉接芯片,實現USB轉串口、USB轉IrDA紅外或者USB轉打印口。為了增加串口通訊的遠距離傳輸及抗干擾能力,RS-232標準使用-15V表示邏輯1,+15V 表示邏輯0。常常會使用MH340芯片對USB/TTL與RS- 232電平的信號進行轉換。

二、標準庫開發

為了解決不同芯片廠商生產的基于Cortex內核的微處理器在軟件上的兼容問題,ARM公司與眾多芯片和軟件廠商共同制定了CMSIS標準(Cortex Microcontroller Software Interface Standard,Cortex微控制器軟件接口標準),意在將所有Cortex芯片廠商產品的軟件接口標準化。

固件庫:

安裝步驟:

img

導入后列表:

三、標準庫點燈

(一)配置GPIO函數:

//1.打開APB2時鐘(不懂為什么第一步是這個的可以參考我上一篇博客)RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);//2.GPIO引腳設置GPIO_InitTypeDef GPIO_InitStructure;//GPIO結構體定義GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//設置GPIO功能模式GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;//設置作用GPIO引腳GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//配置GPIO速度GPIO_Init(GPIOB, &GPIO_InitStructure);//如果你設置的引腳是PB9,()內分別為GPIOB,&你定義的結構體//3.GPIO輸出電平設置GPIO_ResetBits(GPIOB, GPIO_Pin_9);//低電平,點亮LED燈(因為我們的LED燈正極接電源側,負極接引腳PB9,要使LED亮需要使PB9輸出低電平導通)GPIO_SetBits(GPIOB, GPIO_Pin_9);//高電平,LED燈不亮

接線圖:

(二)完整點燈代碼:

?
#include "stm32f10x.h" ? ? ? ? ? ? ? ?  // Device header
#include "Delay.h" ? ? 
?
int main(void)
{/*開啟時鐘*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);   //開啟GPIOA的時鐘//使用各個外設前必須開啟時鐘,否則對外設的操作無效/*GPIO初始化*/GPIO_InitTypeDef GPIO_InitStructure;                    //定義結構體變量GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;        //GPIO模式,賦值為推挽輸出模式GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;               //GPIO引腳,賦值為第0號引腳GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;       //GPIO速度,賦值為50MHzGPIO_Init(GPIOA, &GPIO_InitStructure);                  //將賦值后的構體變量傳遞給GPIO_Init函數//函數內部會自動根據結構體的參數配置相應寄存器//實現GPIOA的初始化/*主循環,循環體內的代碼會一直循環執行*/while (1){/*設置PA0引腳的高低電平,實現LED閃爍,下面展示3種方法*//*方法1:GPIO_ResetBits設置低電平,GPIO_SetBits設置高電平*/GPIO_ResetBits(GPIOA, GPIO_Pin_0);                  //將PA0引腳設置為低電平Delay_ms(500);                                      //延時500msGPIO_SetBits(GPIOA, GPIO_Pin_0);                    //將PA0引腳設置為高電平Delay_ms(500);                                      //延時500ms/*方法2:GPIO_WriteBit設置低/高電平,由Bit_RESET/Bit_SET指定*/GPIO_WriteBit(GPIOA, GPIO_Pin_0, Bit_RESET);        //將PA0引腳設置為低電平Delay_ms(500);                                      //延時500msGPIO_WriteBit(GPIOA, GPIO_Pin_0, Bit_SET);          //將PA0引腳設置為高電平Delay_ms(500);                                      //延時500ms/*方法3:GPIO_WriteBit設置低/高電平,由數據0/1指定,數據需要強轉為BitAction類型*/GPIO_WriteBit(GPIOA, GPIO_Pin_0, (BitAction)0);     //將PA0引腳設置為低電平Delay_ms(500);                                      //延時500msGPIO_WriteBit(GPIOA, GPIO_Pin_0, (BitAction)1);     //將PA0引腳設置為高電平Delay_ms(500);                                      //延時500ms}
}?

四、標準庫串口通信實現

LED GPIO 初始化:

#include "stm32f10x.h"
#include "OLED_Font.h"
?
/*引腳配置*/
#define OLED_W_SCL(x)       GPIO_WriteBit(GPIOB, GPIO_Pin_8, (BitAction)(x))
#define OLED_W_SDA(x)       GPIO_WriteBit(GPIOB, GPIO_Pin_9, (BitAction)(x))
?
/*引腳初始化*/
void OLED_I2C_Init(void)
{RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;GPIO_Init(GPIOB, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;GPIO_Init(GPIOB, &GPIO_InitStructure);OLED_W_SCL(1);OLED_W_SDA(1);
}
?
/*** @brief  I2C開始* @param  無* @retval 無*/
void OLED_I2C_Start(void)
{OLED_W_SDA(1);OLED_W_SCL(1);OLED_W_SDA(0);OLED_W_SCL(0);
}
?
/*** @brief  I2C停止* @param  無* @retval 無*/
void OLED_I2C_Stop(void)
{OLED_W_SDA(0);OLED_W_SCL(1);OLED_W_SDA(1);
}
?
/*** @brief  I2C發送一個字節* @param  Byte 要發送的一個字節* @retval 無*/
void OLED_I2C_SendByte(uint8_t Byte)
{uint8_t i;for (i = 0; i < 8; i++){OLED_W_SDA(Byte & (0x80 >> i));OLED_W_SCL(1);OLED_W_SCL(0);}OLED_W_SCL(1);  //額外的一個時鐘,不處理應答信號OLED_W_SCL(0);
}
?
/*** @brief  OLED寫命令* @param  Command 要寫入的命令* @retval 無*/
void OLED_WriteCommand(uint8_t Command)
{OLED_I2C_Start();OLED_I2C_SendByte(0x78);        //從機地址OLED_I2C_SendByte(0x00);        //寫命令OLED_I2C_SendByte(Command); OLED_I2C_Stop();
}
?
/*** @brief  OLED寫數據* @param  Data 要寫入的數據* @retval 無*/
void OLED_WriteData(uint8_t Data)
{OLED_I2C_Start();OLED_I2C_SendByte(0x78);        //從機地址OLED_I2C_SendByte(0x40);        //寫數據OLED_I2C_SendByte(Data);OLED_I2C_Stop();
}
?
/*** @brief  OLED設置光標位置* @param  Y 以左上角為原點,向下方向的坐標,范圍:0~7* @param  X 以左上角為原點,向右方向的坐標,范圍:0~127* @retval 無*/
void OLED_SetCursor(uint8_t Y, uint8_t X)
{OLED_WriteCommand(0xB0 | Y);                    //設置Y位置OLED_WriteCommand(0x10 | ((X & 0xF0) >> 4));    //設置X位置高4位OLED_WriteCommand(0x00 | (X & 0x0F));           //設置X位置低4位
}
?
/*** @brief  OLED清屏* @param  無* @retval 無*/
void OLED_Clear(void)
{ ?uint8_t i, j;for (j = 0; j < 8; j++){OLED_SetCursor(j, 0);for(i = 0; i < 128; i++){OLED_WriteData(0x00);}}
}
?
/*** @brief  OLED顯示一個字符* @param  Line 行位置,范圍:1~4* @param  Column 列位置,范圍:1~16* @param  Char 要顯示的一個字符,范圍:ASCII可見字符* @retval 無*/
void OLED_ShowChar(uint8_t Line, uint8_t Column, char Char)
{ ? ?   uint8_t i;OLED_SetCursor((Line - 1) * 2, (Column - 1) * 8);       //設置光標位置在上半部分for (i = 0; i < 8; i++){OLED_WriteData(OLED_F8x16[Char - ' '][i]);          //顯示上半部分內容}OLED_SetCursor((Line - 1) * 2 + 1, (Column - 1) * 8);   //設置光標位置在下半部分for (i = 0; i < 8; i++){OLED_WriteData(OLED_F8x16[Char - ' '][i + 8]);      //顯示下半部分內容}
}
?
/*** @brief  OLED顯示字符串* @param  Line 起始行位置,范圍:1~4* @param  Column 起始列位置,范圍:1~16* @param  String 要顯示的字符串,范圍:ASCII可見字符* @retval 無*/
void OLED_ShowString(uint8_t Line, uint8_t Column, char *String)
{uint8_t i;for (i = 0; String[i] != '\0'; i++){OLED_ShowChar(Line, Column + i, String[i]);}
}
?
/*** @brief  OLED次方函數* @retval 返回值等于X的Y次方*/
uint32_t OLED_Pow(uint32_t X, uint32_t Y)
{uint32_t Result = 1;while (Y--){Result *= X;}return Result;
}
?
/*** @brief  OLED顯示數字(十進制,正數)* @param  Line 起始行位置,范圍:1~4* @param  Column 起始列位置,范圍:1~16* @param  Number 要顯示的數字,范圍:0~4294967295* @param  Length 要顯示數字的長度,范圍:1~10* @retval 無*/
void OLED_ShowNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
{uint8_t i;for (i = 0; i < Length; i++)                            {OLED_ShowChar(Line, Column + i, Number / OLED_Pow(10, Length - i - 1) % 10 + '0');}
}
?
/*** @brief  OLED顯示數字(十進制,帶符號數)* @param  Line 起始行位置,范圍:1~4* @param  Column 起始列位置,范圍:1~16* @param  Number 要顯示的數字,范圍:-2147483648~2147483647* @param  Length 要顯示數字的長度,范圍:1~10* @retval 無*/
void OLED_ShowSignedNum(uint8_t Line, uint8_t Column, int32_t Number, uint8_t Length)
{uint8_t i;uint32_t Number1;if (Number >= 0){OLED_ShowChar(Line, Column, '+');Number1 = Number;}else{OLED_ShowChar(Line, Column, '-');Number1 = -Number;}for (i = 0; i < Length; i++)                            {OLED_ShowChar(Line, Column + i + 1, Number1 / OLED_Pow(10, Length - i - 1) % 10 + '0');}
}
?
/*** @brief  OLED顯示數字(十六進制,正數)* @param  Line 起始行位置,范圍:1~4* @param  Column 起始列位置,范圍:1~16* @param  Number 要顯示的數字,范圍:0~0xFFFFFFFF* @param  Length 要顯示數字的長度,范圍:1~8* @retval 無*/
void OLED_ShowHexNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
{uint8_t i, SingleNumber;for (i = 0; i < Length; i++)                            {SingleNumber = Number / OLED_Pow(16, Length - i - 1) % 16;if (SingleNumber < 10){OLED_ShowChar(Line, Column + i, SingleNumber + '0');}else{OLED_ShowChar(Line, Column + i, SingleNumber - 10 + 'A');}}
}
?
/*** @brief  OLED顯示數字(二進制,正數)* @param  Line 起始行位置,范圍:1~4* @param  Column 起始列位置,范圍:1~16* @param  Number 要顯示的數字,范圍:0~1111 1111 1111 1111* @param  Length 要顯示數字的長度,范圍:1~16* @retval 無*/
void OLED_ShowBinNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
{uint8_t i;for (i = 0; i < Length; i++)                            {OLED_ShowChar(Line, Column + i, Number / OLED_Pow(2, Length - i - 1) % 2 + '0');}
}
?
/*** @brief  OLED初始化* @param  無* @retval 無*/
void OLED_Init(void)
{uint32_t i, j;for (i = 0; i < 1000; i++)          //上電延時{for (j = 0; j < 1000; j++);}OLED_I2C_Init();            //端口初始化OLED_WriteCommand(0xAE);    //關閉顯示OLED_WriteCommand(0xD5);    //設置顯示時鐘分頻比/振蕩器頻率OLED_WriteCommand(0x80);OLED_WriteCommand(0xA8);    //設置多路復用率OLED_WriteCommand(0x3F);OLED_WriteCommand(0xD3);    //設置顯示偏移OLED_WriteCommand(0x00);OLED_WriteCommand(0x40);    //設置顯示開始行OLED_WriteCommand(0xA1);    //設置左右方向,0xA1正常 0xA0左右反置OLED_WriteCommand(0xC8);    //設置上下方向,0xC8正常 0xC0上下反置
?OLED_WriteCommand(0xDA);    //設置COM引腳硬件配置OLED_WriteCommand(0x12);OLED_WriteCommand(0x81);    //設置對比度控制OLED_WriteCommand(0xCF);
?OLED_WriteCommand(0xD9);    //設置預充電周期OLED_WriteCommand(0xF1);
?OLED_WriteCommand(0xDB);    //設置VCOMH取消選擇級別OLED_WriteCommand(0x30);
?OLED_WriteCommand(0xA4);    //設置整個顯示打開/關閉
?OLED_WriteCommand(0xA6);    //設置正常/倒轉顯示
?OLED_WriteCommand(0x8D);    //設置充電泵OLED_WriteCommand(0x14);
?OLED_WriteCommand(0xAF);    //開啟顯示OLED_Clear();               //OLED清屏
}

波特率配置:

img

#include "stm32f10x.h" ? ? ? ? ? ? ? ?  // Device header
#include <stdio.h>
#include <stdarg.h>
?
/*** 函 ?  數:串口初始化* 參 ?  數:無* 返 回 值:無*/
void Serial_Init(void)
{/*開啟時鐘*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);  //開啟USART1的時鐘RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);   //開啟GPIOA的時鐘/*GPIO初始化*/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);                  //將PA9引腳初始化為復用推挽輸出/*USART初始化*/USART_InitTypeDef USART_InitStructure;                  //定義結構體變量USART_InitStructure.USART_BaudRate = 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);               //將結構體變量交給USART_Init,配置USART1/*USART使能*/USART_Cmd(USART1, ENABLE);                              //使能USART1,串口開始運行
}
?
/*** 函 ?  數:串口發送一個字節* 參 ?  數:Byte 要發送的一個字節* 返 回 值:無*/
void Serial_SendByte(uint8_t Byte)
{USART_SendData(USART1, Byte);       //將字節數據寫入數據寄存器,寫入后USART自動生成時序波形while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);   //等待發送完成/*下次寫入數據寄存器會自動清除發送完成標志位,故此循環后,無需清除標志位*/
}
?
/*** 函 ?  數:串口發送一個數組* 參 ?  數:Array 要發送數組的首地址* 參 ?  數:Length 要發送數組的長度* 返 回 值:無*/
void Serial_SendArray(uint8_t *Array, uint16_t Length)
{uint16_t i;for (i = 0; i < Length; i ++)       //遍歷數組{Serial_SendByte(Array[i]);      //依次調用Serial_SendByte發送每個字節數據}
}
?
/*** 函 ?  數:串口發送一個字符串* 參 ?  數:String 要發送字符串的首地址* 返 回 值:無*/
void Serial_SendString(char *String)
{uint8_t i;for (i = 0; String[i] != '\0'; i ++)//遍歷字符數組(字符串),遇到字符串結束標志位后停止{Serial_SendByte(String[i]);     //依次調用Serial_SendByte發送每個字節數據}
}
?
/*** 函 ?  數:次方函數(內部使用)* 返 回 值:返回值等于X的Y次方*/
uint32_t Serial_Pow(uint32_t X, uint32_t Y)
{uint32_t Result = 1;    //設置結果初值為1while (Y --)            //執行Y次{Result *= X;        //將X累乘到結果}return Result;
}
?
/*** 函 ?  數:串口發送數字* 參 ?  數:Number 要發送的數字,范圍:0~4294967295* 參 ?  數:Length 要發送數字的長度,范圍:0~10* 返 回 值:無*/
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');    //依次調用Serial_SendByte發送每位數字}
}
?
/*** 函 ?  數:使用printf需要重定向的底層函數* 參 ?  數:保持原始格式即可,無需變動* 返 回 值:保持原始格式即可,無需變動*/
int fputc(int ch, FILE *f)
{Serial_SendByte(ch);            //將printf的底層重定向到自己的發送字節函數return ch;
}
?
/*** 函 ?  數:自己封裝的prinf函數* 參 ?  數:format 格式化字符串* 參 ?  數:... 可變的參數列表* 返 回 值:無*/
void Serial_Printf(char *format, ...)
{char String[100];               //定義字符數組va_list arg;                    //定義可變參數列表數據類型的變量argva_start(arg, format);          //從format開始,接收參數列表到arg變量vsprintf(String, format, arg);  //使用vsprintf打印格式化字符串和參數列表到字符數組中va_end(arg);                    //結束變量argSerial_SendString(String);      //串口發送字符數組(字符串)
}

主函數代碼:

#include "stm32f10x.h" ? ? ? ? ? ? ? ?  // Device header
#include "Delay.h"
#include "OLED.h"
#include "Serial.h"
?
int main(void)
{/*模塊初始化*/OLED_Init();                        //OLED初始化Serial_Init();                      //串口初始化/*串口基本函數*/Serial_SendByte(0x41);              //串口發送一個字節數據0x41uint8_t MyArray[] = {0x42, 0x43, 0x44, 0x45};   //定義數組Serial_SendArray(MyArray, 4);       //串口發送一個數組//Serial_SendString("hello windows!");      //串口發送字符串Serial_SendNumber(111, 3);          //串口發送數字/*下述3種方法可實現printf的效果*//*方法1:直接重定向printf,但printf函數只有一個,此方法不能在多處使用*/printf("\r\nNum2=%d", 222);         //串口發送printf打印的格式化字符串//需要重定向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){Serial_SendString("hello windows");Serial_Printf("\r\n");}
}

結果演示:

STM32以查詢方式接收上位機(win10)串口發來的數據,如果接收到“Y”則點亮鏈接到stm32上的一個LED燈;接收到“N”則熄滅LED燈。

代碼演示:

#include "stm32f10x.h" ? ? ? ? ? ? ? ?  // Device header
#include <stdio.h>
?
void USART_Config(void)
{
//1.開啟GPIOA和USART1的時鐘RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);//2.結構體定義GPIO_InitTypeDef GPIO_InitStructure;USART_InitTypeDef USART_InitStructure;//3.USART設置RX/TX//USART1_TX,默認情況下復用PA9引腳GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//復用推挽輸出GPIO_Init(GPIOA, &GPIO_InitStructure); //USART1_RX,默認情況下復用PA10引腳GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;//浮空輸入GPIO_Init(GPIOA, &GPIO_InitStructure); //4.USART1參數配置USART_InitStructure.USART_BaudRate = 9600;//設置波特率為9600USART_InitStructure.USART_WordLength = USART_WordLength_8b; //數據位占8位USART_InitStructure.USART_StopBits = USART_StopBits_1; //1位停止位USART_InitStructure.USART_Parity = USART_Parity_No; //無校驗USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None;USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;USART_Init(USART1, &USART_InitStructure);
?USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);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);//5.初始化串口1USART_Cmd(USART1, ENABLE); //使能串口1
?
}
?
//重定向c 庫函數printf 到串口,重定向后可使用printf 函數
int fputc(int ch, FILE *f)
{/* 發送一個字節數據到串口 */USART_SendData(USART1, (uint8_t) ch);/* 等待發送完畢 */while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);return (ch);
}
?
///重定向c 庫函數scanf 到串口,重寫向后可使用scanf、getchar 等函數
int fgetc(FILE *f)
{/* 等待串口輸入數據 */while (USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET);return (int)USART_ReceiveData(USART1);
}
?
int main(void)
{USART_Config();//1.開啟GPIOB時鐘RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//2.結構體定義GPIO_InitTypeDef GPIO_InitStructure;//3.GPIO配置  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽輸出GPIO_Init(GPIOA, &GPIO_InitStructure); //4.初始化燈GPIO_SetBits(GPIOA,GPIO_Pin_4);char ch;while(1){printf("請輸入指令:Y亮燈,N滅燈!");ch=getchar();printf("接收到字符:%c\n",ch);switch(ch){case 'N':GPIO_SetBits(GPIOA,GPIO_Pin_4);break;case 'Y':GPIO_ResetBits(GPIOA,GPIO_Pin_4);break;default:break;}// 等待一段時間,以便在串口調試工具中可以看到消息之間的間隔 ?for (uint32_t i = 0; i < 10000000; i++); ? ?} ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 
}
#include "stm32f10x.h" // Device header
#include "Serial.h"
//操作IO口的三個步驟
//1、使用RCC開啟GPIO時鐘
//2、使用GPIO_Init函數初始化GPIO
//3、使用輸出或輸入函數控制GPIO口
uint8_t KeyNum;
uint8_t RxData;
int main()
{OLED_Init();Serial_Init();GPIO_WriteBit(GPIOA,GPIO_Pin_6,Bit_SET);//將A6口初始化為電平
//GPIO_SetBits(GPIOA,GPIO_Pin_6);另一種初始化函數while(1){if(USART_GetFlagStatus(USART1,USART_FLAG_RXNE)==SET)//若接收寄存器數據轉到RDR中,則RXNE標志位置一,標志位自動清零{RxData=USART_ReceiveData(USART1);if(RxData=='Y'){GPIO_ResetBits(GPIOA,GPIO_Pin_6);}if(RxData=='N'){GPIO_SetBits(GPIOA,GPIO_Pin_6);}}}
}

五、Keil的仿真邏輯分析儀觀察時序波形

img

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/38867.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/38867.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/38867.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Spring中利用重載與靜態分派

Spring中利用重載與靜態分派 在Java和Spring框架中&#xff0c;重載&#xff08;Overloading&#xff09;和靜態分派&#xff08;Static Dispatch&#xff09;是兩個非常重要的概念&#xff0c;它們在處理類方法選擇和執行過程中扮演著關鍵角色。本文旨在深入探討Spring環境下…

入選頂會ICML,清華AIR等聯合發布蛋白質語言模型ESM-AA,超越傳統SOTA

作為細胞內無數生化反應的驅動力&#xff0c;蛋白質在細胞微觀世界中扮演著建筑師和工程師的角色&#xff0c;不僅催化著生命活動&#xff0c;更是構筑、維系生物體形態與功能的基礎構件。正是蛋白質之間的互動、協同作用&#xff0c;支撐起了生命的宏偉藍圖。 然而&#xff0…

Ubuntu DNS服務配置 深度解析

測試方法 resolvectl status dig alidns.com 修改實踐 直接用接口配置&#xff0c;沒用 /etc/resolv.conf&#xff0c;有效 /etc/netplan/01-network-manager-all.yaml,無效 /etc/systemd/resolved.conf&#xff0c;見link&#xff0c;為全局配置 [Resolve] DNS1.1.1.1 Fa…

Adobe Premiere 視頻編輯軟件下載安裝,pr全系列分享 輕松編輯視頻

Adobe Premiere&#xff0c;自其誕生之日起&#xff0c;便以其卓越的性能和出色的表現&#xff0c;穩坐視頻編輯領域的王者寶座&#xff0c;贏得了無數專業編輯人員與廣大愛好者的青睞。這款強大的視頻編輯軟件&#xff0c;憑借其豐富的功能和靈活的操作性&#xff0c;為用戶提…

2024年道路運輸安全員(企業管理人員)備考題庫資料。

46.危險貨物道路運輸隨車攜帶的單據&#xff0c;下列選項不屬于的是&#xff08;&#xff09;。 A.道路運輸危險貨物安全卡 B.運單或者電子運單 C.道路危險貨物運輸從業資格證 D.車輛檢測報告 答案&#xff1a;D 47.危險貨物運輸駕駛人員在24小時內實際駕駛車輛時間累計不…

ROS2在rviz2中實時顯示軌跡和點

本文是將《ROS在rviz中實時顯示軌跡和點》博客中rviz軌跡顯示轉為ROS2環境中的rviz2顯示。 ros2的工作空間創建這里就不展示了。 包的創建 ros2 pkg create --build-type ament_cmake showpath --dependencies rclcpp nav_msgs geometry_msgs tf2_geometry_msgsshowpath.cpp…

Windows批處理入門:快速掌握批處理腳本的基本技巧

一、前言 在Windows操作系統中&#xff0c;批處理文件&#xff08;Batch File&#xff09;是一種非常實用的工具&#xff0c;它允許用戶通過簡單的命令行腳本來自動化各種任務。無論是系統管理員、開發人員&#xff0c;還是普通用戶&#xff0c;掌握批處理文件的基本知識都能極…

【漏洞復現】和豐多媒體信息發布系統 QH.aspx 任意文件上傳漏洞

0x01 產品簡介 和豐多媒體信息發布系統也稱數字標牌&#xff08;Digital Signage&#xff09;&#xff0c;是指通過大屏幕終端顯示設備&#xff0c;發布商業、財經和娛樂信息的多媒體專業視聽系統&#xff0c;常被稱為除紙張媒體、電臺、電視、互聯網之外的“第五媒體”。該系…

Ansible如何控制playbook的執行順序

對 Ansible 劇本資源打標簽 在處理大型或復雜的劇本時,如果只希望運行部分劇本或部分任務。可以將標簽應用于可能要跳過或運行的特定資源。 通過標簽來標記資源,在資源上使用tags關鍵字,然后是要應用的標記列表。在Ansible中tags標記可用于下列資源&#xff1a; 每個任務,這…

1-4.時間序列數據建模流程范例

文章最前&#xff1a; 我是Octopus&#xff0c;這個名字來源于我的中文名–章魚&#xff1b;我熱愛編程、熱愛算法、熱愛開源。所有源碼在我的個人github &#xff1b;這博客是記錄我學習的點點滴滴&#xff0c;如果您對 Python、Java、AI、算法有興趣&#xff0c;可以關注我的…

信息學奧賽初賽天天練-41-CSP-J2021基礎題-n個數取最大、樹的邊數、遞歸、遞推、深度優先搜索應用

PDF文檔公眾號回復關鍵字:20240701 2021 CSP-J 選擇題 單項選擇題&#xff08;共15題&#xff0c;每題2分&#xff0c;共計30分&#xff1a;每題有且僅有一個正確選項&#xff09; 4.以比較作為基本運算&#xff0c;在N個數中找出最大數&#xff0c;最壞情況下所需要的最少比…

我在中東做MCN,月賺10萬美金

圖片&#xff5c;Photo by Ben Koorengevel on Unsplash ©自象限原創 作者丨程心 在迪拜購物中心和世界最高建筑哈利法塔旁的主街上&#xff0c;徐晉已經“蹲”了三個小時&#xff0c;每當遇到穿著時髦的年輕男女&#xff0c;他都會上前詢問&#xff0c;有沒有意愿成為…

【計算機網絡】常見的網絡通信協議

目錄 1. TCP/IP協議 2. HTTP協議 3. FTP協議 4. SMTP協議 5. POP3協議 6. IMAP協議 7. DNS協議 8. DHCP協議 9. SSH協議 10. SSL/TLS協議 11. SNMP協議 12. NTP協議 13. VoIP協議 14. WebSocket協議 15. BGP協議 16. OSPF協議 17. RIP協議 18. ICMP協議 1…

網頁自動化測試開發中記錄pytest

1切換cmd文件目錄C:\Users\14600>D: D:\>cd D:\worksoftware D:\worksoftware>2單個py文件打包成.exe文件1.pyinstaller -F -c (項目主文件)test_01shouye.py 該路徑下存在文件名&#xff0c;主項目文件 test_01shouye.py 2.執行spec文件&#xff1a; pyinstaller -F …

C語言部分復習筆記

1. 指針和數組 數組指針 和 指針數組 int* p1[10]; // 指針數組int (*p2)[10]; // 數組指針 因為 [] 的優先級比 * 高&#xff0c;p先和 [] 結合說明p是一個數組&#xff0c;p先和*結合說明p是一個指針 括號保證p先和*結合&#xff0c;說明p是一個指針變量&#xff0c;然后指…

Web2Code :網頁理解和代碼生成能力的評估框架

多模態大型語言模型&#xff08;MLLMs&#xff09;在過去幾年中取得了爆炸性的增長。利用大型語言模型&#xff08;LLMs&#xff09;中豐富的常識知識&#xff0c;MLLMs在處理和推理各種模態&#xff08;如圖像、視頻和音頻&#xff09;方面表現出色&#xff0c;涵蓋了識別、推…

系統中非功能性需求的思考

概要 設計系統時不僅要考慮功能性需求&#xff0c;還要考慮一些非功能性需求&#xff0c;比如&#xff1a; 擴展性可靠性和冗余安全和隱私服務依賴SLA要求 下面對這5項需要考慮的事項做個簡單的說明 1. 可擴展性 數據量增長如何擴展&#xff1f; 流量增長如何擴展&#xf…

【LLM教程-llama】如何Fine Tuning大語言模型?

今天給大家帶來了一篇超級詳細的教程,手把手教你如何對大語言模型進行微調(Fine Tuning)&#xff01;&#xff08;代碼和詳細解釋放在后文&#xff09; 目錄 大語言模型進行微調(Fine Tuning)需要哪些步驟&#xff1f; 大語言模型進行微調(Fine Tuning)訓練過程及代碼 大語言…

VuePress介紹

從本文開始&#xff0c;動手搭建自己的博客&#xff01;希望讀者能跟著一起動手&#xff0c;這樣才能真正掌握。 ? VuePress 是什么 VuePress 是由 Vue 作者帶領團隊開發的&#xff0c;非常火&#xff0c;使用的人很多&#xff1b;Vue 框架官網也是用了 VuePress 搭建的。即…

000.二分查找算法題解目錄

000.二分查找算法題解目錄 69. x 的平方根&#xff08;簡單&#xff09;