基于stm32超声波+蓝牙一体式机器人(小车)

文章详细介绍了基于STM32微控制器设计的一款超声波避障蓝牙遥控小车,包括超声波模块的配置、蓝牙通信模块的使用、OLED显示屏的显示功能以及电机控制。小车能通过超声波传感器检测障碍物并根据距离调整行驶方向,同时可以通过蓝牙模块接收手机指令进行遥控操作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近制作了一款基于stm32的有两个功能的机器人(小车),自我感觉还算不错,于是想分享出来,如果能帮到你那我很开心。如果你感觉哪里有错误和需要改进的地方,欢迎发表意见,我会真诚接受并虚心学习!谢谢哦!

一、成品图片+思维导图

  1. 成品图片

  1. 思维导图

二、超声波小车

  1. 时钟模块

超声波的使用需要配置一个TIM定时器,是用来计算超声波模块的TRIG引脚发射脉冲信号到ECHO引脚接收到返回的脉冲信号这一区间段的时间(单位换算成秒),在我的这个项目中,我使用了TIM4来完成这一步骤。具体配置情况请详见下面的配置函数。(注意!!!第二段代码是TIM4的定时器中断函数,请放在使用TIM4的地方即可)


void Timer4_Init(void)
{
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4,ENABLE);
    
    TIM_InternalClockConfig(TIM4);/*使用内部时钟TIM4*/
    
    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
    TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;//不分频
    TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
    TIM_TimeBaseInitStructure.TIM_Period = 1000 - 1;
    TIM_TimeBaseInitStructure.TIM_Prescaler = 72 - 1;
    TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;
    TIM_TimeBaseInit(TIM4,&TIM_TimeBaseInitStructure);
    
    TIM_ClearFlag(TIM4,TIM_FLAG_Update);
    /*打开中断*/
    TIM_ITConfig(TIM4,TIM_IT_Update,ENABLE);
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//NVIC优先级分组2
    
    NVIC_InitTypeDef NVIC_InitStructure;
    NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
    NVIC_Init(&NVIC_InitStructure);
    
    TIM_Cmd(TIM4,ENABLE);/*使能定时器4*/
}

uint16_t Timer4_GetCounter(void) //获取定时器4的值
{
    return TIM_GetCounter(TIM4);
}

void TIM4_IRQHandler(void)  //定时器4中断
{    
    if(TIM_GetITStatus(TIM4,TIM_IT_Update) == SET)
    {
        TIM_ClearITPendingBit(TIM4,TIM_IT_Update);//更新中断
        count++;
    }
}
  1. 超声波模块

超声波的配置也是相对简单,先给TRIG引脚一个高电平,在使用延时函数延时20微秒,打开TIM4计时,再检测ECHO是否接收,若接收,获取TIM4的计时结果,若为接收,则等待接收。具体请详见下面代码。


/*记录定时器溢出次数*/
uint16_t count = 0;


void Ultrasonic_GPIO_Init(void)
{
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    
    /*TRIG触发信号*/
    GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Pin=GPIO_Pin_15;
    GPIO_Init(GPIOB, &GPIO_InitStructure);

    /*ECOH回响信号*/
    GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD;
    GPIO_InitStructure.GPIO_Pin=GPIO_Pin_14;
    GPIO_Init(GPIOB, & GPIO_InitStructure);
}


float Get_Ultrasonic_distance(void)
{
    Timer4_Init();
    Ultrasonic_GPIO_Init();
    
    float distance = 0 ,sum = 0;
    uint32_t tim = 0 ;
    uint8_t i = 0;
    
    while(i!=5)        //连续测5次,求平均值,使结果准确
    {
        GPIO_ResetBits(GPIOB, GPIO_Pin_15);//先置为低电平
        GPIO_SetBits(GPIOB, GPIO_Pin_15);  //拉高信号,作为触发信号
        Delay_us(20);                          //高电平信号超过10us
        GPIO_ResetBits(GPIOB, GPIO_Pin_15);
        
        /*等待回响信号,回响信号到来*/
        while(GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_14) == RESET);
        TIM_SetCounter(TIM4,0); //将TIM4计数寄存器的计数值清零
        TIM_Cmd(TIM4,ENABLE);//开启定时器计数
        
        i+=1;
        
        /*回响信号消失*/
        while(GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_14) == SET);
        TIM_Cmd(TIM4, DISABLE);//关闭定时器
        
        tim = Timer4_GetCounter();//获取计TIM4数寄存器中的计数值,以便计算回响信号时间
        distance = ((float)(tim + count * 1000)) / 58.0;//通过回响信号计算距离
        sum += distance;
        //TIM4->CNT = 0; 
        TIM_SetCounter(TIM4,0);  //将TIM4计数寄存器的计数值清零
        count = 0;  //中断溢出次数清零
        Delay_ms(20);
    }
    
    distance = (sum / 5.0);    //单位cm 
    return distance;        //距离作为函数返回值
}

void TIM4_IRQHandler(void)  //定时器4中断
{    
    if(TIM_GetITStatus(TIM4,TIM_IT_Update) == SET)
    {
        TIM_ClearITPendingBit(TIM4,TIM_IT_Update);//更新中断
        count++;
    }
}//这里使用了中断函数,就放这里
  1. 逻辑函数

逻辑函数就是小车收到数据该如何转向的问题,请详见下面的代码,注释很详细的。


