STM32单片机学习3--STM32控制键盘

这篇具有很好参考价值的文章主要介绍了STM32单片机学习3--STM32控制键盘。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、前言

单片机型号:STM32F103C8T6
开发环境:Keil5

STM32单片机学习3--STM32控制键盘


二、GPIO的8种工作模式

STM32单片机学习3--STM32控制键盘

4种输入模式

  • 上拉输入模式:在默认状态下(GPIO引脚无输入),读取得的GPIO引脚数据为1,高电平(与Vdd相连的为上拉电阻);
  • 下拉输入模式:在默认状态下(GPIO引脚无输入),读取得的GPIO引脚数据为0,低电平(与Vss相连的为下拉电阻);
  • 浮空输入模式:在芯片内部既没有上拉电阻,也没有下拉电阻,经由触发器输入。配置成这个模式直接用电压表测量其引脚为1点几伏,这是个不确定值。由于其输入阻抗较大,一般把这种模式用于标准的通信协议I2C、USART的接收端;
  • 模拟输入模式:关闭了斯密特触发器,不接上、下拉电阻,经由另一线路吧电压信号传送到片上外设模块。如传送至ADC模块,由ADC采集电压信号,所以使用ADC外设的时候,必须设置为模拟输入模式;

4中输出模式

  • 推挽输出模式:线路经过一个由P-MOS管和N-MOS管组成的单元电路。在输出高电平时,P-MOS管导通;低电平时,N-MOS管导通。两个管子轮流导通,一个负责灌电流,一个负责拉电流,使其负载能力和开关速度都比普通的方式有很大的提高。普通推挽输出模式一般应用在输出电平为0和3.3伏的场合
  • 开漏输出模式:如果我们控制输出为0,低电平,则使N-MOS管导通,使输出接地;如果我们控制输出为1(无法直接输出高电平),则既不输出高电平,也不输出低电平,为高阻态;为正常使用时必须在外部接上一个上拉电阻。它具有“线与”特效,即很多个开漏模式引脚连接到一起时,只有当所有引脚都输出高阻态,才由上拉电阻提供高电平,此高电平的电压为外部上拉电阻所接电源的电压。若其中一个引脚为低电平,那线路就相当于短路解读,使得整条线路都为低电平,0伏;普通开漏输出模式一般应用在电平不匹配的场合,如需要输出5伏的高电平,就需要在外部接一个上拉电阻,电源为5伏,把GPIO设置为开漏模式,当输出高阻态时,由上拉电阻和电源向外输出5伏的电平;
  • 复用推挽输出模式:
  • 复用开漏输出模式:

对相应的复用模式,则是根据GPIO的复用功能来选择的,例如GPIO的引脚用作串口的输出,则使用复用推挽输出模式;如果用在IC、SMBUS这些需要线与功能的复合场所,就使用复用开漏输出模式;

注意:在使用任何一种开漏模式时,都需要接上拉电阻;


三、按键原理

STM32单片机学习3--STM32控制键盘
如上图,按钮一端接地,一端接GPIO引脚,这条路其实是一个断路;控制按钮时我们通常会将GPIO引脚设置为上拉输入模式,上拉输入模式默认为高电平,当按钮没有按下的时候,这个引脚读到的一直是高电平;当按钮被按下的时候,引脚会被强行拉低,此时引脚读到的为低电平,那说明按键已经被按下;


四、单个按钮

我们已经知道了按钮的工作原理,按照下图连接好元器件
STM32单片机学习3--STM32控制键盘
按钮左侧始终是高电位(PC0与之相连也是高电位),按钮右侧始终是低电位,当按钮按下时,PC0引脚会被强行拉低,此时引脚读到的为低电平;

检测到低电平时,翻转引脚PC10的电平为低电位,从而实现按下按钮,小灯亮;松开按钮,小灯灭;

效果展示

STM32单片机学习3--STM32控制键盘

详细代码
led.h

#ifndef __LED_H
#define __LED_H

#include "stm32f10x.h"

void LED_GPIO_Config(void);

#endif		

led.c

#include "led.h"

void LED_GPIO_Config(void)
{
	//定义一个GPIO_InitTypeDef类型的结构体
	GPIO_InitTypeDef GPIO_InitStructure;
	
	//开启GPIOC的外设时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
	
	//选择要控制的GPIO引脚
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
	//设置引脚模式为上拉输入模式
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
	//设置引脚速率为50MHz
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	//调用库函数,初始化GPIOC
	GPIO_Init(GPIOC,&GPIO_InitStructure);
	
	
	//选择要控制的GPIO引脚
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
	//设置引脚模式为推挽输出模式
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
	//设置引脚速率为50MHz
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	//调用库函数,初始化GPIOC
	GPIO_Init(GPIOC,&GPIO_InitStructure);
}

main.c

#include "stm32f10x.h"
#include "led.h"

