proteus仿真电路图,51单片机按键计数

kuaidi.ping-jia.net  作者:佚名   更新日期:2024-07-01
Proteus仿真51单片机,如何实现用 INT0 计数L1 闪烁次数

好像,在 P1.0 输出一次高电平,D1 就亮一次。

但是,D1,缺少电源。
你用 P3.2 给 D1 当做正电源,P3.2 就收不到中断的触发信号。

改动:
把 P3.2,连接到 Q2 集电极,即可收到中断触发信号。
把 D1 正极,串联一个 330 欧姆的电阻,接正5V。

INT0外部中断,低电平有效,当P1.0口低电平时LED发光,可以将两个口连接,检测P1.0的低电平次数,从而计算出LED的亮的次数,即闪烁次数。

按照你题目,用了2个2位显示,实际有4位合一起的。

k3:切换计数模式/预置模式。

计数模式:LED显示计时数字,从0开始计时,直到预置最大值。

预置模式:LED显示当前预置最大值,按k1,k2可对预置值+-操作,长按k1,k2大约2秒,会进入自动加减预置值。直到再次点击k1,k2,k3任意一键停止自动。

k4:在计数模式下使用,每按下一次显示的数字加一(会在正常计时同时额外+1)。

当计数达到预置最大值,会停止计数,LEN闪烁(实际就是交替显示间隔边长),蜂鸣器响。

按键时长、LED动态显示间隔、闪烁间隔、计数速度,均可直接修改常量,需要自己改,我备注写的很详细。

电路基本按照你上图,略有修改。

#include <reg52.h>

#define uint unsigned int

#define uchar unsigned char

#define duan P0

#define on 0

#define off 1

#define SSSPEED 35 //LED交替闪烁间隔时间

#define JSPEED 5000//计数模式,速度默认数值(5000*200us=1S) 值越小计数越快

#define PREESTIME 500//按钮长按时间判定,预设500(大约2秒),需要自改,值越大,长按时间越长

sbit fm=P3^3;

sbit wei1=P3^4;

sbit wei2=P3^5;

sbit wei3=P3^6;

sbit wei4=P3^7;

sbit k1=P1^4;

sbit k2=P1^5;

sbit k3=P1^6;

sbit k4=P1^7;

uint jsSpd=JSPEED;//计时速度,默认1s一次(5000*200us)

uint ssSpd=SSSPEED;//LED交替闪烁速度

//共阳极

int delay(uint xms);

void init();

void jspause();//计数器开启/停止

void setnumYS();//设置预设数值

void numJsChange();//计数模式数字改变

void showLED();

int pressWait(uint btn);

uint g=0;

uint s=0;

uint b=0;

uint q=0;

uint count=0;

uint ispause=1;

uint numYS=0;//预设数值

uint numJS=0;//实际计时的数字

uint isMaxJs=0;//标识:计时达最大。  达最大1,否0

uint isk3press=0;//标识:k3按钮是否被点击。 点击1,否0

uint ispress1=0;//标识:k1被长按

uint ispress2=0;//标识:k2被长按

uint isbtn4=0;//标识:k4被按下

uint btnName=0;//按钮长按计时

void main()

{

init();

while(1)

{

if(ispause==1 && ispress1==1 && numYS<9999) //预置模式下,k1已长按,自动增

{

numYS++;

setnumYS();

}

if(ispause==1 && ispress2==1 && numYS>0)  //预置模式下,k2已长按,自动减

{

numYS--;

setnumYS();

}

if(isMaxJs==0 && numJS>=numYS && ispause==0) //计时模式下达最大值

{

fm=on;

ssSpd=1000;//增加LED交替间隔,实现数字闪烁

isMaxJs=1;

EA=0;

setnumYS();

numJS=0;

}

if(k1==0 ||k2==0|| k3==0) //k1\k2\k3任意一个按钮被按下,停止预置数自动增长

{

ispress1=0;

ispress2=0;

}

if(k1==0 && ispause==1)//预置模式下+

{

delay(10);

if(k1==0)

{

btnName=1;

if(pressWait(btnName))//判断连按

{

while(k1==0);

ispress1=1;

}

else if(numYS<9999)

{

numYS++;

setnumYS();

}

}

}

if(k2==0 && ispause==1)//预置模式下-

{

delay(10);

if(k2==0)

{

btnName=2;

if(pressWait(btnName))//判断连按

{

while(k2==0);

ispress2=1;

}

else if(numYS>0)

{

numYS--;

setnumYS();

}

}

}

if(k3==0)

{

delay(10);

if(k3==0)

{

while(k3==0);

fm=off;

jspause();

}

}

if(k4==0 && ispause==0)//计数模式下按下k4,k4的防抖写在中断中

{

delay(10);

if(k4==0)

{

while(k4==0);

isbtn4=1;

}

}

showLED();

}

}

void showLED()

{

uchar nums[10]={0xc0,0xf9,0xa4,0xB0,0x99,0x92,0x82,0xf8,0x80,0x98};

if(g>=0)

{

duan=nums[g];

wei4=on;

delay(ssSpd);

wei4=off;

}

if(s>0 || (s==0 && b>0))

{

duan=nums[s];

wei3=on;

delay(ssSpd);

wei3=off;

}

if(b>0 || (b==0 && q>0))

{

duan=nums[b];

wei2=on;

delay(ssSpd);

wei2=off;

}

if(q>0)

{

duan=nums[q];

wei1=on;

delay(ssSpd);

wei1=off;

}

}