void Ultrasonic_car(void)
{
    int8_t Speed = 50;  /*速度范围为[0,100]*/
    float left_distance = 0.0;
    float right_distance = 0.0;
    float front_distance = 0.0;
    
    Delay_ms(500);
    SG90_Front();//舵机面向前方
    front_distance = Get_Ultrasonic_distance();//测出前方的距离
    OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
    
    SG90_Left();//舵机转向左边
    Delay_ms(500);
    left_distance = Get_Ultrasonic_distance();//测出左边的距离
    OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
    
    SG90_Right();//舵机转向右边
    Delay_ms(500);
    right_distance = Get_Ultrasonic_distance();//测出右边的距离
    OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
    SG90_Front();//舵机面向前方
    
    if((front_distance>left_distance)&&(front_distance>right_distance))
    {
        Car_Forword(Speed);
        front_distance = Get_Ultrasonic_distance();//测出前方的距离
        OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
    }
    else if((left_distance>front_distance)&&(left_distance>right_distance))
    {
        Car_Left(Speed);
        Delay_ms(500);
        Car_Forword(Speed);
        front_distance = Get_Ultrasonic_distance();//测出前方的距离
        OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
    }
    else if((right_distance>front_distance)&&(right_distance>left_distance))
    {
        Car_Right(Speed);
        Delay_ms(500);
        Car_Forword(Speed);
        front_distance = Get_Ultrasonic_distance();//测出前方的距离
        OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
    }
    else Car_Stop();
    
    while(1)
    {
        OLED_ShowNum(2,7,Speed,3);
        front_distance = Get_Ultrasonic_distance();//测出前方的距离
        OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
        if(front_distance<=50)
        {
            Car_Stop();
            SG90_Left();//舵机转向左边
            Delay_ms(800);
            left_distance = Get_Ultrasonic_distance();//测出左边的距离
            OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
            
            SG90_Right();//舵机转向y边
            Delay_ms(800);
            right_distance = Get_Ultrasonic_distance();//测出y边的距离
            OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
            SG90_Front();//舵机面向前方
            front_distance = Get_Ultrasonic_distance();//测出前方的距离
            OLED_ShowNum(1,6,Get_Ultrasonic_distance(),5);
            
            if(left_distance>right_distance)
            {
                Car_Left(Speed);
                Delay_ms(500);
                Car_Stop();
                Delay_s(1);
                Car_Forword(Speed);
            }
            else if((left_distance<right_distance))
            {
                Car_Right(Speed);
                Delay_ms(500);
                Car_Stop();
                Delay_s(1);
                Car_Forword(Speed);
            }
            else
            {
                Car_Back(Speed);
                Delay_ms(800);
                Car_Left(Speed);
                Delay_ms(800);
                Car_Forword(Speed);
            }
        }
        
    }
}

写到这里,超声波小车就结束了,重点很少,主要是得理解思路。上面函数中出现的其他函数的调用,会在下面(五、通用功能的函数文件)给出C文件的。

三、蓝牙小车(具体用到的GPIO请见上面的思维导图)(注意!!!5V供电)

用到的模块是HC-05蓝牙模块,蓝牙模块上的TXD引脚和RXD引脚分别接stm32的USART串口上的RXD和TXD对应的引脚。(我使用的是STM32F103C8T6,它的USART1的RXD和TXD分别是GPIOA9和GPIOA10),具体请见下面的代码。

  1. 串口初始化模块


void Serial_Init(void)
{
    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);
    
    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_InitTypeDef USART_InitStructure;
    USART_InitStructure.USART_BaudRate = 9600;
    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_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);
    
    USART_Cmd(USART1, ENABLE);
}
  1. 蓝牙模块(发送和接收数据的函数)


uint8_t Serial_RxData;
uint8_t Serial_RxFlag;

void Serial_SendByte(uint8_t Byte)    //发送一个字节
{
    USART_SendData(USART1, Byte);
    while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);
}

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]);
    }
}

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');
    }
}

uint8_t Serial_GetRxFlag(void) //获取中断标志位
{
    if (Serial_RxFlag == 1)
    {
        Serial_RxFlag = 0;
        return 1;
    }
    return 0;
}

uint8_t Serial_GetRxData(void) //获取收到的数据
{
    return Serial_RxData;
}

void USART1_IRQHandler(void) //串口USART1的中断
{
    if (USART_GetITStatus(USART1, USART_IT_RXNE) == SET)
    {
        Serial_RxData = USART_ReceiveData(USART1);
        Serial_RxFlag = 1;
        USART_ClearITPendingBit(USART1, USART_IT_RXNE);
    }
}
  1. 逻辑函数(手机连接蓝牙模块,发送指令使小车完成相应的动作)


uint8_t RxData=0;

void Hc05_car(void)
{
    while(1)
    {
        if(Serial_GetRxFlag() == 1)
        {
            RxData=Serial_GetRxData();
            //OLED_ShowHexNum(3,1,RxData,2);
        }
        
        switch(RxData)
        {
            case 1:Car_Forword(100);break;
            case 2:Car_Stop();break;
            case 3:Car_Back(100);break;
            case 4:Car_Stop();break;
            case 5:Car_Left(100);break;
            case 6:Car_Stop();break;
            case 7:Car_Right(100);break;
            case 8:Car_Stop();break;
            case 9:Buzzer_ON1();break;
            case 10:Buzzer_OFF();break;
            case 11:LED_ON();break;
            case 12:LED_OFF();break;
            case 13:LED_Flash_ON();break;
            case 14:LED_Flash_OFF();break;
        }
    }
}

四、其他模块和功能

  1. OLED显示屏(四引脚)(温馨提示!!!关于这个模块掌握的还不够好,但是我可以分享出大佬代码,供大家参考。。。共有三个文件,大家直接复制粘贴用即可)

OLED.c


#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清屏
}

OLED.h


#ifndef __OLED_H
#define __OLED_H

void OLED_Init(void);
void OLED_Clear(void);
void OLED_ShowChar(uint8_t Line, uint8_t Column, char Char);
void OLED_ShowString(uint8_t Line, uint8_t Column, char *String);
void OLED_ShowNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length);
void OLED_ShowSignedNum(uint8_t Line, uint8_t Column, int32_t Number, uint8_t Length);
void OLED_ShowHexNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length);
void OLED_ShowBinNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length);

#endif

OLED_Font.h


#ifndef __OLED_FONT_H
#define __OLED_FONT_H

