嵌入式-Stm32-江科大基于标准库的GPIO4个小实验
文章目录
- 一 、硬件介绍
- 二 、实验:LED闪烁、LED流水灯、蜂鸣器提示
- 2.1 需求1:面包板上的LED以1s为周期进行闪烁。亮0.5s,灭0.5s.....
- 2.2 需求2: 8个LED实现流水灯
- 2.3 需求3:蜂鸣器不断地发出滴滴、滴滴.....的提示音。蜂鸣器低电平触发。
- 三、硬件介绍-按键开关、光敏电阻
- 四、 实验 按键控制LED、光敏传感器控制蜂鸣器
- 4.1 需求1:一个按键开关控制一个LED,每次按下按键,LED就改变自己的亮灭状态;两套系统互不影响
- 4.2 需求2:光敏电阻被遮挡,蜂鸣器长鸣,光敏电阻不被遮挡,蜂鸣器不响。
- 道友:蹉跎者光阴如梭,有志者岁月如歌
核心:本文共4个小实验:
第一个:LED灯闪烁
第二个:LED流水灯
第三个:按键控制LED
第四个:光敏传感器控制蜂鸣器
一 、硬件介绍
1.1 LED、蜂鸣器、面包板
- LED:发光二极管,正向通电点亮,反向通电不亮。
- 有源蜂鸣器(本实验):内部自带振荡源,将正负极接上直流电压即可持续发声,频率固定。如下图所示的蜂鸣器模块使用三极管作为开关。
- 无源蜂鸣器:内部不带振荡源,需要控制器提供振荡脉冲才可发声,调整提供振荡脉冲的频率,可发出不同频率的声音(UP的51单片机课程中用晶振提供振荡源)。
注:LED长脚为正极,灯里面内部小头为正极。本实验的蜂鸣器低电平驱动。
最右边的半圆符号是蜂鸣器
1.2 硬件电路
上图给出了led和蜂鸣器的驱动电路图。注意,三极管的发射极一定要直接接正电源/地,这是因为三极管的开启需要发射极和基极之间有一定的电压(电压差),如果接在负载侧则有可能会导致三极管无法正常开启。
三极管开关是最简单的驱动电路
这里的保护电阻(限流电阻)在UP视频中为了方便没有加,自己设计电路需要加上去。
很多单片机中或者芯片片,都使用了高电平弱驱动,低电平强驱动的规则。
上右一的图是PNP三极管驱动电路类型:当基极R1给低电平,蜂鸣器能被驱动工作,当R1给高电平,三极管截止电流,蜂鸣器不工作,PNP的三极管最好接在上边,NPN的三极管最好接在下边。
下右二的图是NPN类型:当基极(R1)给高电平导通,低电平断开
1.3 三极管的简单介绍
三极管(Transistor):
三极管是一种由三个半导体区域组成的电子元件,通常是NPN或PNP型。它具有三个引脚:发射极(Emitter)、基极(Base)和集电极(Collector)。三极管是一种可控电流放大器,可以根据输入信号的变化来控制输出电流。它可以作为开关或放大器使用。在放大器应用中,输入信号作用在基极上,通过控制基极电流,可以放大电流并将其输出到集电极。在开关应用中,基极电流的变化可以控制集电极与发射极之间的电流流动,实现开关功能。
PNP三极管驱动电路类型:当基极R1给低电平,蜂鸣器能被驱动工作,当R1给高电平,三极管截止电流,蜂鸣器不工作,PNP的三极管最好接在上边,NPN的三极管最好接在下边。
NPN类型:当基极给高电平导通,低电平断开。
1.4 面包板
上图是面包板。可以看出,面包板中间的金属爪是竖着排列的,用于插各种元器件;最上和最下的两条金属爪是横着排列的,一般用于供电。注意,在使用面包板之前,一定要观察孔位的连接情况,比如接电源那里可能会有断开的情况,而且用杜邦线插拔可能会接触不到位的情况。
二 、实验:LED闪烁、LED流水灯、蜂鸣器提示
2.1 需求1:面包板上的LED以1s为周期进行闪烁。亮0.5s,灭0.5s…
- LED低电平驱动
低电平驱动(Low-Level Driving)是指在数字电路中,输出信号的电平为低电平(通常为低电压)时,能够正确地驱动连接的设备或元件。在数字电路中,通常将高电平定义为逻辑1,低电平定义为逻辑0。相当于低电平触发。
- 需要用到延时函数Delay.h、Delay.c ,在UP提供的“程序源码”中,为了方便管理,应在工程内建System文件夹,专门存放这些可以复用的代码。
上图为LED闪烁接线图,实际上,应该在led和驱动电源之间接上保护电阻,但由于本电路过于简单,于是直接省略保护电阻。后面的"LED流水灯“、”蜂鸣器提示”实验同样省略保护电阻(我试过如果Vcc接5V甚至更大,如图的小发光二极管(容忍电压上限3v左右)会冒烟烧坏)
2.1 代码展示
选中函数名,按F12进入查看函数参数原型(部分注释没有改过来别搞乱了,关注点是红色框框)
上图为除库函数之外的LED闪烁-代码调用
main.c
#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}
}
Delay.c //延时模块,不用管代码怎么实现,直接拿来用就行
#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)); //等待计数到0SysTick->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);}
}
Delay.h
#ifndef __DELAY_H //这是预处理指令,用来判断某个宏是否已经定义,如果该宏(没有)定义,则执行下面的代码
#define __DELAY_H //宏定义void Delay_us(uint32_t us);
void Delay_ms(uint32_t ms);
void Delay_s(uint32_t s);#endif
当一个头文件(例如delay.h)被多个源文件引用时,可以使用预处理指令 #ifndef、#define 和 #endif 来确保头文件只被编译一次。
#ifndef 预处理指令:
#ifndef 是 “if not defined” 的简写。
#ifndef 用于检查某个宏是否未定义。如果该宏未定义,则执行下面的代码。
如果宏已经定义过了,#ifndef 中的代码将被忽略。
举例子说明:
#endif
是C/C++中的预处理指令,用于结束条件编译块。
当在代码中使用
#ifdef
、#ifndef
或#if
这些条件编译指令时,需要使用#endif
来标记条件编译块的结束位置。例如,以下是一个简单的条件编译示例:
#ifdef DEBUG_MODE// 调试模式下执行的代码printf("Debug mode is enabled.\n"); #else// 正常模式下执行的代码printf("Debug mode is disabled.\n"); #endif 在这个示例中, #ifdef DEBUG_MODE 判断是否定义了 DEBUG_MODE 宏。如果定义了该宏,则执行调试模式下的代码;否则,执行正常模式下的代码。最后的 #endif 指令用于标记条件编译块的结束位置。它表示上面的条件编译块已经结束,之后的代码将会被正常编译。总结来说, #endif 用于结束条件编译块,它与 #ifdef 、 #ifndef 或 #if 一起使用,确保条件编译的范围正确闭合。
注意:此后Delay.h、Delay.c 将作为常用处理函数长期存放于System文件夹中,后续如果使用到将直接调用,故后续不会再在笔记中展示源代码。
编程感想:
- Keil编译过后,整个工程会比较大,不利于分享给别人。可以使用UP主提供的批处理程序(KeilKill.bat),删掉工程中的中间文件后再分享给别人(编译前2M,编译后20M,所以要使用批处理程序),其他人使用的时候只需要重新编译一下就行。
- 本教程用到了RCC和GPIO两个外设,这些外设的库函数在Library中,一般存放在相应的.h文件的最后。
- 将LED的短脚接负极,长脚接PA0口,就是高电平驱动方式,但是现象和低电平相同。
- 将GPIO设置成开漏输出模式,可以发现高电平(高阻态)无驱动能力,低电平有驱动能力,所以一般常使用推挽输出模式 ,以便同时具备高电平和低电平的驱动能力。
5.记得添加文件夹的路径
2.2 需求2: 8个LED实现流水灯
面包板上的8个LED以0.5s切换一个的速度,实现流水灯。低电平驱动。
上图为LED流水灯-接线图
代码调用关系与“2-1的LED闪烁实验”相同,下面是代码展示:
main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"int main(void){
// //配置RCC寄存器,使能GPIOC的时钟。GPIO都属于APB2外设
// RCC->APB2ENR = 0x00000010;
// //配置寄存器PC13:通用推挽输出模式、输出模式50MHz
// GPIOC->CRH = 0x00300000;
// //输出数据寄存器PC13:
// //GPIOC->ODR = 0x00002000;//LED灭
// GPIOC->ODR = 0x00000000;//LED亮//1.开启GPIO的外设时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//2.配置寄存器PC13的端口模式//2.1 首先配置GPIO结构体GPIO_InitTypeDef GPIO_InitStructure ;//给结构体起名字GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//寄存器模式为通用推挽输出GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7;//寄存器引脚为13GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//输出速度为50MHz//2.2 然后才能调用函数配置寄存器GPIO_Init(GPIOA,&GPIO_InitStructure);//3.配置数据寄存器PC13的输出//GPIO_SetBits(GPIOC,GPIO_Pin_13);//设置高电平,灯灭
// GPIO_ResetBits(GPIOC,GPIO_Pin_13);//设置低电平,灯亮while(1)
{//使用GPIO_SetBits、GPIO_ResetBits进行赋值,这里仅用于演示“或操作”同时赋值GPIO_SetBits(GPIOA,GPIO_Pin_0 | GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7);GPIO_ResetBits(GPIOA,GPIO_Pin_0 | GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7);//使用GPIO_Write,同时设置GPIOA所有引脚的高低电平,实现LED流水灯//对指定的端口同时赋值GPIO_Write(GPIOA,~0x01); //0x0001 二进制:0000 0000 0000 0001Delay_ms(500);GPIO_Write(GPIOA,~0x02); //二进制:0000 0000 0000 0010Delay_ms(500);GPIO_Write(GPIOA,~0x04);//二进制:0000 0000 0000 1000Delay_ms(500);GPIO_Write(GPIOA,~0x08);Delay_ms(500);GPIO_Write(GPIOA,~0x010);Delay_ms(500);GPIO_Write(GPIOA, ~0x20);Delay_ms(500);GPIO_Write(GPIOA, ~0x40);Delay_ms(500);GPIO_Write(GPIOA, ~0x80);//对应的二进制数 1000 0000,表示最高位的引脚
}
}
思路:使用或操作 | 就可以实现只初始化定义某几个GPIO,或者某几个外设的时钟
2.3 需求3:蜂鸣器不断地发出滴滴、滴滴…的提示音。蜂鸣器低电平触发。
注:蜂鸣器执行四个动作为1个周期,分别是响0.1s,静音0.1s,响0.1s,静音0.1s。
上图是蜂鸣器提示-接线图(GPIOB,Pin_12)
代码调用关系与“LED闪烁”实验相同,下面是代码展示:
mian.c
#include "stm32f10x.h" // Device header
#include "Delay.h"int main(void){// 开启APB2-GPIOB的外设时钟RCCRCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);// 初始化PB12端口:定义结构体及参数GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;GPIO_Init(GPIOB, &GPIO_InitStructure);while(1){GPIO_ResetBits(GPIOB, GPIO_Pin_12);Delay_ms(100);GPIO_SetBits(GPIOB, GPIO_Pin_12);Delay_ms(100);GPIO_ResetBits(GPIOB, GPIO_Pin_12);Delay_ms(100);GPIO_SetBits(GPIOB, GPIO_Pin_12);Delay_ms(100);Delay_ms(600);};
}
编程感想:
1.控制蜂鸣器的IO端口可以随便选,但是不要选择三个JTAG调试端口:PA15、PB3、PB4。本实验选择PB12端口进行输出。
2.关于调用库函数,有以下几种方法:
- 直接查看每一个外设的.h函数,拖到最后就可以看到本外设的所有库函数,然后在对应的.c文件中查看函数定义和调用方式即可。
- 查看库函数的用户手册-“stm32f103xx固件函数库用户手册.pdf”,这个中文版比较老;新版本的用户手册可以在st公司的帮助文档中查看,但只有英文版。
- CSDN百度别人的代码。
三、硬件介绍-按键开关、光敏电阻
上图是按键开关实物图,虽然前面说过,GPIO端口有专门的肖特基触发器对输入信号进行整型,但按键开关的抖动幅度大、时间长,所以还需要“软件消抖”。基本思路就是延迟5-10ms,跳过抖动时间范围即可。
上图是按键开关硬件电路,上图给出了按键开关的硬件电路设计图(下接按键(常用)和上接按键)。对于按键开关来说,常见以上四种设计方法,而行业规范中,单片机端口一般都有上拉输入模式(弱上拉),所以基本上就选择内部上拉/外部上拉的设计电路。
- 下接按键(常用):第一个图,按下按键,引脚为低电平,松手,引脚为高电平
- 如果电路同时存在内部上拉和外部上拉,那么其高电平的驱动能力更强,但是低电平会更加耗电。两个下拉电路则可以使低电平驱动能力更强,而不会明显增加损耗。
- 浮空输入模式下,每部没有上下拉,此时必须在外部有上下拉电路。
- 注意内部和外部的上下拉必须一致,内部有上下拉时,就可以不用配置外部上下拉。
上图是传感器模块实物图,从左到右依次是:光敏电阻传感器、热敏电阻传感器、对射式红外传感器、反射式红外传感器。传感器(光敏电阻/热敏电阻/红外接收管等)的电阻会随外界模拟量的变化而变化,通过与定值电阻分压可得到模拟电压输出,再通过电压比较器进行二值化即可得到数字电压输出。
上图是传感器模块原理图:
上述所给出的是一个比较通用的传感器模块格式
- 左起第三个模块:下面的可变电阻就是各种传感器模块所对应的阻值,与上面的分压电压R1进行,进而输出模拟电压值A0。电容C2是滤波电容。
- 左边起第一个模块:使用LM393模块,通过运算放大器实现“电压比较器”的功能。IN-是一个可以调节的值,IN+则直接连接传感器的模拟分压AO,当AO>IN-时,数字输出DO拉高;当AO<IN-时,数字输出DO拉低。
- 左边起第二个模块:通过一个滑变电阻器实现比较电压IN-的调整。
- 左边起第四个模块:电源指示灯。
- 左边起第五个模块:传感器模块的端口。LED2用于指示数字输出DO的值。注意R5上拉电阻保证DO的默认值为高。
补充情况:
1.对于对射式红外传感器来说,N1就是红外接收管,并且额外还有一个点亮红外发射管的电路,模拟电压表示时接收红外信号的强度。并且该模块常用于检测通断,所以用两个电阻将阈值固定为1/2的参考电压,而不是采用滑动变阻器。
2.对于反射式红外式传感器,向下发射和接收红外光,可以做循迹小车。
而对于传感器模块的电路设计来说,由于采用模块的方案,所以直接给传感器接上VCC和GND,将模拟信号AO和数字信号DO接在stm32的对应端口上即可。
本次实验采用数字信号DO接入,关于模拟信号接入的使用方法在后面AD/DA的实验中继续讲解。
四、 实验 按键控制LED、光敏传感器控制蜂鸣器
4.1 需求1:一个按键开关控制一个LED,每次按下按键,LED就改变自己的亮灭状态;两套系统互不影响
- LED低电平驱动。
- 按键B11控制LEDA2,按键B1控制LEDA1.
- LED的状态改变是“松开触发”。
上图分别依次是按键控制LED-接线图和代码调用(不包括库函数)
以下是代码展示
main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "LED.h"
#include "Key.h"uint8_t KeyNum; //定义用于接收按键键码的变量int main(void)
{/*模块初始化*/LED_Init(); //LED初始化Key_Init(); //按键初始化while (1){KeyNum = Key_GetNum(); //获取按键键码if (KeyNum == 1) //按键1按下{LED1_Turn(); //LED1翻转}if (KeyNum == 2) //按键2按下{LED2_Turn(); //LED2翻转}}
}
LED.h
#ifndef __LED_H
#define __LED_Hvoid LED_Init(void);
void LED1_ON(void);
void LED1_OFF(void);
void LED1_Turn(void);
void LED2_ON(void);
void LED2_OFF(void);
void LED2_Turn(void);#endif
Key.h
#ifndef __KEY_H
#define __KEY_Hvoid Key_Init(void);
uint8_t Key_GetNum(void);#endif
LED.c
#include "stm32f10x.h" // Device header/*** 函 数:LED初始化,初始化PA2、PA1作为两个LED的输出端口* 参 数:无* 返 回 值:无*/
void LED_Init(void)
{/*开启时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //开启GPIOA的时钟/*GPIO初始化*///初始化PA的输出端口:定义结构体及参数GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//推挽输出GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_2;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure); //将PA1和PA2引脚初始化为推挽输出/*设置GPIO初始化后的默认电平,默认输出为低电平LED初始不亮*/GPIO_SetBits(GPIOA, GPIO_Pin_1 | GPIO_Pin_2); //设置PA1和PA2引脚为高电平
}/*** 函 数:LED1开启* 参 数:无* 返 回 值:无*/
void LED1_ON(void)
{GPIO_ResetBits(GPIOA, GPIO_Pin_1); //设置PA1引脚为低电平
}/*** 函 数:LED1关闭* 参 数:无* 返 回 值:无*/
void LED1_OFF(void)
{GPIO_SetBits(GPIOA, GPIO_Pin_1); //设置PA1引脚为高电平
}/*** 函 数:LED1状态翻转* 参 数:无* 返 回 值:无*/
void LED1_Turn(void)
{if (GPIO_ReadOutputDataBit(GPIOA, GPIO_Pin_1) == 0) //获取输出寄存器的状态,如果当前引脚输出低电平{GPIO_SetBits(GPIOA, GPIO_Pin_1); //则设置PA1引脚为高电平}else //否则,即当前引脚输出高电平{GPIO_ResetBits(GPIOA, GPIO_Pin_1); //则设置PA1引脚为低电平}
}/*** 函 数:LED2开启* 参 数:无* 返 回 值:无*/
void LED2_ON(void)
{GPIO_ResetBits(GPIOA, GPIO_Pin_2); //设置PA2引脚为低电平
}/*** 函 数:LED2关闭* 参 数:无* 返 回 值:无*/
void LED2_OFF(void)
{GPIO_SetBits(GPIOA, GPIO_Pin_2); //设置PA2引脚为高电平
}/*** 函 数:LED2状态翻转* 参 数:无* 返 回 值:无*/
void LED2_Turn(void)
{if (GPIO_ReadOutputDataBit(GPIOA, GPIO_Pin_2) == 0) //获取输出寄存器的状态,如果当前引脚输出低电平{ GPIO_SetBits(GPIOA, GPIO_Pin_2); //则设置PA2引脚为高电平} else //否则,即当前引脚输出高电平{ GPIO_ResetBits(GPIOA, GPIO_Pin_2); //则设置PA2引脚为低电平}
}
Key.c
#include "stm32f10x.h" // Device header
#include "Delay.h"/*** 函 数:按键初始化,初始化B11、B1作为按键2、按键1* 参 数:无* 返 回 值:无*/
void Key_Init(void)
{/*开启时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //开启GPIOB的时钟/*GPIO初始化*/GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_11;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOB, &GPIO_InitStructure); //将PB1和PB11引脚初始化为上拉输入
}/*** 函 数:按键获取键码* 参 数:无* 返 回 值:按下按键的键码值,范围:0~2,返回0代表没有按键按下* 注意事项:此函数是阻塞式操作,当按键按住不放时,函数会卡住,直到按键松手*/
uint8_t Key_GetNum(void)
{uint8_t KeyNum = 0; //定义变量,默认键码值为0if (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_1) == 0) //读PB1输入寄存器的状态,如果为0,则代表按键1按下{Delay_ms(20); //延时消抖while (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_1) == 0); //等待按键松手Delay_ms(20); //延时消抖KeyNum = 1; //置键码为1}if (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_11) == 0) //读PB11输入寄存器的状态,如果为0,则代表按键2按下{Delay_ms(20); //延时消抖while (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_11) == 0); //等待按键松手Delay_ms(20); //延时消抖KeyNum = 2; //置键码为2}return KeyNum; //返回键码值,如果没有按键按下,所有if都不成立,则键码为默认值0
}
编程感想:
1 .代码提示框如果不弹出,可以使用CTRL+space 弹出代码提示框。如果没用的话,大概率是和中英切换快捷键冲突,输入法右键设置取消即可。
2.GPIO配置好之后默认就是低电平,所以配置好后LED会默认是亮的状态。
3.本工程创建了全新的驱动函数文件夹Hardware,专门用于存放程序中使用到的外设(如LED、按键、光敏传感器等)的驱动函数。做好驱动代码的提取是非常重要的,可以极大地方便梳理框架。
4.其实写完之后发现,这个按键开关非常不灵敏,经常出现按键松手后LED没有反应的情况,大概这就是设置“光敏传感器控制蜂鸣器”实验的原因吧
4.2 需求2:光敏电阻被遮挡,蜂鸣器长鸣,光敏电阻不被遮挡,蜂鸣器不响。
- 蜂鸣器低电平驱动。
- 光敏传感器,光强越强阻值越小,分压越小;DO的LED指示灯低电平驱动。
上图是光敏传感器控制蜂鸣器的接线图,下图是代码调用(不包括库函数)
main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "Buzzer.h"
#include "LightSensor.h"int main(void)
{/*模块初始化*/Buzzer_Init(); //蜂鸣器初始化LightSensor_Init(); //光敏传感器初始化while (1){if (LightSensor_Get() == 1) //如果当前光敏输出1{Buzzer_ON(); //蜂鸣器开启}else //否则{Buzzer_OFF(); //蜂鸣器关闭}}
}
Buzzer.h
#ifndef __BUZZER_H
#define __BUZZER_Hvoid Buzzer_Init(void);
void Buzzer_ON(void);
void Buzzer_OFF(void);
void Buzzer_Turn(void);#endif
Buzzer.c
#include "stm32f10x.h" // Device header/*** 函 数:蜂鸣器初始化* 参 数:无* 返 回 值:无*/
void Buzzer_Init(void)
{/*开启时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //开启GPIOB的时钟/*GPIO初始化*/GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOB, &GPIO_InitStructure); //将PB12引脚初始化为推挽输出/*设置GPIO初始化后的默认电平*/GPIO_SetBits(GPIOB, GPIO_Pin_12); //设置PB12引脚为高电平
}/*** 函 数:蜂鸣器开启* 参 数:无* 返 回 值:无*/
void Buzzer_ON(void)
{GPIO_ResetBits(GPIOB, GPIO_Pin_12); //设置PB12引脚为低电平
}/*** 函 数:蜂鸣器关闭* 参 数:无* 返 回 值:无*/
void Buzzer_OFF(void)
{GPIO_SetBits(GPIOB, GPIO_Pin_12); //设置PB12引脚为高电平
}/*** 函 数:蜂鸣器状态翻转* 参 数:无* 返 回 值:无*/
void Buzzer_Turn(void)
{if (GPIO_ReadOutputDataBit(GPIOB, GPIO_Pin_12) == 0) //获取输出寄存器的状态,如果当前引脚输出低电平{GPIO_SetBits(GPIOB, GPIO_Pin_12); //则设置PB12引脚为高电平}else //否则,即当前引脚输出高电平{GPIO_ResetBits(GPIOB, GPIO_Pin_12); //则设置PB12引脚为低电平}
}
LightSensor.h
#ifndef __LIGHT_SENSOR_H
#define __LIGHT_SENSOR_Hvoid LightSensor_Init(void);
uint8_t LightSensor_Get(void);#endif
LightSensor.c
#include "stm32f10x.h" // Device header/*** 函 数:光敏传感器初始化* 参 数:无* 返 回 值:无*/
void LightSensor_Init(void)
{/*开启时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //开启GPIOB的时钟/*GPIO初始化*/GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOB, &GPIO_InitStructure); //将PB13引脚初始化为上拉输入
}/*** 函 数:获取当前光敏传感器输出的高低电平* 参 数:无* 返 回 值:光敏传感器输出的高低电平,范围:0/1*/
uint8_t LightSensor_Get(void)
{return GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_13); //返回PB13输入寄存器的状态
}
编程感想:
1.模块化编程香,每个模块都写一个专门的驱动函数存放在Hardware文件夹中,使得main.c函数极其简单,几乎就可以直接按照人类正常的功能逻辑书写代码
2.蜂鸣器是低电平驱动 光敏传感器,光强越强那阻值越小,分压越小;光敏电阻被阻挡,蜂鸣器长鸣。
3.获取光敏电阻返回的GPIO值初始值是1,当有光源时,变为0。
参考:B站STM32江协自动化&【哈工大虎慕】
道友:蹉跎者光阴如梭,有志者岁月如歌
相关文章:

