一.问题描述
用Protues画仿真图如下,主要完成功能当开关闭合时,灯亮;开关断开时灯灭。
程序流程,配置A0\2\4管脚为输出管脚。配置B0管脚为输入管脚,依据图,为上拉输入模式。
设想的执行流程--仿真运行时:当手动开关闭合,检测到B0管脚为0,让A0\2\4管脚输出高电平,灯亮;当手动开关断开,B0管脚悬空,由于输入上拉模式,检测到B0管脚为1,让A0\2\4管脚输出低电平,灯灭;
在keil里主程序代码如下:
#include "stm32f10x.h"
int main()
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//使能A组管脚对应时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);//使能B组管脚对应时钟
//初始化A组;
GPIO_InitTypeDef zhangsan;//GPIO_InitStructure;
zhangsan.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_2|GPIO_Pin_4;//初始化赋值使用0\2\4管脚
zhangsan.GPIO_Speed=GPIO_Speed_50MHz;//管脚速率
zhangsan.GPIO_Mode=GPIO_Mode_Out_PP;//配置管脚为推挽输出
GPIO_Init(GPIOA,&zhangsan);//执行初始化函数,初始化A0、a2管脚
//初始化B组;
zhangsan.GPIO_Pin=GPIO_Pin_0;//初始化赋值使用0管脚
zhangsan.GPIO_Mode=GPIO_Mode_IPU;//配置管脚为上拉输入模式
GPIO_Init(GPIOB,&zhangsan);//执行初始化函数,初始化A0、a2管脚
//判断B0管脚输入的高低电平,如果B0是1(高电平),LED灭——A0\2\4输出低电平;如果B0是0(低电平),LED亮——A0\2\4输出高电平
int B0zhi=0;
while(1)
{
B0zhi=GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_0);
if(B0zhi==0)//如果B0管脚输入的低电平,A0\2\4输出高电平-LED亮
GPIO_SetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_2|GPIO_Pin_4);//GPIO_Write(GPIOA,0x0015);//管脚A0、2、4输出高电平,其他管脚输出低电平
else//否则,A0\2\4输出低电平-LED灭 GPIO_ResetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_2|GPIO_Pin_4);//GPIO_Write(GPIOA,0x0000);//管脚A组输出低电平
}
}
将代码编译载入到仿真图后,运行的效果,当开关闭合时,led亮,当开关断开时,led亮。没有达到预期效果。
二.问题分析
怀疑是Protues软件对于stm32的电源配置没到位。当设置为上拉输入时,内部的VDD电源没有起到该有的作用。
所有很多开发,建议使用真实的硬件板子完成。
三.解决方法
1.代码不变的情况下,增加部分电路如下
2.配置B0管脚为浮空输入模式
用Protues画仿真图如下,
在keil里主程序主函数代码如下:(相对于原代码,只是修改了zhangsan.GPIO_Mode=GPIO_Mode_IN_FLOATING;//配置管脚为浮空输入模式)
#include "stm32f10x.h"
int main()
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//使能A组管脚对应时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);//使能B组管脚对应时钟
//初始化A组;
GPIO_InitTypeDef zhangsan;//GPIO_InitStructure;
zhangsan.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_2|GPIO_Pin_4;//初始化赋值使用0\2\4管脚
zhangsan.GPIO_Speed=GPIO_Speed_50MHz;//管脚速率
zhangsan.GPIO_Mode=GPIO_Mode_Out_PP;//配置管脚为推挽输出
GPIO_Init(GPIOA,&zhangsan);//执行初始化函数,初始化A0、a2管脚
//初始化B组;
zhangsan.GPIO_Pin=GPIO_Pin_0;//初始化赋值使用0管脚
zhangsan.GPIO_Mode=GPIO_Mode_IN_FLOATING;//配置管脚为浮空输入模式
GPIO_Init(GPIOB,&zhangsan);//执行初始化函数,初始化A0、a2管脚
//判断B0管脚输入的高低电平,如果B0是1(高电平),LED灭——A0\2\4输出低电平;如果B0是0(低电平),LED亮——A0\2\4输出高电平
int B0zhi=0;
while(1)
{
B0zhi=GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_0);
if(B0zhi==0)//如果B0管脚输入的低电平,A0\2\4输出高电平-LED亮
GPIO_SetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_2|GPIO_Pin_4);//GPIO_Write(GPIOA,0x0015);//管脚A0、2、4输出高电平,其他管脚输出低电平
else//否则,A0\2\4输出低电平-LED灭
GPIO_ResetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_2|GPIO_Pin_4);//GPIO_Write(GPIOA,0x0000);//管脚A组输出低电平
}
}
3.配置B0管脚为输入下拉模式
用Protues画仿真图如下,
在keil里主程序主函数代码如下:
int main()
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//使能A组管脚对应时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);//使能B组管脚对应时钟
//初始化A组;
GPIO_InitTypeDef zhangsan;//GPIO_InitStructure;
zhangsan.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_2|GPIO_Pin_4;//初始化赋值使用0\2\4管脚
zhangsan.GPIO_Speed=GPIO_Speed_50MHz;//管脚速率
zhangsan.GPIO_Mode=GPIO_Mode_Out_PP;//配置管脚为推挽输出
GPIO_Init(GPIOA,&zhangsan);//执行初始化函数,初始化A0、a2管脚
//初始化B组;
zhangsan.GPIO_Pin=GPIO_Pin_0;//初始化赋值使用0管脚
zhangsan.GPIO_Mode=GPIO_Mode_IPD;//配置管脚为上拉输入模式
GPIO_Init(GPIOB,&zhangsan);//执行初始化函数,初始化A0、a2管脚
//判断B0管脚输入的高低电平,如果B0是1(高电平),LED灭——A0\2\4输出低电平;如果B0是0(低电平),LED亮——A0\2\4输出高电平
int B0zhi=0;
while(1)
{
B0zhi=GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_0);
if(B0zhi==1)//如果B0管脚输入的低电平,A0\2\4输出高电平-LED亮
GPIO_SetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_2|GPIO_Pin_4);//GPIO_Write(GPIOA,0x0015);//管脚A0、2、4输出高电平,其他管脚输出低电平
else//否则,A0\2\4输出低电平-LED灭
GPIO_ResetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_2|GPIO_Pin_4);//GPIO_Write(GPIOA,0x0000);//管脚A组输出低电平
}
}