/*OLED字模库,宽8像素,高16像素*/
const uint8_t OLED_F8x16[][16]=
{
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,//  0
    
    0x00,0x00,0x00,0xF8,0x00,0x00,0x00,0x00,
    0x00,0x00,0x00,0x33,0x30,0x00,0x00,0x00,//! 1
    
    0x00,0x10,0x0C,0x06,0x10,0x0C,0x06,0x00,
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,//" 2
    
    0x40,0xC0,0x78,0x40,0xC0,0x78,0x40,0x00,
    0x04,0x3F,0x04,0x04,0x3F,0x04,0x04,0x00,//# 3
    
    0x00,0x70,0x88,0xFC,0x08,0x30,0x00,0x00,
    0x00,0x18,0x20,0xFF,0x21,0x1E,0x00,0x00,//$ 4
    
    0xF0,0x08,0xF0,0x00,0xE0,0x18,0x00,0x00,
    0x00,0x21,0x1C,0x03,0x1E,0x21,0x1E,0x00,//% 5
    
    0x00,0xF0,0x08,0x88,0x70,0x00,0x00,0x00,
    0x1E,0x21,0x23,0x24,0x19,0x27,0x21,0x10,//& 6
    
    0x10,0x16,0x0E,0x00,0x00,0x00,0x00,0x00,
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,//' 7
    
    0x00,0x00,0x00,0xE0,0x18,0x04,0x02,0x00,
    0x00,0x00,0x00,0x07,0x18,0x20,0x40,0x00,//( 8
    
    0x00,0x02,0x04,0x18,0xE0,0x00,0x00,0x00,
    0x00,0x40,0x20,0x18,0x07,0x00,0x00,0x00,//) 9
    
    0x40,0x40,0x80,0xF0,0x80,0x40,0x40,0x00,
    0x02,0x02,0x01,0x0F,0x01,0x02,0x02,0x00,//* 10
    
    0x00,0x00,0x00,0xF0,0x00,0x00,0x00,0x00,
    0x01,0x01,0x01,0x1F,0x01,0x01,0x01,0x00,//+ 11
    
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
    0x80,0xB0,0x70,0x00,0x00,0x00,0x00,0x00,//, 12
    
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
    0x00,0x01,0x01,0x01,0x01,0x01,0x01,0x01,//- 13
    
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
    0x00,0x30,0x30,0x00,0x00,0x00,0x00,0x00,//. 14
    
    0x00,0x00,0x00,0x00,0x80,0x60,0x18,0x04,
    0x00,0x60,0x18,0x06,0x01,0x00,0x00,0x00,/// 15
    
    0x00,0xE0,0x10,0x08,0x08,0x10,0xE0,0x00,
    0x00,0x0F,0x10,0x20,0x20,0x10,0x0F,0x00,//0 16
    
    0x00,0x10,0x10,0xF8,0x00,0x00,0x00,0x00,
    0x00,0x20,0x20,0x3F,0x20,0x20,0x00,0x00,//1 17
    
    0x00,0x70,0x08,0x08,0x08,0x88,0x70,0x00,
    0x00,0x30,0x28,0x24,0x22,0x21,0x30,0x00,//2 18
    
    0x00,0x30,0x08,0x88,0x88,0x48,0x30,0x00,
    0x00,0x18,0x20,0x20,0x20,0x11,0x0E,0x00,//3 19
    
    0x00,0x00,0xC0,0x20,0x10,0xF8,0x00,0x00,
    0x00,0x07,0x04,0x24,0x24,0x3F,0x24,0x00,//4 20
    
    0x00,0xF8,0x08,0x88,0x88,0x08,0x08,0x00,
    0x00,0x19,0x21,0x20,0x20,0x11,0x0E,0x00,//5 21
    
    0x00,0xE0,0x10,0x88,0x88,0x18,0x00,0x00,
    0x00,0x0F,0x11,0x20,0x20,0x11,0x0E,0x00,//6 22
    
    0x00,0x38,0x08,0x08,0xC8,0x38,0x08,0x00,
    0x00,0x00,0x00,0x3F,0x00,0x00,0x00,0x00,//7 23
    
    0x00,0x70,0x88,0x08,0x08,0x88,0x70,0x00,
    0x00,0x1C,0x22,0x21,0x21,0x22,0x1C,0x00,//8 24
    
    0x00,0xE0,0x10,0x08,0x08,0x10,0xE0,0x00,
    0x00,0x00,0x31,0x22,0x22,0x11,0x0F,0x00,//9 25
    
    0x00,0x00,0x00,0xC0,0xC0,0x00,0x00,0x00,
    0x00,0x00,0x00,0x30,0x30,0x00,0x00,0x00,//: 26
    
    0x00,0x00,0x00,0x80,0x00,0x00,0x00,0x00,
    0x00,0x00,0x80,0x60,0x00,0x00,0x00,0x00,//; 27
    
    0x00,0x00,0x80,0x40,0x20,0x10,0x08,0x00,
    0x00,0x01,0x02,0x04,0x08,0x10,0x20,0x00,//< 28
    
    0x40,0x40,0x40,0x40,0x40,0x40,0x40,0x00,
    0x04,0x04,0x04,0x04,0x04,0x04,0x04,0x00,//= 29
    
    0x00,0x08,0x10,0x20,0x40,0x80,0x00,0x00,
    0x00,0x20,0x10,0x08,0x04,0x02,0x01,0x00,//> 30
    
    0x00,0x70,0x48,0x08,0x08,0x08,0xF0,0x00,
    0x00,0x00,0x00,0x30,0x36,0x01,0x00,0x00,//? 31
    
    0xC0,0x30,0xC8,0x28,0xE8,0x10,0xE0,0x00,
    0x07,0x18,0x27,0x24,0x23,0x14,0x0B,0x00,//@ 32
    
    0x00,0x00,0xC0,0x38,0xE0,0x00,0x00,0x00,
    0x20,0x3C,0x23,0x02,0x02,0x27,0x38,0x20,//A 33
    
    0x08,0xF8,0x88,0x88,0x88,0x70,0x00,0x00,
    0x20,0x3F,0x20,0x20,0x20,0x11,0x0E,0x00,//B 34
    
    0xC0,0x30,0x08,0x08,0x08,0x08,0x38,0x00,
    0x07,0x18,0x20,0x20,0x20,0x10,0x08,0x00,//C 35
    
    0x08,0xF8,0x08,0x08,0x08,0x10,0xE0,0x00,
    0x20,0x3F,0x20,0x20,0x20,0x10,0x0F,0x00,//D 36
    
    0x08,0xF8,0x88,0x88,0xE8,0x08,0x10,0x00,
    0x20,0x3F,0x20,0x20,0x23,0x20,0x18,0x00,//E 37
    
    0x08,0xF8,0x88,0x88,0xE8,0x08,0x10,0x00,
    0x20,0x3F,0x20,0x00,0x03,0x00,0x00,0x00,//F 38
    
    0xC0,0x30,0x08,0x08,0x08,0x38,0x00,0x00,
    0x07,0x18,0x20,0x20,0x22,0x1E,0x02,0x00,//G 39
    
    0x08,0xF8,0x08,0x00,0x00,0x08,0xF8,0x08,
    0x20,0x3F,0x21,0x01,0x01,0x21,0x3F,0x20,//H 40
    
    0x00,0x08,0x08,0xF8,0x08,0x08,0x00,0x00,
    0x00,0x20,0x20,0x3F,0x20,0x20,0x00,0x00,//I 41
    
    0x00,0x00,0x08,0x08,0xF8,0x08,0x08,0x00,
    0xC0,0x80,0x80,0x80,0x7F,0x00,0x00,0x00,//J 42
    
    0x08,0xF8,0x88,0xC0,0x28,0x18,0x08,0x00,
    0x20,0x3F,0x20,0x01,0x26,0x38,0x20,0x00,//K 43
    
    0x08,0xF8,0x08,0x00,0x00,0x00,0x00,0x00,
    0x20,0x3F,0x20,0x20,0x20,0x20,0x30,0x00,//L 44
    
    0x08,0xF8,0xF8,0x00,0xF8,0xF8,0x08,0x00,
    0x20,0x3F,0x00,0x3F,0x00,0x3F,0x20,0x00,//M 45
    
    0x08,0xF8,0x30,0xC0,0x00,0x08,0xF8,0x08,
    0x20,0x3F,0x20,0x00,0x07,0x18,0x3F,0x00,//N 46
    
    0xE0,0x10,0x08,0x08,0x08,0x10,0xE0,0x00,
    0x0F,0x10,0x20,0x20,0x20,0x10,0x0F,0x00,//O 47
    
    0x08,0xF8,0x08,0x08,0x08,0x08,0xF0,0x00,
    0x20,0x3F,0x21,0x01,0x01,0x01,0x00,0x00,//P 48
    
    0xE0,0x10,0x08,0x08,0x08,0x10,0xE0,0x00,
    0x0F,0x18,0x24,0x24,0x38,0x50,0x4F,0x00,//Q 49
    
    0x08,0xF8,0x88,0x88,0x88,0x88,0x70,0x00,
    0x20,0x3F,0x20,0x00,0x03,0x0C,0x30,0x20,//R 50
    
    0x00,0x70,0x88,0x08,0x08,0x08,0x38,0x00,
    0x00,0x38,0x20,0x21,0x21,0x22,0x1C,0x00,//S 51
    
    0x18,0x08,0x08,0xF8,0x08,0x08,0x18,0x00,
    0x00,0x00,0x20,0x3F,0x20,0x00,0x00,0x00,//T 52
    
    0x08,0xF8,0x08,0x00,0x00,0x08,0xF8,0x08,
    0x00,0x1F,0x20,0x20,0x20,0x20,0x1F,0x00,//U 53
    
    0x08,0x78,0x88,0x00,0x00,0xC8,0x38,0x08,
    0x00,0x00,0x07,0x38,0x0E,0x01,0x00,0x00,//V 54
    
    0xF8,0x08,0x00,0xF8,0x00,0x08,0xF8,0x00,
    0x03,0x3C,0x07,0x00,0x07,0x3C,0x03,0x00,//W 55
    
    0x08,0x18,0x68,0x80,0x80,0x68,0x18,0x08,
    0x20,0x30,0x2C,0x03,0x03,0x2C,0x30,0x20,//X 56
    
    0x08,0x38,0xC8,0x00,0xC8,0x38,0x08,0x00,
    0x00,0x00,0x20,0x3F,0x20,0x00,0x00,0x00,//Y 57
    
    0x10,0x08,0x08,0x08,0xC8,0x38,0x08,0x00,
    0x20,0x38,0x26,0x21,0x20,0x20,0x18,0x00,//Z 58
    
    0x00,0x00,0x00,0xFE,0x02,0x02,0x02,0x00,
    0x00,0x00,0x00,0x7F,0x40,0x40,0x40,0x00,//[ 59
    
    0x00,0x0C,0x30,0xC0,0x00,0x00,0x00,0x00,
    0x00,0x00,0x00,0x01,0x06,0x38,0xC0,0x00,//\ 60
    
    0x00,0x02,0x02,0x02,0xFE,0x00,0x00,0x00,
    0x00,0x40,0x40,0x40,0x7F,0x00,0x00,0x00,//] 61
    
    0x00,0x00,0x04,0x02,0x02,0x02,0x04,0x00,
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,//^ 62
    
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
    0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,//_ 63
    
    0x00,0x02,0x02,0x04,0x00,0x00,0x00,0x00,
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,//` 64
    
    0x00,0x00,0x80,0x80,0x80,0x80,0x00,0x00,
    0x00,0x19,0x24,0x22,0x22,0x22,0x3F,0x20,//a 65
    
    0x08,0xF8,0x00,0x80,0x80,0x00,0x00,0x00,
    0x00,0x3F,0x11,0x20,0x20,0x11,0x0E,0x00,//b 66
    
    0x00,0x00,0x00,0x80,0x80,0x80,0x00,0x00,
    0x00,0x0E,0x11,0x20,0x20,0x20,0x11,0x00,//c 67
    
    0x00,0x00,0x00,0x80,0x80,0x88,0xF8,0x00,
    0x00,0x0E,0x11,0x20,0x20,0x10,0x3F,0x20,//d 68
    
    0x00,0x00,0x80,0x80,0x80,0x80,0x00,0x00,
    0x00,0x1F,0x22,0x22,0x22,0x22,0x13,0x00,//e 69
    
    0x00,0x80,0x80,0xF0,0x88,0x88,0x88,0x18,
    0x00,0x20,0x20,0x3F,0x20,0x20,0x00,0x00,//f 70
    
    0x00,0x00,0x80,0x80,0x80,0x80,0x80,0x00,
    0x00,0x6B,0x94,0x94,0x94,0x93,0x60,0x00,//g 71
    
    0x08,0xF8,0x00,0x80,0x80,0x80,0x00,0x00,
    0x20,0x3F,0x21,0x00,0x00,0x20,0x3F,0x20,//h 72
    
    0x00,0x80,0x98,0x98,0x00,0x00,0x00,0x00,
    0x00,0x20,0x20,0x3F,0x20,0x20,0x00,0x00,//i 73
    
    0x00,0x00,0x00,0x80,0x98,0x98,0x00,0x00,
    0x00,0xC0,0x80,0x80,0x80,0x7F,0x00,0x00,//j 74
    
    0x08,0xF8,0x00,0x00,0x80,0x80,0x80,0x00,
    0x20,0x3F,0x24,0x02,0x2D,0x30,0x20,0x00,//k 75
    
    0x00,0x08,0x08,0xF8,0x00,0x00,0x00,0x00,
    0x00,0x20,0x20,0x3F,0x20,0x20,0x00,0x00,//l 76
    
    0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x00,
    0x20,0x3F,0x20,0x00,0x3F,0x20,0x00,0x3F,//m 77
    
    0x80,0x80,0x00,0x80,0x80,0x80,0x00,0x00,
    0x20,0x3F,0x21,0x00,0x00,0x20,0x3F,0x20,//n 78
    
    0x00,0x00,0x80,0x80,0x80,0x80,0x00,0x00,
    0x00,0x1F,0x20,0x20,0x20,0x20,0x1F,0x00,//o 79
    
    0x80,0x80,0x00,0x80,0x80,0x00,0x00,0x00,
    0x80,0xFF,0xA1,0x20,0x20,0x11,0x0E,0x00,//p 80
    
    0x00,0x00,0x00,0x80,0x80,0x80,0x80,0x00,
    0x00,0x0E,0x11,0x20,0x20,0xA0,0xFF,0x80,//q 81
    
    0x80,0x80,0x80,0x00,0x80,0x80,0x80,0x00,
    0x20,0x20,0x3F,0x21,0x20,0x00,0x01,0x00,//r 82
    
    0x00,0x00,0x80,0x80,0x80,0x80,0x80,0x00,
    0x00,0x33,0x24,0x24,0x24,0x24,0x19,0x00,//s 83
    
    0x00,0x80,0x80,0xE0,0x80,0x80,0x00,0x00,
    0x00,0x00,0x00,0x1F,0x20,0x20,0x00,0x00,//t 84
    
    0x80,0x80,0x00,0x00,0x00,0x80,0x80,0x00,
    0x00,0x1F,0x20,0x20,0x20,0x10,0x3F,0x20,//u 85
    
    0x80,0x80,0x80,0x00,0x00,0x80,0x80,0x80,
    0x00,0x01,0x0E,0x30,0x08,0x06,0x01,0x00,//v 86
    
    0x80,0x80,0x00,0x80,0x00,0x80,0x80,0x80,
    0x0F,0x30,0x0C,0x03,0x0C,0x30,0x0F,0x00,//w 87
    
    0x00,0x80,0x80,0x00,0x80,0x80,0x80,0x00,
    0x00,0x20,0x31,0x2E,0x0E,0x31,0x20,0x00,//x 88
    
    0x80,0x80,0x80,0x00,0x00,0x80,0x80,0x80,
    0x80,0x81,0x8E,0x70,0x18,0x06,0x01,0x00,//y 89
    
    0x00,0x80,0x80,0x80,0x80,0x80,0x80,0x00,
    0x00,0x21,0x30,0x2C,0x22,0x21,0x30,0x00,//z 90
    
    0x00,0x00,0x00,0x00,0x80,0x7C,0x02,0x02,
    0x00,0x00,0x00,0x00,0x00,0x3F,0x40,0x40,//{ 91
    
    0x00,0x00,0x00,0x00,0xFF,0x00,0x00,0x00,
    0x00,0x00,0x00,0x00,0xFF,0x00,0x00,0x00,//| 92
    
    0x00,0x02,0x02,0x7C,0x80,0x00,0x00,0x00,
    0x00,0x40,0x40,0x3F,0x00,0x00,0x00,0x00,//} 93
    
    0x00,0x06,0x01,0x01,0x02,0x02,0x04,0x04,
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,//~ 94
};