嵌入式-Stm32-江科大基于标准库的GPIO4个小实验
文章目录 一 、硬件介绍二 、实验:LED闪烁、LED流水灯、蜂鸣器提示2.1 需求1:面包板上的LED以1s为周期进行闪烁。亮0.5s,灭0.5s.....2.2 需求2: 8个LED实现流水灯2.3 需求3:蜂鸣器不断地发出滴滴、滴滴.....的提示音。蜂鸣器低电平触发。 三、…...

HackTheBox - Medium - Linux - Noter
Noter Noter 是一种中型 Linux 机器,其特点是利用了 Python Flask 应用程序,该应用程序使用易受远程代码执行影响的“节点”模块。由于“MySQL”守护进程以用户“root”身份运行,因此可以通过利用“MySQL”的用户定义函数来利用它来获得RCE并…...

Uniapp多选Popup(弹出层)
uniapp中多选组件很少,故个人简单开发了一个,可简单使用,也可根据个人需求稍微改进 支持的功能 单选多选(默认)限制选择数量默认选中禁用选项 属性说明 属性默认值说明singlefalsetrue为开启单选,否则为…...

什么是网络安全?网络安全概况
网络安全涉及保护我们的计算机网络、设备和数据免受未经授权的访问或破坏。 这个领域包括多种技术、过程和控制措施,旨在保护网络、设备和数据免受攻击、损害或未授权访问。网络安全涉及多个方面,包括但不限于信息安全、应用程序安全、操作系统安全等 …...