void setnumYS()//设置预设数值

{

q=numYS/1000;

b=(numYS%1000)/100;

s=(numYS%100)/10;

g=numYS%10;

}

void jspause()

{

if(ispause==0 || isMaxJs==1)//关闭计时模式 / 启动预置模式

{

EA=0;

isMaxJs=0;

ispause=1;

ssSpd=SSSPEED;

ispress1=0;

ispress2=0;

setnumYS();

}

else if(ispause==1) //启动计时模式 / 关闭预置模式

{

ispause=0;

q=b=s=g=0;

numJS=0;

ssSpd=SSSPEED;

EA=1;

}

}

void init()

{

TMOD=0x02; //T0 工作模式2 自动装填8位 200us

TH0=0x38;

TL0=0x38;

EA=0;

ET0=1;

TR0=1;

wei1=off;

wei2=off;

wei3=off;

wei4=off;

}

void numJsChange()//计数模式数字改变

{

if(g==9)

{

g=0;

if(s==9)

{

s=0;

if(b==9)

{

b=0;

if(q==9)

{

q=0;

}

else

q++;

}

else

b++;

}

else

s++;

}

else

g++;

}

void ct() interrupt  1 //一次中断200us

{

if(count<jsSpd)

count++;

else

{

count=0;

numJsChange();

numJS++;

}

if(isbtn4==1)

{

isbtn4=0;

numJsChange();

numJS++;

}

}

int pressWait(uint btn)

{

uint i,j;

for(i=PREESTIME;i>0;i--)

for(j=110;j>0;j--)

{

if((k1==1 && btn==1)||(k2==1 && btn==2))

return 0;

}

return 1;

}

int delay(uint xms)

{

uint i,j;

for(i=xms;i>0;i--)

for(j=110;j>0;j--)

{

if(k1==0 || k2==0 ||k3==0)

return 1;

}

return 0;

}



这个题目,低于40估计没人做,看图貌似简单,按键+数码管显示,但还需要判断按键的长按和短按,并分别作出处理,还有闪烁处理。 程序处理还是比较麻烦的。

  • proteus中仿真时,51单片机全输出高电平是什么原因?
    答:应该是单片机没有加载HEX代码文件,就是单片机并没有执行程序。写好的程序,用keil 编译出HEX代码文件,然后双击仿真图中的单片机,如下图,点画红圈的文件夹图标,代到HEX代码文件,加进来,在画红圈的那个里应该有一个HEX文件。
  • 51单片机C程序实验LED灯:[4]花样流水灯
    答:前篇的升级版。目标:实现花样流水灯。由于花样流水灯的变换很多,所需变量太多,所以一般没有使用移动语句,而是利用数组编译好每个时刻各个灯的赋值,然后读取数据输出到端口显示。proteus电路图 画法依旧不说,给出元件清单,其实都差不多,与上篇一致51单片机 电容 瓷片电容 晶振 红色二极管 电阻 电路图...
  • 基于proteus的51单片机控制伺服电动机的仿真电路图
    答:找到供电端和PWM控制端子即可,这里给一个图仅代参考:
  • 一个关于Proteus使用C51仿真的问题,还知道的朋友指点下。谢谢!_百度知 ...
    答:先用KEIL 调试生成.HEX文件,这个文件才可以下载到单片机中,要加入PROTEUS中的单片机中,则鼠标左键双击那仿真的51单片机,然后在program file 选项中选择刚才调试生成的.HEX文件,然后确定,就可以运行了~
  • 8051单片机很简单的问题
    答:在proteus中仿真的时候,这些电路都可以忽略的,只要指定单片机的运行频率就能仿真运行。但要是实际做电路,就必须照着图上的老老实实的连接好元件了,不然无法工作。
  • proteus只能仿真51单片机对吗
    答:proteus只能仿真51单片机是不对的。根据查询相关资料信息,Proteus可以仿真支持的多种单片机,包括51、AVR、ARM、PIC、8051系列等,可以仿真其他电路,例如LED驱动器、过程控制器、定时器/计数器、开关电源、DAC等。
  • 急求解决,proteus仿真 51单片机P0口总承高阻态
    答:P0口与74LS373,8255之间采用总线连接的,每一条导线就必须加网络标号,你没有加,这样,三者之间的导线就无法对应,等于三者之间没有连接上。加网络标号的方法:先点左边工具条上的LBL,之后再将鼠标移动到导线上时,就会出现一个小叉,点左键就弹出一个对话框了,输入字符就是网络标号了。要求相互连接...
  • 51 单片机流水灯proteus仿真电路图
    答:看图
  • 在proteus中进行单片机的仿真学习,怎么向单片机(80C51)中写入程序...
    答:利用keil编写程序,C和汇编都行,生成HEX文件,在proteus在连接好线后,双击单片机,在属性对话框里面有一个文件夹图标,点那个添加刚才编译好的HEX文件,就可以了。
  • ...一体数码管实现1234轮流显示的程序代码是什么,求!proteus仿真...
    答:下面是使用AT89S51单片机控制4位一体数码管轮流显示1234的程序代码:```c include <reg51.h> // 数码管显示字型码表 unsigned char code LEDDisplay[] = { 0x3F, // 0 0x06, // 1 0x5B, // 2 0x4F // 3 };sbit DIO = P1^0; // 串行数据输入 sbit RCLK = P1^1; ...