#endif
  1. 舵机(注意5V供电!!!橙色接PWM,红色接5V,黑色接GND)

我使用的舵机模块是SG90,并使用stm32的TIM3的通道1输出PWM信号来控制旋转角度和方向,具体请详见代码和注释。


/*这里用TIM3的通道1来产生PWM信号调速舵机的旋转角度*/
void PWM_TIM3_Init(void)/*这个是TIM3的配置,和上面的说明一样*/
{
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,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_6;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA,&GPIO_InitStructure);
    
    TIM_InternalClockConfig(TIM3);
    
    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
    TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
    TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
    TIM_TimeBaseInitStructure.TIM_Period = 20000 - 1;    //ARR
    TIM_TimeBaseInitStructure.TIM_Prescaler = 72 - 1;    //PSC
    TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;
    TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);
    
    TIM_OCInitTypeDef TIM_OCInitStructure;
    TIM_OCStructInit(&TIM_OCInitStructure);
    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCNPolarity_High;
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
    TIM_OCInitStructure.TIM_Pulse = 0;    //CCR的值
    
    TIM_OC1Init(TIM3,&TIM_OCInitStructure);
    
    TIM_Cmd(TIM3,ENABLE);
}

/*封装TIM的通道的调用函数,不封装也可以,看自己情况*/
void PWM_TIM3_SetCompare1(uint16_t Compare)    //TIM3 的通道1
{
    TIM_SetCompare1(TIM3,Compare);
}
  1. L298N(我在关于PWM的讲解有介绍)