c语言小游戏之扫雷
目录 一:游戏设计理念及思路 二:初步规划的游戏界面 三:开始扫雷游戏的实现 注:1.创建三个文件,test.c用来测试整个游戏的运行,game.c用来实现扫雷游戏的主体,game.h用来函数声明和包含头文…...

如何本地安装Python Flask并结合内网穿透实现远程开发
💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…...

在线App封装技术:HTML5的新生命
HTML5封装的魅力所在HTML5带来了丰富的多媒体功能、地理位置服务、离线存储等特性,使得Web应用的体验更加接近原生App。封装HTML5到App中,可以大大缩短开发周期,降低开发成本,并且一次编写,多平台运行,极大…...

Spring Boot 4.0:构建云原生Java应用的前沿工具
目录 前言 Spring Boot简介 Spring Boot 的新特性 1. 支持JDK 17 2. 集成云原生组件 3. 响应式编程支持 4. 更强大的安全性 5. 更简化的配置 Spring Boot 的应用场景 1. 云原生应用开发 2. 响应式应用程序 3. 安全性要求高的应用 4. JDK 17的应用 总结 作…...

Debian系统写Mysql时中文出现乱码无法定入的问题解决方案
原因是操作系统可能精简安装,没有GBK字符集,只有UTF8在转换或使用的时候有问题。 使用locale -a查看系统支持的字符集。正常的比较全的字符集的操作系统如下: 有问题的操作系统字符集如下: 解决方案: 步骤1&#…...

