今天写一下按键加上蜂鸣器的相关知识,emmm,学到这里,就觉得32和51其实是很相像的,底层思想也差不多一样。51的底层再加上一个初始化函数,就构成了32的底层,而初始化函数部分的方法和步骤也是大同小异,所以大家如果学过51的话,32也会很好入门的~~~
老规矩,我们还是先看一下按键部分的原理图:
然后在原理图上找一下对应的引脚:
初始化按键的思想和LED配置部分差不多,而且比LED的配置简单:配置端口时钟(时钟使能GPIOA和GPIOB),设置引脚号,配置端口模式。
这里说明一点:我们知道配置LED时是需要设置引脚速率的,那么初始化按键时为什么不需要设置引脚速率了呢?我们需要知道这里的引脚速率是输出速率,但是不是IO口输出信号的速率而是IO驱动电路的响应速度,这个速率是只有存在信号输出时才需要设置的,按键属于一个向单片机输入信号的模块,按下按键,将对应引脚拉低,相当于输入一个低电平的信号,所以这里是不需要设置引脚速率的。
话不多说,直接上代码:
void KeyInit(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_8;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//设置端口模式为浮空输入
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_2;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOB, &GPIO_InitStructure);
}
LED的配置那篇博客里已经做了详细的说明,这里就不过多陈述了,不懂的话可以去看看我的那篇博客,嘻嘻~
初始化部分写完之后,再加上51按键部分的底层就可以了(下面给出完整的代码):
#include "stm32f10x.h"
#include "key.h"
u8 KeySta[4] = {1, 1, 1, 1};//存放按键的当前状态
extern void KeyAction(u8 keycode);
void KeyInit(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB, ENABLE);//
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_8;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_2;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOB, &GPIO_InitStructure);
}
void KeyScan()
{
u8 i;
static u8 keybuff[4] = {0xFF, 0xFF, 0xFF, 0xFF};//注意这里的静态变量
keybuff[0] = (keybuff[0] << 1) | KEY1;
keybuff[1] = (keybuff[1] << 1) | KEY2;
keybuff[2] = (keybuff[2] << 1) | KEY3;
keybuff[3] = (keybuff[3] << 1) | KEY4;
for(i = 0; i < 4; i++)
{
if(keybuff[i] == 0x00)
{
KeySta[i] = 0;
}
else if(keybuff[i] == 0xFF)
{
KeySta[i] = 1;
}
else
{}
}
}
void KeyDriver()
{
u8 i;
static u8 backup[4] = {1, 1, 1, 1};
for(i = 0; i < 4; i++)
{
if(KeySta[i] != backup[i])
{
if(backup[i] == 1)
{
KeyAction(i+1);
}
backup[i] = KeySta[i];
}
}
}
按键动作函数(KeyAction):
void KeyAction(int code)
{
if(code == 1)//按下按键
{
GPIOD->ODR |= (1 << 2);
GPIOC->ODR = 0xFE00;//开启LED1
GPIOD->ODR &= ~(1 << 2);
}
}
按键引脚定义(相当于51的sbit管脚定义):
#define KEY1 GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0);
#define KEY2 GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_8);
#define KEY3 GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_1);
#define KEY4 GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_2);
原理图:
从原理图上可以看出:N_Buz为高电平时,蜂鸣器关闭;N_Buz为低电平时,蜂鸣器开启
解释一下为什么是高电平关,低电平开: 这里涉及到了三极管的开关特性:当b、e之间的饱和管压降大于等于0.7v时,三极管就处于导通状态(e、c之间是导通的),相反,e、c之间不导通(可以理解为这个三极管在电路当中是断开的);
然后我们再来看原理图,很明显:只有当N_BUZ输入为低电平时,b、e之间才会产生一个大于0.7v的饱和管压降,三极管才会导通,蜂鸣器才会响。
对应引脚:
这里需要注意!!!:PB4这个引脚是具有复用功能的引脚,也就是说它不仅可以作为普通的IO口使用,还具有别的功能,关于IO口的复用功能部分可以参照《STM32中文手册》8.3节的内容,看一下图:
stm32复位后引脚PB4默认配置为JTAG的RST引脚功能,所以我们在写初始化蜂鸣器函数时,要先把PB4设置成普通的IO口,这一过程也叫做端口重映射(通过配置重映射寄存器(AFIO_MAPR),将一些复用功能映射到其他引脚),stm32的库函数里有专门的开启重映射的函数:
这里要注意一点!!:重映射必须要使能AFIO时钟!!!就是说我们应该先使能AFIO时钟,然后再开启重映射
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_AFIO, ENABLE);//使能GPIOB和AFIO时钟
回到重映射函数,其中GPIO_Remap是GPIO端口重映射,我们要设置寄存器AFIO_MAPR的SWJ CFG[2:0]位,使PB4口重映射为普通IO口:
从表格上我们可以看到:当SWJ CFG[2:0]位为001时,JTAG使能,SWD使能,但是没有JNTRST,而且此时IO口是可以用的,也就是说把PB4重映射成了普通IO口。
我们来看一下库函数中对端口重映射的相关定义:
部分解释:
#define GPIO_Remap_SWJ_NoJTRST ((uint32_t)0x00300100) /*SWD使能、JTAG使能但是不包括JNRST引脚 */
#define GPIO_Remap_SWJ_JTAGDisable ((uint32_t)0x00300200) /*SWD使能、JTAG失能 */
#define GPIO_Remap_SWJ_Disable ((uint32_t)0x00300400) /*SWD与JTAG全部失能 */
配置步骤: 配置端口时钟(时钟使能),端口重映射、设置引脚号,设置引脚速率,配置端口模式,配置输出数据。
初始化代码:
void BeeInit(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB |RCC_APB2Periph_AFIO, ENABLE);
GPIO_PinRemapConfig(GPIO_Remap_SWJ_NoJTRST, ENABLE);//开启重映射
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;//定义管脚
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//输出速率50MHZ
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//设置输出模式为强推挽式输出
GPIO_Init(GPIOB, &GPIO_InitStructure);
// BeeOff();//设置初始化引脚状态,蜂鸣器上电时关闭
GPIO_SetBits(GPIOB, GPIO_Pin_4);//设置初始化引脚状态,蜂鸣器上电时关闭
}
这里谈谈 GPIO_SetBits()这个函数,这个函数是在stm32的库函数中定义的,可以理解为一个引脚设置函数,我们来看看这个函数的函数体:
void GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{
/* Check the parameters */
assert_param(IS_GPIO_ALL_PERIPH(GPIOx));
assert_param(IS_GPIO_PIN(GPIO_Pin));
GPIOx->BSRR = GPIO_Pin;
}
BSRR是端口位设置/清除寄存器,功能是单独设置引脚位, 该函数实现的功能我的理解是实现对应引脚置1;
相对应的还有一个GPIO_ResetBits()函数,这个函数的功能是:实现对应引脚置0, 可以看一下函数体:
void GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{
/* Check the parameters */
assert_param(IS_GPIO_ALL_PERIPH(GPIOx));
assert_param(IS_GPIO_PIN(GPIO_Pin));
GPIOx->BRR = GPIO_Pin;
}
可以看到该函数设置的寄存器是:BRR,BRR是端口位清除寄存器,功能是:单独清除引脚位
beep.h文件中我作了相应的定义:
#define BeeOff() GPIO_SetBits(GPIOB, GPIO_Pin_4);
#define BeeOn() GPIO_ResetBits(GPIOB, GPIO_Pin_4);
51底层的移植:
void BeeScan(u8 ms)//中断扫描函数,ms值与定时器定时值有关
{
if(BeepTimer > 0)
{
BeepTimer -= ms;
if(BeepTimer <= 0)
{
BeeOff();
BeepTimer = 0;
}
}
}
void Beep(s32 time)
{
BeepTimer = time;
if(BeepTimer == 0)
{
BeeOff();
}
else
{
BeeOn();
}
}