这里我放上小车的两路电机(4个电机,因为我采用并联方法,使一OUT1/OUT2,OUT3/OUT4分别驱动两个电机)所用到的PWM调速的文件。其中使用定时器TIM2的通道1和通道2


void PWM_TIM2_Init(void)/*配置TIM用来输出PWM波形时,要参考使用手册,因为它的GPIO口是官方确定的*/
{
    /*打开TIM和GPIO的时钟*/
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);    
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
    /*配置GPIO的输出模式、GPIO口、频率*/    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;/*复用推挽输出*/
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA,&GPIO_InitStructure);/*GPIO初始化*/
    /**/
    TIM_InternalClockConfig(TIM2);
    /*这里通过ARR,PSC,CCR的值来计算计时频率和占空比的值*/
    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
    TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
    TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
    TIM_TimeBaseInitStructure.TIM_Period = 100 - 1;    //ARR
    TIM_TimeBaseInitStructure.TIM_Prescaler = 720 - 1;    //PSC
    TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;/*确定占空比*/
    TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStructure);
    /*这里配置输出比较单元*/
    TIM_OCInitTypeDef TIM_OCInitStructure;
    TIM_OCStructInit(&TIM_OCInitStructure);
    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCNPolarity_High;
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
    TIM_OCInitStructure.TIM_Pulse = 0;    //CCR的值
    /*输出PWM波形的GPIO口的初始化*/
    TIM_OC1Init(TIM2,&TIM_OCInitStructure);    /*非常重要且容易忘记,配置好PWM的gpio口后一定要记得初始化*/
    TIM_OC2Init(TIM2,&TIM_OCInitStructure);
    /*使能TIM2*/
    TIM_Cmd(TIM2,ENABLE);
}