CPMS靶场练习
关键:找到文件上传点,分析对方验证的手段 首先查看前端发现没有任何上传的位置,找到网站的后台,通过弱口令admin 123456可以进入 通过查看网站内容发现只有文章列表可以进行文件上传;有两个图片上传点 图片验证很严格…...

CTFhub-bak文件
CTFhub-Web-信息泄露-备份文件下载-bak文件 题目信息 解题过程 看到提示说和index.php有关,在url后面加index.php.bak,跳转到http://challenge-7a4da2076cfabae6.sandbox.ctfhub.com:10800/index.php.bak网址,即: 跳转到下载页…...

本地部署轻量级web开发框架Flask并实现无公网ip远程访问开发界面
文章目录 1. 安装部署Flask2. 安装Cpolar内网穿透3. 配置Flask的web界面公网访问地址4. 公网远程访问Flask的web界面 本篇文章主要讲解如何在本地安装Flask,以及如何将其web界面发布到公网进行远程访问。 Flask是目前十分流行的web框架,采用Python编程语…...

面试题-MySQL如何定位慢查询
慢查询出现的情况就这些:聚合查询、多表查询、表数据量过大查询、深度分页查询。 表象:页面加载过慢、接口压测响应时间过长(超过1S)。 假如你的业务接口就是比较慢,你怎么知道是SQL的问题呢?就算是SQL的…...

