我们公司自己做了一个攻牙机,我想着只是简单的攻牙也不行,得知道一天生产了多少数量,所以就想要设计一个
单片机程序来实现按键开关启动一次,数码管计数一次。但是程序还不能实现不良的不计数,“空”转的也不计数,这些实现起来相对而言比较复杂了(软件还好,硬件可能要求比较高)。以下程序足以满足我们小公司了,程序详见下(希望大家一起探讨)。最开始写的代码,我把按键扫描和显示功能都放到了main函数里面,但是按键一次数码管会闪一次,觉得不好,后面网上查了下资料,就把显示功能放在了定时器0里面,每隔10ms扫描一次就对了。扫描时间不能太短也不能太长,效果是实现了,但是始终还是不明白为什么。代码详见下:
- #include<reg52.h>
- #define uchar unsigned char
- #define uint unsigned int
- uchar code table[]={
- 0x3f,0x06,0x5b,0x4f,
- 0x66,0x6d,0x7d,0x07,
- 0x7f,0x6f,0x77,0x7c,
- 0x39,0x5e,0x79,0x71,
- 0x77,0x1F,0x4E,0x2D,
- 0x4F,0x47 };
- sbit button = P3^0;
- sbit dula = P2^6;
- sbit wela = P2^7;
- sbit lcden = P3^4;
- uchar count = 0;
- void buttonScan();
- void display(uchar count);
- void delayms(uint ms);
- void main()
- {
- lcden = 0;
- TMOD = 0x01;
- TH0 = (65536 - 9174)/256;
- TL0 = (65536 - 9174)%256;
- EA = 1;
- ET0 = 1;
- TR0 = 1;
- while(1)
- {
- buttonScan();
- //display(count);
- }
- }
- void buttonScan()
- {
- if(button == 0)
- {
- delayms(20);
- if(button == 0)
- {
- while(!button);
- count++;
- }
- }
- }
- void display(uchar count)
- {
- uchar w = count/10000;
- uchar q = count%10000/1000;
- uchar b = count%10000%1000/100;
- uchar s = count%10000%1000%100/10;
- uchar g = count%10;
- dula = 1;
- P0 = table[w];
- dula = 0;
- P0 = 0xff;
- wela = 1;
- P0 = 0xfe;
- wela = 0;
- delayms(1);
- dula = 1;
- P0 = table[q];
- dula = 0;
- P0 = 0xff;
- wela = 1;
- P0 = 0xfd;
- wela = 0;
- delayms(1);
- dula = 1;
- P0 = table[b];
- dula = 0;
- P0 = 0xff;
- wela = 1;
- P0 = 0xfb;
- wela = 0;
- delayms(1);
- dula = 1;
- P0 = table[s];
- dula = 0;
- P0 = 0xff;
- wela = 1;
- P0 = 0xf7;
- wela = 0;
- delayms(1);
- dula = 1;
- P0 = table[g];
- dula = 0;
- P0 = 0xff;
- wela = 1;
- P0 = 0xef;
- wela = 0;
- delayms(1);
-
- }
- void Interrupt_T0()interrupt 1
- {
- TH0 = (65536 - 9174)/256;
- TL0 = (65536 - 9174)%256;
- display(count);
- }
- void delayms(uint ms)
- {
- uint i,j;
- for(i=ms;i>0;i--)
- for(j=110;j>0;j--);
- }
复制代码
先说第一个延时的问题,在按键扫描中延时不用精确延时,跟delayms(20)的精确度关系不大,也不是说这个延时时间不长,而是在主函数main()里,可以不使用这种delayms(20)
void delayms(uint ms)
{
uint i,j;
for(i=ms;i>0;i--)
for(j=110;j>0;j--);
}
这个延时就是利用CPU消耗一些指令来累计时间达到延时的效果,简单有效,缺点就是在执行这些指令的同时CPU其他任务全部照顾不到,根本不能去执行(除了中断里的,但是中断也不能写一大堆任务)。所以利用定时器中断来做这个延时就OK了,单片机的定时器外设和单片机主程序的运行是真正的并行执行,定时器计时时间到了告诉主程序一下,主程序只需要去处理一下就可以。定时器的消抖跟你理解的意思是差不多的,论坛中有很多参考代码我就不写了。
第二个问题,我刚才才发现你的display里面是一次性全部做的显示,并且中间通过了一些小延时做了间隔,这个延时也要删掉,原因同上。数码管的显示方法有很多,要看程序框架如何规划了,我说一个你试试。
void display(uchar count)里的count可以认为是缓冲数据buf,每次执行显示的时候传入缓冲数据去做显示就可以了。按键或者其他外部因素仅仅是更新这个数据buf,并不需要立即就去调用显示。
显示可以按照一个封闭的状态机的模式去写,扔在定时器中断里,比如10ms中断一次,第一次也就是状态1显示第一位数码管,第二次中断显示第二位数码管,第三次中断……状态就是一个unsigned char 型的变量做指示就可以的。这样中断每次执行的代码就是显示其中一位数码管并且更新状态,代码量也不大,也不应delay去做延时了。
我居然打了这么多字……
10ms中断一次显示一位,能不能保证数码管都同时点亮额。那display函数里面应该还有可以控制数码管显示位的才行,或者说每隔10ms,数码管显示控制位循环左移一位。。。。。。
10Ms能不能就实际检验一下好了,要显示的位数越多,这个中断时间就要设置的越短一些,满足人眼的视觉暂留最低显示频率。display函数里面为了实现每次进去显示不同的位,所以肯定需要一个只是标志位,static也可以,全局变量也可以,然后每次进去执行一次++,然后判别,去显示对应的位就可以了。
额,明白。非常感谢你!
一周热门 更多>