/*封装TIM的通道的调用函数,不封装也可以,看自己情况*/
void PWM_TIM2_SetCompare1(uint16_t Compare)    //TIM2 的通道1
{
    TIM_SetCompare1(TIM2,Compare);/*很重要,改变占空比参数就靠这个函数*/
}

void PWM_TIM2_SetCompare2(uint16_t Compare)    //TIM2 的通道2
{
    TIM_SetCompare2(TIM2,Compare);
}
  1. 电源(12V锂电池组,可充电)

  1. 多路输出的电源模块(稳压模块)

由于电源是12V,但是STM32最小系统板供电是3.3V,而且舵机和超声波是5V供电,所以我觉得这个模块是必不可少的东西。

五、通用功能的函数文件

  1. 延时函数


#include "stm32f10x.h"

/**
  * @brief  微秒级延时
  * @param  xus 延时时长,范围:0~233015
  * @retval 无
  */
void Delay_us(uint32_t xus)
{
    SysTick->LOAD = 72 * xus;                //设置定时器重装值
    SysTick->VAL = 0x00;                    //清空当前计数值
    SysTick->CTRL = 0x00000005;                //设置时钟源为HCLK,启动定时器
    while(!(SysTick->CTRL & 0x00010000));    //等待计数到0
    SysTick->CTRL = 0x00000004;                //关闭定时器
}

/**
  * @brief  毫秒级延时
  * @param  xms 延时时长,范围:0~4294967295
  * @retval 无
  */
void Delay_ms(uint32_t xms)
{
    while(xms--)
    {
        Delay_us(1000);
    }
}
 
/**
  * @brief  秒级延时
  * @param  xs 延时时长,范围:0~4294967295
  * @retval 无
  */
void Delay_s(uint32_t xs)
{
    while(xs--)
    {
        Delay_ms(1000);
    }
} 
  1. 直流电机函数


#include "stm32f10x.h"                  // Device header
#include "Motor.h"
#include "PWM.h"


void Motor_R_Init(void)
{
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA,&GPIO_InitStructure);
    
    PWM_TIM2_Init();
}

void Motor_L_Init(void)
{
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4|GPIO_Pin_5;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA,&GPIO_InitStructure);
    
    PWM_TIM2_Init();
}

void Motor_R_SetSpeed(int8_t Speed)
{
    if(Speed>=0)
    {
        GPIO_SetBits(GPIOA,GPIO_Pin_2);
        GPIO_ResetBits(GPIOA,GPIO_Pin_3);
        PWM_TIM2_SetCompare1(Speed);
    }
    else
    {
        GPIO_SetBits(GPIOA,GPIO_Pin_3);
        GPIO_ResetBits(GPIOA,GPIO_Pin_2);
        PWM_TIM2_SetCompare1(-Speed);
    }
}
void Motor_L_SetSpeed(int8_t Speed)
{
    if(Speed>=0)
    {
        GPIO_SetBits(GPIOA,GPIO_Pin_4);
        GPIO_ResetBits(GPIOA,GPIO_Pin_5);
        PWM_TIM2_SetCompare2(Speed);
    }
    else
    {
        GPIO_SetBits(GPIOA,GPIO_Pin_5);
        GPIO_ResetBits(GPIOA,GPIO_Pin_4);
        PWM_TIM2_SetCompare2(-Speed);
    }
}
  1. 小车运动函数


#include "stm32f10x.h"                  // Device header
#include "Car.h"
#include "Motor.h"

void Car_Init(void)
{
    Motor_R_Init();
    Motor_L_Init();
}

void Car_Forword(uint8_t Speed)
{
    Motor_R_SetSpeed(Speed);
    Motor_L_SetSpeed(Speed);
}

void Car_Back(uint8_t Speed)
{
    Motor_R_SetSpeed(-Speed);
    Motor_L_SetSpeed(-Speed);
}

void Car_Left(uint8_t Speed)
{
    Motor_R_SetSpeed(Speed);
    Motor_L_SetSpeed(-Speed);
}

void Car_Right(uint8_t Speed)
{
    Motor_R_SetSpeed(-Speed);
    Motor_L_SetSpeed(Speed);
}