【C++PCL】点云处理K-Means点云分割
目录 1.原理介绍 2.代码效果 3.源码展示 4.参数调试 5.注意事项...

代码随想录算法训练DAY25|回溯2
算法训练DAY25|回溯2 216.组合总和III 力扣题目链接 找出所有相加之和为 n 的 k 个数的组合。组合中只允许含有 1 - 9 的正整数,并且每种组合中不存在重复的数字。 说明: 所有数字都是正整数。 解集不能包含重复的组合。 示例 1: 输入: k 3, n …...

hanlp,pkuseg,jieba,cutword分词实践
总结:只有jieba,cutword,baidu lac成功将色盲色弱成功分对,这两个库字典应该是最全的 hanlp[持续更新中] https://github.com/hankcs/HanLP/blob/doc-zh/plugins/hanlp_demo/hanlp_demo/zh/tok_stl.ipynb import hanlp # hanlp.pretrained.tok.ALL # 语种见名称最…...

一个简单的Vue实例
Vue.js 是一个流行的 JavaScript 框架,用于构建交互式的网页应用。一个基本的 Vue 实例包含数据对象、模板、挂载点、方法和生命周期钩子等。 以下是一个简单的 Vue 实例示例: // 创建一个新的 Vue 实例 var app new Vue({el: #app, // 指定一个挂载点…...

