功能介绍:
1. 三位数码管显示数据00.0~99.9
2.一次计时中可记录4个数据(按一下存一次数据),存在EEPROM中(若按了五下,则会丢失第一次按下的数据)
3.两个按键功能为,暂停,启动,存数,取数,翻页(查看下一个存的时间)
硬件连接
1.两个573控制数码管的段选,位选,573数据输入端接P0口,段选,位选分别接,P1^1,P1^0
2.P1^3,P1^4分别接EEPROM的时钟线(SCL),数据线(SDA)
3.独立按键K0.K1分别接外部中断0,1接口 P3^2,P3^3
4.使用定时器0的工作方式2
介绍
1.K0分 短按(小于2s),长按(大于2s),短按为定时器启停,长按进入读取数据程序
2.K0未长按,按一下k1存入EEPROM中一个数,最多存四个数,按五下则丢失第一个数
3.k0长按后,k1存数功能失效,变成翻页功能,按一下数码管显示下一个保存的数据
4.单片机复位键,控制程序重头执行。
下面为我编写的程序
#include
typedef unsigned int uint; //对数据类型进行声明定义
typedef unsigned char uchar;
void delayms(uint); //延时函数声明
void display(uchar,uchar);//数码管显示函数声明
void delay(); //短暂延时
void start(); //开始信号
void stop(); //停止信号
void respons(); //应答
void init(); //初始化 两条线都拉高
void write_byte(uchar); //写一个字节
uchar read_byte(); //读一个字节,带带返回值的函数
void write_add(uchar,uchar);//写入 指定地址 数据
uchar read_add(uchar); //读出 指定地址 数据
uchar num0,num1;//存放eeprom中的数据 ,十分位,整数位
uint cc=0; //定义定时器0溢出次数
uchar zant=0; //暂停开始标志位,外部中断0控制
uchar fanye=0; //定义外部中断1 存数次数标志位,翻页次数
uchar gg=0; //定义存第几数 ,翻到第几个数
uchar qc=0; //定义长按外部中断0 取出保存的数
uchar code led1[]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F} ; //定义0~9的数组
uchar code led2[]={0xbF,0x86,0xdB,0xcF,0xe6,0xeD,0xfD,0x87,0xfF,0xeF} ;//0~9有小数点
uchar shi=0; // 定义数码管十位
uchar ge=0; //定义数码管个位
sbit wei=P1^0; //定义位选端
sbit duan=P1^1; //定义段选端
sbit int0=P3^2; //外部中断0,端口
sbit int1=P3^3; //外部中断1,端口
sbit scl=P1^3; //时钟线
sbit sda=P1^4; //数据线
void main()
{
IT0=1; //外部0低电平触发
EX0=1; //开外部中断0
IT1=1; //外部1低电平触发
EX1=1; //开外部中断1
ET0=1; //开定时器0
EA=1; // 开总中断
IP=0x04; //外部1优先
TMOD=0x02; //定义定时器0,八位自动装填
TH0=0x1a; //晶振11.0592,初值26,时间250us
TL0=0x1a;
while(1)
{
if(num0>=10) //次数达到满
{
num0=0;
num1++;
if(num1>=100)
{num1=0;}
}
display(num0,num1); //显示LED函数
if(fanye==1) //存数据 判断外部中断1是否按下
{
switch(gg) //通过gg的值“按第几下”,存到不同的位置
{
case 1:
init();
write_add(0x01,num0);
init();
write_add(0x02,num1);
break;
case 2:
init();
write_add(0x03,num0);
init();
write_add(0x04,num1);
break;
case 3:
init();
write_add(0x05,num0);
init();
write_add(0x06,num1);
break;
case 4:
init();
write_add(0x07,num0);
init();
write_add(0x08,num1);
break;
}
fanye=0; //退出存数,直到下次外部中断1到来
}
if(qc==1) //判断外部中断0长按键 开始读存进EEPROM的数据
{
while(1)
{
switch(gg) //通过判断外部中断1,按键次数gg实现翻页功能
{ //因为又进入到现在这个大循环 所以外部中断1中的
case 1: //fanye=1;不能在返回上面控制 存第几个数据了,相当与无用
init();
num0=read_add(0x01);
init();
num1=read_add(0x02);
break;
case 2:
init();
num0=read_add(0x03);
init();
num1=read_add(0x04);
break;
case 3:
init();
num0=read_add(0x05);
init();
num1=read_add(0x06);
break;
case 4:
init();
num0=read_add(0x07);
init();
num1=read_add(0x08);
break;
}
display(num0,num1); //程序将在此一直判断gg,实现翻页,显示存的数据
} //若想再次计数,按下单片机的RST复位键,程序重头执行
}
}
}
void wbzd0()interrupt 0
{
uchar p=0; //判断按键按下的时间
delayms(20); //消抖
if(int0==0)
{
while(!int0&&p<250) //最多按25秒
{ p++;
delayms(100);}
if(p>15) //长按2秒 调出数据,
{
qc=1; //主程序开始 读eeprom数据
TR0=0; //停止定时器工作
}
else // 低于两秒
{
zant=~zant;
TR0=zant; //来回反转实现定时器0的启停
}
}
}
void wbzd1()interrupt 2
{
delayms(20); //消抖
if(int1==0)
{
fanye=1; //主程序进入存数据,后跳出,直到下次按键按下
gg++; //用于判断存数据位,和读数据位,在不同的位置实现的功能不同
if(gg==5)
{gg=1;}
while(!int1); //按键不松,程序不往下执行
}
}
void T0_time()interrupt 1 //八位自动装填
{
cc++;
if(cc>=400) //0.1秒到来
{
cc=0;
num0++;
}
}
void display(uchar num0,uchar num1) //显示函数
{
shi=num1/10;
ge=num1%10;
duan=1;
P0=led1[num0]; //送入十分位位段选数据
duan=0;
P0=0xff;
wei=1;
P0=0xfb; //选择左侧第3位位选打开
wei=0;
delayms(2);
duan=1;
P0=led2[ge]; //送入个位段选数据
duan=0;
P0=0xff;
wei=1;
P0=0xfd; //选择左侧第2位位选打开
wei=0;
delayms(2);
duan=1; //打开段选
P0=led1[shi]; //送入段选数据 十位
duan=0; //关闭段选
P0=0xff; //消影
wei=1; // 打开位选
P0=0xfe; // 打开最右侧位选
wei=0; // 关闭位选
delayms(2); // 延时一会 便于观察
}
void delayms(uint x) //延时函数
{
uint p,q;
for(p=x;p>0;p--)
for(q=110;q>0;q--);
}
void delay()
{;;}
void start() //开始信号
{
sda=1;
delay();
scl=1;
delay();
sda=0;
delay();
}
void stop() //停止信号
{
sda=0;
delay();
scl=1;
delay();
sda=1;
delay();
}
void respons() //应答
{
uchar i;
scl=1;
delay();
while((sda==1)&&(i<250)) //防止一直没有收到应答,程序停止
{i++;}
scl=0;
delay();
}
void init() //初始化 两条线都拉高
{
sda=1;
delay();
scl=1;
delayms(1); //加延时,否则用不了
}
void write_byte(uchar date) //写一个字节
{
uchar i,temp;
temp=date;
for(i=0;i<8;i++)
{
temp=temp<<1;
scl=0;
delay();
sda=CY; //最高位移入PSW寄存器中CY位中
delay(); //将CY中的值赋给sda
scl=1;
delay();
}
scl=0;
delay();
sda=1;
delay();
}
uchar read_byte() //读一个字节,带带返回值的函数
{
uchar i,k;
scl=0;
delay();
sda=1;
delay();
for(i=0;i<8;i++)
{
scl=1;
delay();
k=(k<<1)|sda; //将K左移一位后与sda进行或运算,
scl=0; //依次把8个位放入一个字节中来完成接收
delay();
}
return k; //返回值为k
}
void write_add(uchar address,uchar date) //写入 指定地址 数据
{
start();
write_byte(0xa0); //接下来为写操作
respons();
write_byte(address); //送入地址
respons();
write_byte(date); //写入数据
respons();
stop();
delayms(1); //加延时,否则用不了
}
uchar read_add(uchar address) //读出 指定地址 数据
{
uchar date;
start();
write_byte(0xa0); //接下来为写操作
respons();
write_byte(address); //送入地址 ,下面读数据的地址
respons();
start();
write_byte(0xa1); //接下来为读操作 存储器发送数据
respons();
date=read_byte(); //把SDA数据线上的数据用date存
stop();
return date;
delayms(1); //加延时,
}
上一篇:基于单片机的智能密码锁
下一篇:C51 实现动态内存分配
推荐阅读最新更新时间:2024-03-16 16:24