void Car_Stop(void)
{
    Motor_R_SetSpeed(0);
    Motor_L_SetSpeed(0);
}
  1. PWM调速(我有对PWM专门的讲解文章,不懂的话可以去看看关于PWM的讲解


#include "stm32f10x.h"                  // Device header
#include "PWM.h"  

void PWM_TIM2_Init(void)/*配置TIM用来输出PWM波形时,要参考使用手册,因为它的GPIO口是官方确定的*/
{
    /*打开TIM和GPIO的时钟*/
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);    
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
    /*配置GPIO的输出模式、GPIO口、频率*/    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;/*复用推挽输出*/
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA,&GPIO_InitStructure);/*GPIO初始化*/
    /**/
    TIM_InternalClockConfig(TIM2);
    /*这里通过ARR,PSC,CCR的值来计算计时频率和占空比的值*/
    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
    TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
    TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
    TIM_TimeBaseInitStructure.TIM_Period = 100 - 1;    //ARR
    TIM_TimeBaseInitStructure.TIM_Prescaler = 720 - 1;    //PSC
    TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;/*确定占空比*/
    TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStructure);
    /*这里配置输出比较单元*/
    TIM_OCInitTypeDef TIM_OCInitStructure;
    TIM_OCStructInit(&TIM_OCInitStructure);
    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCNPolarity_High;
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
    TIM_OCInitStructure.TIM_Pulse = 0;    //CCR的值
    /*输出PWM波形的GPIO口的初始化*/
    TIM_OC1Init(TIM2,&TIM_OCInitStructure);    /*非常重要且容易忘记,配置好PWM的gpio口后一定要记得初始化*/
    TIM_OC2Init(TIM2,&TIM_OCInitStructure);
    /*使能TIM2*/
    TIM_Cmd(TIM2,ENABLE);
}


/*这里用TIM3的通道1来产生PWM信号调速舵机的旋转角度*/
void PWM_TIM3_Init(void)/*这个是TIM3的配置,和上面的说明一样*/
{
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,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_6;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA,&GPIO_InitStructure);
    
    TIM_InternalClockConfig(TIM3);
    
    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
    TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
    TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
    TIM_TimeBaseInitStructure.TIM_Period = 20000 - 1;    //ARR
    TIM_TimeBaseInitStructure.TIM_Prescaler = 72 - 1;    //PSC
    TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;
    TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);
    
    TIM_OCInitTypeDef TIM_OCInitStructure;
    TIM_OCStructInit(&TIM_OCInitStructure);
    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCNPolarity_High;
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
    TIM_OCInitStructure.TIM_Pulse = 0;    //CCR的值
    
    TIM_OC1Init(TIM3,&TIM_OCInitStructure);
    
    TIM_Cmd(TIM3,ENABLE);
}

/*封装TIM的通道的调用函数,不封装也可以,看自己情况*/
void PWM_TIM2_SetCompare1(uint16_t Compare)    //TIM2 的通道1
{
    TIM_SetCompare1(TIM2,Compare);/*很重要,改变占空比参数就靠这个函数*/
}

void PWM_TIM2_SetCompare2(uint16_t Compare)    //TIM2 的通道2
{
    TIM_SetCompare2(TIM2,Compare);
}

void PWM_TIM3_SetCompare1(uint16_t Compare)    //TIM3 的通道1
{
    TIM_SetCompare1(TIM3,Compare);
}
  1. 按键函数


#include "stm32f10x.h"                  // Device header
#include "Key.h"
#include "Delay.h"


void Key_Init(void)
{
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3 | GPIO_Pin_5;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOB,&GPIO_InitStructure);
}

uint8_t Key_GetNum(void)
{
    uint8_t KeyNum = 0;
    if(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_3) == 0)
    {
        Delay_ms(20);
        while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_3) == 0);
        Delay_ms(20);
        KeyNum=1;
    }
    if(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_5) == 0)
    {
        Delay_ms(20);
        while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_5) == 0);
        Delay_ms(20);
        KeyNum=2;
    }
    return KeyNum;
}
  1. 蜂鸣器


#include "stm32f10x.h"                  // Device header
#include "Buzzer.h"
#include "Delay.h"

void Buzzer_Init(void)
{
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOB,&GPIO_InitStructure);
}

void Buzzer_OFF(void) //关闭蜂鸣器
{
    Buzzer_Init();
    GPIO_SetBits(GPIOB,GPIO_Pin_11);    //GPIOB设置为高电平
}

void Buzzer_ON1(void)
{
    Buzzer_Init();
    GPIO_ResetBits(GPIOB,GPIO_Pin_11);    //GPIOB设置为d电平
}

void Buzzer_ON2(void) //打开蜂鸣器
{
    Buzzer_Init();
    GPIO_ResetBits(GPIOB,GPIO_Pin_11);    //GPIOB设置为低电平
    Delay_ms(100);
    GPIO_SetBits(GPIOB,GPIO_Pin_11);    //GPIOB设置为高电平
    Delay_ms(100);
    
    GPIO_ResetBits(GPIOB,GPIO_Pin_11);    //GPIOB设置为低电平
    Delay_ms(100);
    GPIO_SetBits(GPIOB,GPIO_Pin_11);    //GPIOB设置为高电平
    Delay_ms(700);

}
  1. 提示灯


#include "stm32f10x.h"                  // Device header
#include "LED.h"
#include "Delay.h"

void LED_Init(void)
{
    //第一步,使能GPIO的时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
    //第二步,定义GPIO的初始化结构体类型
    GPIO_InitTypeDef GPIO_InitStructure;
    //第三步,配置GPIO的输出类型为推挽输出
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    //第四步,配置GPIO的引脚
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10 | GPIO_Pin_1;
    //第五步,配置GPIO的输出速度
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    //第六步,初始化GPIOx的寄存器
    GPIO_Init(GPIOB,&GPIO_InitStructure);
    
//    GPIO_ResetBits(GPIOA,GPIO_Pin_0);    //GPIOA设置为低电平,LED点亮
//    GPIO_SetBits(GPIOA,GPIO_Pin_0);        //GPIOA设置为高电平,LED灭
//    GPIO_WriteBit(GPIOA,GPIO_Pin_0,Bit_RESET);    //LED亮
//    GPIO_WriteBit(GPIOA,GPIO_Pin_0,Bit_SET);    //LED灭
}