【GoLang入门教程】Go语言几种标准库介绍(八)
ChatGPT 和文心一言哪个更好用? 文章目录 ChatGPT 和文心一言哪个更好用?强烈推荐前言几种库runtime库 ( 运行时接口)常用的函数:示例 sort库(排序接口)主要的函数和接口:示例 strings库(字符串转换、解析及…...

[系统安全] 五十四.恶意软件分析 (6)PE文件解析及利用Python获取样本时间戳
您可能之前看到过我写的类似文章,为什么还要重复撰写呢?只是想更好地帮助初学者了解病毒逆向分析和系统安全,更加成体系且不破坏之前的系列。因此,我重新开设了这个专栏,准备系统整理和深入学习系统安全、逆向分析和恶意代码检测,“系统安全”系列文章会更加聚焦,更加系…...

kafka入门(九):副本
副本 副本(Replica),指的是分布式系统对数据和服务提供的一种冗余方式。 Kafka通过多副本机制实现故障自动转移,在Kafka集群中某个broker节点失效的情况下仍然保证服务可用。 kafka 副本之间是 一主多从的关系。 其中 leader 副…...

【5G 接口协议】N2接口协议NGAP(NG Application Protocol)介绍
博主未授权任何人或组织机构转载博主任何原创文章,感谢各位对原创的支持! 博主链接 本人就职于国际知名终端厂商,负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作,目前牵头6G算力网络技术标准研究。 博客…...

2024年甘肃省职业院校技能大赛信息安全管理与评估 样题三 模块二
竞赛需要完成三个阶段的任务,分别完成三个模块,总分共计 1000分。三个模块内容和分值分别是: 1.第一阶段:模块一 网络平台搭建与设备安全防护(180 分钟,300 分)。 2.第二阶段:模块二…...

Python自动化我选DrissionPage,弃用Selenium
DrissionPage 是一个基于 python 的网页自动化工具。 它既能控制浏览器,也能收发数据包,还能把两者合而为一。 可兼顾浏览器自动化的便利性和 requests 的高效率。 它功能强大,内置无数人性化设计和便捷功能。 它的语法简洁而优雅&#x…...

MQ 消息丢失、重复、积压问题,如何解决?
面试官在面试候选人时,如果发现候选人的简历中写了在项目中使用了 MQ 技术(如 Kafka、RabbitMQ、RocketMQ),基本都会抛出一个问题:在使用 MQ 的时候,怎么确保消息 100% 不丢失? 这个问题在实际…...

【Linux】第三十三站:日志
文章目录 一、实现一个简单的日志1.简介2.可变参数3.错误等级4.时间5.打印每一条参数6.与前面的一些代码搭配使用 二、完整代码 一、实现一个简单的日志 1.简介 我们运行代码的时候,我们希望有各种各样的运行时候的一些信息。这也就是日志 它一半有日志时间&…...

MVC和MVVM区别和VUE关系
MVC(Model-View-Controller)和 MVVM(Model-View-ViewModel)是两种常见的前端架构模式,它们的主要区别在于处理业务逻辑和数据操作的方式。 MVC中,View(视图)可以直接访问Model&…...

vue3自定义按钮点击变颜色实现(多选功能)
实现效果图: 默认选中第一个按钮,未选中按钮为粉色,点击时颜色变为红色 利用动态类名,当定义isChange数值和下标index相同时,赋予act类名,实现变色效果 <template><div class"page"&…...

Redis的key过期策略是怎么实现的
这是一道经典的Redis面试题,一个Redis中可能存在很多很多的key,这些key中可能有很大一部分都有过期时间,此时Redis服务器咋知道哪些key已经过期,哪些还没过期呢? 如果直接遍历所有的key,这显然是行不通的&…...

vue+elenemt分页+springboot
目录 1、编写模板 2、发请求调接口 3、后端返回数据 1.编写实体类 2、UserController 3、Userservice接口 4、(mapper接口)UserMapper 5、xml 1、编写模板 <!-- 搜素框 --><el-input placeholder"请输入姓名" v-model"ke…...

C++ :命名空间域
目录 冲突与命名: 举个例子: 全局与局部: 域作用限定符: 命名空间域: 冲突与命名: 在C语言中,我们通常会使用stdlib.h 而stdlib.h 本质上是一个函数的库,在程序中使用的大多数…...