image.png这里需要注意的是:KEY0、KEY1 和 KEY2 是低电平有效的,而 KEY_UP 是高电平有效
    的,并且外部都没有上下拉电阻,所以,需要在 STM32F4 内部设置上下拉。

    利用位带操作 实现GPIO的控制
    sys.h

    1. #define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x2000000+((addr &0xFFFFF)<<5)+(bitnum<<2))
    2. #define MEM_ADDR(addr) *((volatile unsigned long *)(addr))
    3. #define BIT_ADDR(addr, bitnum) MEM_ADDR(BITBAND(addr, bitnum))
    4. #define GPIOA_ODR_Addr (GPIOA_BASE+20) //0x40020014
    5. #define GPIOE_ODR_Addr (GPIOE_BASE+20) //0x40021014
    6. #define GPIOA_IDR_Addr (GPIOA_BASE+16) //0x40020010
    7. #define GPIOE_IDR_Addr (GPIOE_BASE+16) //0x40021010
    8. #define PAout(n) BIT_ADDR(GPIOA_ODR_Addr,n) //输出
    9. #define PAin(n) BIT_ADDR(GPIOA_IDR_Addr,n) //输入
    10. #define PEout(n) BIT_ADDR(GPIOE_ODR_Addr,n) //输出
    11. #define PEin(n) BIT_ADDR(GPIOE_IDR_Addr,n) //输入

    key.h
    声明 按键初始化函数void KEY_Init() 和 按键扫描函数u8 KEY_Scan(u8) 和 宏定义一些 语句

    1. void KEY_Init(void); //IO 初始化
    2. u8 KEY_Scan(u8); //按键扫描函数
    3. #define KEY0 GPIO_ReadInputDataBit(GPIOE,GPIO_Pin_4) //PE4
    4. #define KEY1 GPIO_ReadInputDataBit(GPIOE,GPIO_Pin_3) //PE3
    5. #define KEY2 GPIO_ReadInputDataBit(GPIOE,GPIO_Pin_2) //PE2
    6. #define WK_UP GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_0) //PA0
    7. #define KEY0_PRES 1
    8. #define KEY1_PRES 2
    9. #define KEY2_PRES 3
    10. #define WKUP_PRES 4

    key.c
    编写 按键初始化函数 和 按键扫描函数

    1. void KEY_Init(void) //键盘初始化函数
    2. {
    3. GPIO_InitTypeDef GPIO_InitStructure;
    4. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA|RCC_AHB1Periph_GPIOE,ENABLE);//使能 GPIOA,GPIOE 时钟
    5. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4;
    6. //KEY0 KEY1 KEY2 对应引脚
    7. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;//普通输入模式
    8. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100M
    9. GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
    10. GPIO_Init(GPIOE, &GPIO_InitStructure);//初始化 GPIOE2,3,4
    11. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;//WK_UP 对应引脚 PA0
    12. GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN ;//下拉
    13. GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化 GPIOA0
    14. }
    15. //按键处理函数
    16. //返回按键值
    17. //mode:0,不支持连续按;1,支持连续按;
    18. //0,没有任何按键按下
    19. //1,KEY0 按下 2,KEY1 按下 3,KEY2 按下 4,WKUP 按下 WK_UP
    20. //注意此函数有响应优先级,KEY0>KEY1>KEY2>WK_UP!!
    21. u8 KEY_Scan(u8 mode)
    22. {
    23. static u8 key_up=1;//按键按松开标志
    24. if(mode)key_up=1; //支持连按
    25. if(key_up&&(KEY0==0||KEY1==0||KEY2==0||WK_UP==1))
    26. {
    27. delay_ms(10);//去抖动
    28. key_up=0;
    29. if(KEY0==0)return 1;
    30. else if(KEY1==0)return 2;
    31. else if(KEY2==0)return 3;
    32. else if(WK_UP==1)return 4;
    33. }else if(KEY0==1&&KEY1==1&&KEY2==1&&WK_UP==0)key_up=1;
    34. return 0;// 无按键按下
    35. }

    main.c

    1. #include "key.h" // 将接口驱动头文件包含进来
    2. #include "delay.h"
    3. #include "led.h"
    4. #include "beep.h"
    5. int main(void)
    6. {
    7. u8 key; //保存键值
    8. delay_init(168); //初始化延时函数
    9. LED_Init(); //初始化 LED 端口
    10. BEEP_Init(); //初始化蜂鸣器端口
    11. KEY_Init(); //初始化与按键连接的硬件接口
    12. LED0=0; //先点亮红灯
    13. while(1)
    14. {
    15. key=KEY_Scan(0); //得到键值
    16. if(key)
    17. {
    18. switch(key)
    19. {
    20. case WKUP_PRES: //控制蜂鸣器
    21. BEEP=!BEEP;
    22. break;
    23. case KEY0_PRES: //控制 LED0 翻转
    24. LED0=!LED0;
    25. break;
    26. case KEY1_PRES: //控制 LED1 翻转
    27. LED1=!LED1;
    28. break;
    29. case KEY2_PRES: //同时控制 LED0,LED1 翻转
    30. LED0=!LED0;
    31. LED1=!LED1;
    32. break;
    33. }
    34. }
    35. else delay_ms(10);
    36. }
    37. }

    效果为
    按下key_up 蜂鸣器响,再次按下,停止响
    按下key_0 DS0灭,再按亮
    按下key_1 DS1亮 再按灭
    按下key_2 DS1亮 DS0灭 再按下 DS0亮 DS1灭