void LED_ON(void)
{
    LED_Init();
    GPIO_WriteBit(GPIOB,GPIO_Pin_1,Bit_RESET);    //LED亮
}

void LED_OFF(void)
{
    LED_Init();
    GPIO_WriteBit(GPIOB,GPIO_Pin_1,Bit_SET);    //LED灭
    Delay_ms(500);
}

void LED_Flash_ON(void)
{
    LED_Init();
    GPIO_WriteBit(GPIOB,GPIO_Pin_10,Bit_RESET);    //LED亮
    Delay_ms(500);
    GPIO_WriteBit(GPIOB,GPIO_Pin_10,Bit_SET);    //LED灭
    Delay_ms(500);
}

void LED_Flash_OFF(void)
{
    LED_Init();
    GPIO_WriteBit(GPIOB,GPIO_Pin_10,Bit_RESET);    //LED灭
}

六、主函数

  1. main.c


#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "Motor.h"
#include "PWM.h"  
#include "LED.h"
#include "Car.h"
#include "SG90.h"
#include "Ultrasonic.h"
#include "Timer.h"
#include "Buzzer.h"
#include "Key.h"
#include "Serial.h"
#include "Hc05.h"

uint8_t Speed = 50;  /*速度范围为[0,100]*/
uint8_t KeyNum;
uint8_t Mode=0;
//extern uint8_t RxData;

int main(void)
{
    OLED_Init();
    Car_Init();
    Key_Init();
    Serial_Init();
    SG90_Servo_Init();
    Ultrasonic_GPIO_Init();
    
    OLED_ShowString(1,1,"Dist:");
    OLED_ShowString(2,1,"Speed:");
    OLED_ShowString(3,1,"MOde:");
    
    while(1)
    {
        Mode = Key_GetNum();
        OLED_ShowNum(3,6,Mode,3);
        if(Mode)
        {
            switch(Mode)
            {
                case 1: Hc05_car();break;
                case 2: Ultrasonic_car();break;
            }
        }  
    }
}

七、电路连接表述

电路连接的时候一定要细心,防止出现短路的情况。接下来,我会描述一下我的电路连接思路,供大家参考一下。

首先从电源出发,电源正负极分别接到L298N的12V和GND位置上,这样就可以给L298N供电,使其驱动4个直流电机。再把12V的电源正负极接到稳压模块上,把电压降到5V和3.3V,这样就可以用3.3V来给stm32最小系统板供电,5V来给舵机和超声波以及蓝牙模块供电。

接4个电机时采用并联的方法,这里不太好描述,上个图片一目了然!!

像这样,应该很容易理解吧!

最后就是操作步骤了:连接好电路,打开开关,此时OLED显示屏上的第三行内容是Mode:000,这时如果按下B5引脚上的按键,Mode;002,这时候就开始了超声波避障模式。如果按下B3引脚上的按键,Mode;001,这时候就是蓝牙控制模式,打开手机连接蓝牙,操作自己设置的键盘就可以控制小车了。(补充;OLED显示屏第一行内容是Dist: ,它显示的是超声波模式下探测到的距离的实时值。第二行内容是Speed: ,它显示的是小车的速度值)

### 构建基于STM32超声波测距小车项目的教程 #### 一、硬件连接 为了实现基于STM32超声波测距小车项目,需要完成HC-SR04超声波模块与STM32微控制器之间的接线工作。以下是具体的硬件连接方法: | **HC-SR04引脚** | **功能描述** | **STM32对应引脚** | |------------------|----------------------|--------------------| | VCC | 提供电源 | STM32的3.3V或5V供电引脚[^1] | | GND | 地 | STM32的地(GND) | | TRIG | 触发信号输入 | STM32的一个GPIO口 (需设置为推挽输出模式)[^3] | | ECHO | 返回信号输出 | STM32的一个GPIO口 (需配置为外部中断或定时器捕获模式)[^3] | 注意,在实际操作中,TRIGECHO可以分别连接到不同的GPIO端口上,具体取决于所使用的开发板及其外设资源分配。 --- #### 二、软件设计概述 整个系统的软件部分主要分为以下几个方面: 1. 初始化GPIO接口用于控制触发信号(TRIG),并读取返回信号(ECHO)。 2. 配置定时器(Timer)或者使用HAL库中的`HAL_Delay()`函数生成精确的时间延迟以满足HC-SR04的工作需求。 3. 编写算法计算从发出超声波至收到回响之间经历的时间间隔,并据此换算成物理距离。 4. 结合蓝牙通信协议栈(如果涉及),将检测结果发送给远程设备以便进一步分析处理。 下面给出一段简化版的核心C语言代码示例来展示如何利用上述原理进行基本的距离测算过程: ```c #include "sr04.h" void SR04_Init() { // 初始化TRIG/ECHO对应的GPIO管脚 } uint32_t Get_Distance_cm() { uint32_t duration; // 发送10us高电平脉冲激活传感器 TRIG_H; delay_us(10); TRIG_L; // 等待直到Echo变为低电平时停止计时 while(HAL_GPIO_ReadPin(Echo_GPIO_Port, Echo_Pin)==RESET); TIM_StartCounter(TIMx); // 开始计数 while(HAL_GPIO_ReadPin(Echo_GPIO_Port, Echo_Pin)!=RESET); duration = TIM_GetCounterValue(TIMx)*1e-6; return ((float)(duration/58));// 计算厘米单位下的距离值 } ``` 此段伪代码片段仅作为参考用途,请依据实际情况调整参数设定以及调用的具体API名称等细节内容[^2]。 --- #### 三、常见问题排查指南 当遇到无法正常获取有效数值的情况时,可以从以下几个角度出发寻找原因所在: - 检查连线是否存在虚焊现象; - 确认是否正确初始化了相关外设寄存器; - 查阅官方手册核实当前选用型号支持的最大频率范围是否匹配预期应用场合的要求; ---
评论 35
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值