GPIO_InitTypeDef GPIO_InitStructure;
void Delay(int m,int n)	//延时函数
{
	int i,j;
	for(i=0;i<m;i++) {
		for(j=0;j<n;j++);
	}
}

int main(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);

	LED_GPIO_Config();	//初始化GPIO
	
	while (1)
	{
		uint8_t flag = GPIO_ReadInputDataBit(GPIOC,GPIO_Pin_0);
		if(flag == 0) {
			//按钮按下,置低电位,小灯亮
			GPIO_ResetBits(GPIOC,GPIO_Pin_10);
		}else {
			//按钮松开,置高电位,小灯灭
			GPIO_SetBits(GPIOC,GPIO_Pin_10);
		}
		Delay(100,100);	//延时消抖
	}//while(1)
}

五、按钮键盘

键盘由多行多列按钮组成,程序设计通常采用逐行逐列进行扫描,4*4的矩阵键盘一共需要8个GPIO引脚,将控制行的引脚设置成输出模式,控制列的引脚设置成上拉输入模式;

先扫描第一行,那么就将PD0~PD2输出高电平,PD3输出低电平,记为0xF7;控制列的引脚为输入引脚,将其和0xF7相与,如果哪一位为0,那么就证明哪一个被按下;
STM32单片机学习3--STM32控制键盘
2、代码

接线:4*4矩阵键盘,行从上至下依此接B5、B6、B7、B8;列从左至右依此接A1、A2、A3、A4;
按键从左至右,从上至下,依此编号为1、2、3、… 、16

STM32单片机学习3--STM32控制键盘

Key.h

#ifndef __KEY_H
#define __KEY_H

#include "stm32f10x.h"

void delay_us(uint32_t delay_us);
void delay_ms(uint16_t delay_ms);

void KEY_GPIO_Config(void);
int scan(void);

#endif

Key.c

#include "Key.h"

void KEY_GPIO_Config(void)
{
	//定义一个GPIO_InitTypeDef类型的结构体
	GPIO_InitTypeDef GPIO_InitStructure;
	
	//开启GPIOA、GPIOB的外设时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB,ENABLE);
	
	///控制行
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_8;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB,&GPIO_InitStructure);
	
	///读取列
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3 | GPIO_Pin_4;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;
	GPIO_Init(GPIOA,&GPIO_InitStructure);
}

int scan(void)
{
	uint8_t flag = 1;
	
	//扫描第一行
	GPIO_ResetBits(GPIOB,GPIO_Pin_5);
	GPIO_SetBits(GPIOB,GPIO_Pin_6);
	GPIO_SetBits(GPIOB,GPIO_Pin_7);
	GPIO_SetBits(GPIOB,GPIO_Pin_8);
	
	//扫描第一列
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_1);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_1);
		if(flag == 0) {
			flag = 1;
			return 1;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_2);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_2);
		if(flag == 0) {
			flag = 1;
			return 2;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_3);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_3);
		if(flag == 0) {
			flag = 1;
			return 3;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4);
		if(flag == 0) {
			flag = 1;
			return 4;
		}
	}
	
	//扫描第二行
	GPIO_SetBits(GPIOB,GPIO_Pin_5);
	GPIO_ResetBits(GPIOB,GPIO_Pin_6);
	GPIO_SetBits(GPIOB,GPIO_Pin_7);
	GPIO_SetBits(GPIOB,GPIO_Pin_8);
	
	//扫描第二列
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_1);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_1);
		if(flag == 0) {
			flag = 1;
			return 5;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_2);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_2);
		if(flag == 0) {
			flag = 1;
			return 6;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_3);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_3);
		if(flag == 0) {
			flag = 1;
			return 7;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4);
		if(flag == 0) {
			flag = 1;
			return 8;
		}
	}
	
	//扫描第三行
	GPIO_SetBits(GPIOB,GPIO_Pin_5);
	GPIO_SetBits(GPIOB,GPIO_Pin_6);
	GPIO_ResetBits(GPIOB,GPIO_Pin_7);
	GPIO_SetBits(GPIOB,GPIO_Pin_8);
	
	//扫描第三列
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_1);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_1);
		if(flag == 0) {
			flag = 1;
			return 9;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_2);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_2);
		if(flag == 0) {
			flag = 1;
			return 10;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_3);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_3);
		if(flag == 0) {
			flag = 1;
			return 11;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4);
		if(flag == 0) {
			flag = 1;
			return 12;
		}
	}
	
	//扫描第四行
	GPIO_SetBits(GPIOB,GPIO_Pin_5);
	GPIO_SetBits(GPIOB,GPIO_Pin_6);
	GPIO_ResetBits(GPIOB,GPIO_Pin_7);
	GPIO_SetBits(GPIOB,GPIO_Pin_8);
	
	//扫描第四列
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_1);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_1);
		if(flag == 0) {
			flag = 1;
			return 13;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_2);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_2);
		if(flag == 0) {
			flag = 1;
			return 14;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_3);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_3);
		if(flag == 0) {
			flag = 1;
			return 15;
		}
	}
	flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4);
	if(flag == 0) {
		delay_ms(200);
		flag = GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4);
		if(flag == 0) {
			flag = 1;
			return 16;
		}
	}
	
	return -1;
}
  • STM32用GPIO_ReadInputDataBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)读取GPIO引脚的电平(配置为输入模式);
  • 一般检测到低电平后延时一会再次检测,是常见的消抖手段,防止误触;
    STM32单片机学习3--STM32控制键盘

main.c文章来源地址https://www.toymoban.com/news/detail-498514.html

#include "stm32f10x.h"
#include "Key.h"

int main(void)
{
	KEY_GPIO_Config();
	while(1)
	{
		switch(scan())
		{
			case 1:
				break;
			case 2:
				break;
			case 3:
				break;
			case 4:
				break;
			case 5:
				break;
			case 6:
				break;
			case 7:
				break;
			case 8:
				break;
			case 9:
				break;
			case 10:
				break;
			case 11:
				break;
			case 12:
				break;
			case 13:
				break;
			case 14:
				break;
			case 15:
				break;
			case 16:
				break;
		}
		
	}
	
}

到了这里,关于STM32单片机学习3--STM32控制键盘的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【Proteus仿真】【STM32单片机】智能窗帘控制系统设计

    本项目使用Proteus8仿真STM32单片机控制器,使用LCD1602显示模块、按键模块、HC05蓝牙、DHT11温湿度、PCF8591 ADC模块、光线传感器、28BYJ48步进电机等。 主要功能: 系统运行后,LCD1602显示温湿度和光线强度值和系统模式状态。 系统模式以自动模式运行,当按下K4键切换为手动模式

    2023年04月24日
    浏览(38)
  • 【STM32单片机】STM32控制SG90舵机的PWM部分参数的设置解答

    我们要知道,SG90舵机接收的PWM信号的参数:f=50Hz,T=1/f,所以周期为20ms。 当高电平的脉宽在0.5ms-2.5ms之间时舵机就可以对应旋转到不同的角度。 换句话说,我们要用单片机产生一个 周期(20ms) 的PWM波,然后获得对应这些时长(分别是 0.5ms 、 1ms 、 1.5ms 、 2ms 、 2.5ms )的 高电

    2024年02月13日
    浏览(57)
  • 【Proteus仿真】【STM32单片机】交通灯控制系统设计

    本项目使用Proteus8仿真STM32单片机控制器,使用数码管、按键、交通信号灯模块等。 系统运行后,交通灯系统开始运行,数码管显示初始时间,默认南北绿灯5S,东西红灯8S,绿灯过后南北黄灯3秒;然后切换东西绿灯5S,南北红灯8S,绿灯过后南北黄灯3秒;如此循环。 可使用

    2024年02月11日
    浏览(44)
  • 【Proteus仿真】【STM32单片机】蔬菜大棚温湿度控制系统设计

    本项目使用Proteus8仿真STM32单片机控制器,使用LCD1602液晶、DHT11温湿度、蜂鸣器、按键、LED、继电器、电机模块等。 主要功能: 系统运行后,LCD1602显示DHT11温湿度采集值, 当按下K3键进入阈值设置,第一次进入温度上限, 第二次进入温度下限,第三次进入湿度上限, 第四次

    2023年04月08日
    浏览(43)
  • stm32单片机开关控制LED灯(中断方式)(proteus电路图)

      补充一点:拼多多上面LED额定电流20mA,额定电压2V,额定电阻100欧姆,后边这个是STM32 如果只是用面包板做实验,输入输出都不是很多,就直接用灌电流的方式把LED直接接到STM32了,即使是设计产品,如果能够直接接,那么肯定也是不会浪费钱去买额外的原件的,所以只有

    2024年02月11日
    浏览(44)
  • 单片机毕设 STM32智能饮水机控制系统(源码+硬件+论文)

    🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。 为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天

    2024年02月11日
    浏览(47)
  • JDY-31蓝牙模块远程控制STM32F103单片机

       手机app通过蓝牙模块发送指令实时控制单片机的外设功能,比如发送衣柜开关门指令(舵机旋转),衣架上升降落(步进电机正转反转),远程开启去污除湿功能(继电器控制打开关闭小风扇+加热片)。 本次例子:手机APP连接蓝牙模块远程控制SG90舵机正转(0-180°)和反

    2024年02月01日
    浏览(74)
  • stm32F103单片机pwm控制伺服驱动器进而控制步进电机

    1.需要一台步进电机,私服驱动器,stm32单片机;先按照说明文档,连接好硬件相关线路,对应好sign+,sign-,puls+,puls-线路,其中sign只是个io口拉高拉低操作,puls是pwm波形输出口,虽然有两根线,但只是需要控制一个IO口输出波形即可; 采用主从模式,TIM1为主定时器,tim3未从定

    2024年02月04日
    浏览(56)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包