/*课程设计终于弄完了,总想共享一下源代码,*/
/*遂在51hei上贴出来,大家看有什么问题吧~如果哪里写得不好还请众亲们指出哦~*/
/*---附:实物图--*/
/*===Chip:STC12C5A32S2======================*/ /*===Software:Keil 4 C51========================*/ /*===Author:梁鹏===================================*/ /*===Organization:广西民族大学07自动化===========*/ /*===Date:2010年05月26日======================*/ /*===Version:1.0===============================*/ /*===声明:本程序属原创作品,仅供学习交流,不得用于 商业用途,如需转载,请注明出处,谢谢合作!======*/ /*--------文件名:Supper_clock.C-----------*/ #include#include #include #include #include #include #include #define KEY_PORT P3 //按键端口 sbit MODE_KEY=KEY_PORT^3; //模式选择键 sbit HOU_KEY=KEY_PORT^4; //时增一键 sbit MIN_KEY=KEY_PORT^5; //分增一键 sbit CLKON_KEY=KEY_PORT^6; //闹钟ON/OFF控制键 sbit BUZ_OUT=P2^3; //蜂鸣器输出口 sbit DS18B20_PORT=P2^2; //DS18B20数据接口 sbit LAMP_PORT=P2^1; //继电器控制口 /*=========子函数的声明==============*/ void init(); //系统初始化 void display_time(); //显示时间的子函数 void display_clk(); //显示闹钟的子函数 void judge(); //判断秒增量是否大于限定值 void clk_judge(); //检测是否到达闹铃时间 void beep(); //蜂鸣器响子函数 void buzzer(); //闹铃响子函数 void get_adc_data(); //AD采样数据及处理子函数 void serial_judge(); //串口数据处理子函数 //void delay_ms(uint); //MS级延时子程序 /*==========变量的声明===============*/ bit pause_flags=0,clk_flags=0,clk_on=0,clk_on_or_off=1,flash_flags=0; bit clk1_off=0,clk2_off=0,clk3_off=1; //各闹钟是否开启的变量 bit serial_finish=0; //串口接收结束标志位 uchar com_send,com_get; //串口通信相关变量 uchar hour,min,sec; //时间变量 uchar dis_hour_clk,dis_min_clk; //用于显示闹钟时间的变量 uchar hour_clk1,min_clk1; //闹钟1有关变量 uchar hour_clk2,min_clk2; //闹钟2有关变量 uchar hour_clk3,min_clk3; //闹钟3有关变量 uchar mode_flags; //状态标志位 uint ovf_cnt=0; //定时中断次数纪录 uchar first_on_flags=0; //掉电保存变量 uint ad_data_channal7; //ADC采样数据1 uint ad_data_channal6; //ADC采样数据2 float vol_data,vol_data1; //采样电压值转换 uchar send_temp[10],send_i_cnt; //发送缓冲 /*===================================*/ /*************************************/ /*************************************/ void main() { init(); //程序初始化 LCD_init(); //1602初始化 init_24c02(); //24C02掉电保存初始化 serial_port_init(); //串行通信初始化 first_on_flags=read_add_24c02(2); beep(); if(first_on_flags!=33){ //首次上电,写首次上电标记 write_add_24c02(2,33); } else{ hour =read_add_24c02(3); min =read_add_24c02(4); hour_clk1 =read_add_24c02(5); min_clk1 =read_add_24c02(6); hour_clk2 =read_add_24c02(7); min_clk2 =read_add_24c02(8); hour_clk3 =read_add_24c02(9); min_clk3 =read_add_24c02(10); clk1_off =read_add_24c02(11); clk2_off =read_add_24c02(12); clk3_off =read_add_24c02(13); } while(1) { get_adc_data(); //获取AD值 display_time(); //时间显示 judge(); } } /*************************************/ /*************************************/ void init() { hour=6;min=29;sec=55; //初始化系统时间 hour_clk1=6;min_clk1=30; //初始化系统时钟 hour_clk2=6;min_clk2=31; hour_clk3=6;min_clk3=32; AURX1|=0x04; //AD转换结果存储格式控制 P1ASF=1<<7; //模拟通道选择 P1ASF=1<<6; BUZ_OUT=1; //蜂鸣器不响 LAMP_PORT=0; //继电器不吸合 P0=255; //LCD数据口状态初始化 TCON|=0x01; //INT0下降沿触发 EX0=1; //外部中断0中断允许 INT0=1; //P3.2置位 TMOD|=0x02; //定时器0工作在方式1 TH0=5; //250us计数初值高八位 TL0=5; //250us计数初值低八位 ET0=1; //定时器0中断允许 EA=1; //全局中断允许 TR0=1; //定时器0开启 } /**************************/ void display_time() { Locate(1,1); if((mode_flags==1)&&flash_flags) LCD_str_(" "); else LCD_str_("Time: "); //调整模式下,“Time:”闪烁 LCD_2char(hour); //显示时 if(flash_flags&&(!pause_flags)) LCD_char(' '); else LCD_char(':'); //非暂停模式下,“:”闪烁 LCD_2char(min); //显示分 if(flash_flags&&(!pause_flags)) LCD_char(' '); else LCD_char(':'); LCD_2char(sec); //显示秒 LCD_str_(" "); if(mode_flags>0){display_clk();} //暂停时,显示闹钟 else{ Locate(2,1); LCD_float_(vol_data,2); LCD_str_("V "); LCD_4char(ad_data_channal7); LCD_str_(" "); Locate(2,13); if(!clk_on_or_off)LCD_str_(" ON"); //全局显示闹钟的ON/OFF else LCD_str_(" OFF"); } } void display_clk() { Locate(2,1); if(flash_flags&&(mode_flags>1))LCD_str_(" ");//调整模式下,“Clkx:”闪烁 else{ if(mode_flags>0&&mode_flags<3) {LCD_str_("Clk1: ");dis_hour_clk=hour_clk1;dis_min_clk=min_clk1;} if(mode_flags==3) {LCD_str_("Clk2: ");dis_hour_clk=hour_clk2;dis_min_clk=min_clk2;} if(mode_flags==4) {LCD_str_("Clk3: ");dis_hour_clk=hour_clk3;dis_min_clk=min_clk3;} }//根据不同的状态,显示不同的闹钟时间 LCD_2char(dis_hour_clk); LCD_char(':'); LCD_2char(dis_min_clk); LCD_str_(" "); Locate(2,13); if(mode_flags<2){ //调整时间的模式下,全局显示闹钟的开或者关 if(!clk_on_or_off)LCD_str_(" ON"); else LCD_str_(" OFF"); } /*根据不同调闹和开关状态显示指定闹钟的开关*/ if( (mode_flags==2&&!clk1_off)|| (mode_flags==3&&!clk2_off)|| (mode_flags==4&&!clk3_off)) LCD_str_(" ON"); else LCD_str_(" OFF"); } /**************************/ void judge() { if(sec>=60){ //秒增量是否超过限定的判断 sec=0; if(++min>=60){ min=0; if(++hour>=24)hour=0; } } if(mode_flags<1) {pause_flags=0;} //是否暂停的判断 else {pause_flags=1;} /*以下为时限判断*/ if(hour>23)hour=0; //调整时增量是否超限的判断 if(min>59)min=0; if(hour_clk1>23) hour_clk1=0; if(min_clk1>59) min_clk1=0; if(hour_clk2>23) hour_clk2=0; if(min_clk2>59) min_clk2=0; if(hour_clk3>23) hour_clk3=0; if(min_clk3>59) min_clk3=0; /*以上为时限判断*/ /*闹铃是否开启判断*/ if(!clk1_off||!clk2_off||!clk3_off) clk_on_or_off=0; else clk_on_or_off=1; /*蜂鸣器是否响判断*/ if(!pause_flags&&!clk_on_or_off){clk_judge();}
//非暂停模式下,闹钟打开情况下,检测闹钟 if (!clk_flags) //使用clk_flags和clk_on检测是否闹铃 { if(((hour!=hour_clk1)||(min!=min_clk1))&& ((hour!=hour_clk2)||(min!=min_clk2))&& ((hour!=hour_clk3)||(min!=min_clk3)))clk_flags=1;clk_on=0;BUZ_OUT=1; } else { if(clk_on){buzzer();BUZ_OUT=1;}} //满足闹铃响所需的所有条件,发出闹铃声 /*以下为EEPROM操作*/ if(sec==5){//一分钟存一次 write_add_24c02(3,hour); LCD_delay(100); write_add_24c02(4,min); LCD_delay(100); write_add_24c02(5,hour_clk1);LCD_delay(100); write_add_24c02(6,min_clk1);LCD_delay(100); write_add_24c02(7,hour_clk2);LCD_delay(100); write_add_24c02(8,min_clk2);LCD_delay(100); write_add_24c02(9,hour_clk3);LCD_delay(100); write_add_24c02(10,min_clk3);LCD_delay(100); write_add_24c02(11,clk1_off);LCD_delay(100); write_add_24c02(12,clk2_off);LCD_delay(100); write_add_24c02(13,clk3_off);LCD_delay(100); } /*以下为串口接收完成处理*/ if(serial_finish){ serial_judge(); //串口数据处理 serial_finish=0; //清0接收结束标志 } } void clk_judge() { if(((hour==hour_clk1)&&(min==min_clk1)&&!clk1_off)|| ((hour==hour_clk2)&&(min==min_clk2)&&!clk2_off)|| ((hour==hour_clk3)&&(min==min_clk3)&&!clk3_off))clk_on=1; else{clk_on=0;} } void beep() { uchar beep_cnt; for(beep_cnt=0;beep_cnt<60;beep_cnt++){ BUZ_OUT=0; LCD_delay(2); BUZ_OUT=1; //确保每次闹铃结束后蜂鸣器不响 LCD_delay(1); } } void buzzer() { uchar i; for(i=0;i<2;i++){ beep(); LCD_delay(500); } } /**************************/ void INT0_isr() interrupt 0 { LCD_delay(30); if(!INT0){ beep(); if(!MODE_KEY){ /*MODE_KEY*/ if(++mode_flags>=5){ pause_flags=0; mode_flags=0; flash_flags=0; } } if(!HOU_KEY){ /*HOU_KEY*/ if(mode_flags==1){++hour;} if(mode_flags==2){++hour_clk1;} if(mode_flags==3){++hour_clk2;} if(mode_flags==4){++hour_clk3;} } if(!MIN_KEY){ /*MIN_KEY*/ if(mode_flags==1){++min;} if(mode_flags==2){++min_clk1;} if(mode_flags==3){++min_clk2;} if(mode_flags==4){++min_clk3;} } if(!CLKON_KEY){ if(mode_flags==2)clk1_off=~clk1_off; if(mode_flags==3)clk2_off=~clk2_off; if(mode_flags==4)clk3_off=~clk3_off; } clk_flags=0; /*闹钟发生时,按下任意键停止*/ } INT0=1; } //================================ void timer0_ovf_isr() interrupt 1 { if(ovf_cnt==1846)flash_flags=~flash_flags; if(++ovf_cnt==3691){ //12MHZ自动重载4000次为1s,11.071692HZ下为3691 ovf_cnt=0; if(!pause_flags)sec++; //非暂停模式下,秒增一 flash_flags=~flash_flags; } //每间隔半秒flash_flags取反一次 } //================================ void receive_uart() interrupt 4 { if(RI){ RI=0; com_get=SBUF; serial_finish=1; } else{ TI=0; } } //================================ void serial_judge() { switch(com_get){ case 0x31: Uart_send_int(hour,2); Uart_send_(':'); Uart_send_int(min,2); Uart_send_(':'); Uart_send_int(sec,2); Uart_send(" "); break; case 0x32: Uart_send("The brightness:"); Uart_send_flt(vol_data1,3); Uart_send("V "); break; case 0x33: Uart_send("The voltage:"); Uart_send_flt(vol_data,3); Uart_send("V "); break; case 0x34:LAMP_PORT=!LAMP_PORT;break; default : Uart_send("Command Error! "); Uart_send("1->To get the system time. "); Uart_send("2->To get the brightness. "); Uart_send("3->To get the voltage. "); Uart_send("4->To control the 220V light ON/OFF. "); } com_get=0; } //================================ void get_adc_data() { ad_data_channal7=ADC_GET(7); LCD_delay(1000); ad_data_channal6=ADC_GET(6); LCD_delay(1000); vol_data=(float)ad_data_channal6/1024; vol_data*=5.0; vol_data1=(float)ad_data_channal7/1024; vol_data1*=5.0; } //================================ /*void delay_ms(uint xms) { uint i,j; for(i=0;i< xms;i++) for(j=0;j< 990;j++); }*/ //================================
上面是主程序,已经调试通过,下面是部分.h文件,限于篇幅,完整的代码大家可从下面的链接下载
http://www.51hei.com/ziliao/file/naozhong1602.rar , 只要更改一下端口便可适应于其他电路.
/*--------文件名:CLK_LCD1602.H-------*/ #ifndef _CLK_LCD1602_H_ #define _CLK_LCD1602_H_ #define uint unsigned int #define uchar unsigned char #define LCM_P P2 //LCD的控制端口 #define LCM_DATA P0 //LCD的数据口 #define LCM_RS_0 LCM_P&=~(1<<5) #define LCM_RS_1 LCM_P|= 1<<5 #define LCM_RW_0 LCM_P&=~(1<<6) #define LCM_RW_1 LCM_P|= 1<<6 #define LCM_EN_0 LCM_P&=~(1<<7) #define LCM_EN_1 LCM_P|= 1<<7 /*========================================*/ #define LCD_str(x,y,s) Locate(x,y);LCD_str_(s); #define LCD_float(x,y,flt) Locate(x,y);LCD_float_(flt); /*========================================*/ void LCD_init(); //LCM1602的初始化函数,在使用1602之前都必须调用 void Locate(uchar,uchar); //显示定位函数 void LCD_half(uchar); //送半字节函数 void LCD_char(uchar); //写一个字符函数 void LCD_2char(uchar); //显示两个字符 void LCD_4char(uint); //显示四个字符 void LCD_cmd(uchar); //写命令函数 void LCD_str_(uchar str[]); //写字符串数据函数 void LCD_float_(float,uchar); //写浮点数据函数 void LCD_delay(uint); //延时子函数 /******************************************/ /*-函数功能:液晶使用初始化---------------*/ /*-入口参数:无*/ /******************************************/ void LCD_init() { LCD_cmd(0x01); LCD_delay(10); LCD_cmd(0x28); //4位数据、双行显示、5*7(0x38为八位) LCM_EN_1;_nop_();_nop_();_nop_(); LCM_EN_0; /*此处必须加上这两句*/ LCD_delay(10); LCD_cmd(0x28); LCD_cmd(0x06); LCD_cmd(0x0c); LCD_cmd(0x01); LCD_delay(10); } /******************************************/ /*-函数功能:显示数据定位函数-------------*/ /*-入口参数:行坐标x、列坐标y-------------*/ /******************************************/ void Locate(uchar x,uchar y) { x&=0x01; LCD_cmd((x==0)?(y+0xbf):(y+0x7f)); } /******************************************/ /*-函数功能:送半字节函数-----------------*/ /*-入口参数:要写到液晶指令或数据寄存器的-*/ /* 字节的高四位或低四位---------*/ /******************************************/ void LCD_half(uchar dataw_) { LCM_DATA=(LCM_DATA&0x0f)|(dataw_); LCM_EN_1;_nop_();_nop_();_nop_(); LCM_EN_0; LCD_delay(1);//实际使用中加上10ms的延时 } /******************************************/ /*-函数功能:写一位数据函数---------------*/ /*-入口参数:数据内容---------------------*/ /******************************************/ void LCD_char(uchar dataw) { LCM_RS_1;LCM_RW_0;LCM_EN_0;_nop_(); LCD_half(dataw&0xf0); LCD_half(dataw<<4); } /*========================================*/ void LCD_cmd(uchar cmd) { LCM_RS_0;LCM_RW_0;LCM_EN_0;_nop_(); LCD_half(cmd&0xf0); LCD_half(cmd<<4); } /*========================================*/ void LCD_str_(uchar *str) { while(*str)LCD_char(*str++); } /*========================================*/ void LCD_2char(uchar num_2) { uchar num_temp; num_temp=num_2/10; LCD_char(num_temp+0x30); num_temp=num_2%10; LCD_char(num_temp+0x30); } void LCD_4char(uint num_4) { uint num_tmp; num_tmp=num_4/1000; LCD_char(num_tmp+0x30); num_tmp=num_4/100; num_tmp=num_tmp%10; LCD_char(num_tmp+0x30); num_tmp=num_4/10; num_tmp=num_tmp%10; LCD_char(num_tmp+0x30); num_tmp=num_4%10; LCD_char(num_tmp+0x30); } /*========================================*/ void LCD_float_(float flt,uchar n) { uchar counter=0,num_str[10],neg_flags=0,n_; long int num; n_=n; while(n_){n_--;flt*=10;} num=flt; if(!num)num_str[counter++]=0x30; if(num<0){num=-num,neg_flags=1;} while(num!=0) { num_str[counter++]=num%10+0x30; num/=10; if(counter==n)num_str[counter++]='.'; } if(neg_flags){num_str[counter++]='-';} while(counter--)(LCD_char(num_str[counter])); } /*========================================*/ void LCD_delay(uint xus) { uint i=0,j=0; for(i=0;i< xus;i++) for(j=0;j< 123;j++); } /*========================================*/ #endif /*----------文件名:SERIAL_UART.H-------------*/ #ifndef _SERIAL_UART_H_ #define _SERIAL_UART_H_ #define uint unsigned int #define uchar unsigned char #define Fosc 11059200 #define Baud 9600 #define Reload (256-((2*Fosc)/12/32/Baud)) //SMOD==1 //#define Reload (256-(Fosc/12/32/Baud)) //SMOD==0 sfr AUXR = 0x8e; sfr AUXR1 = 0xA2; sfr BRT = 0x9c; //独立波特率发生器重装载寄存器 void serial_port_init(); void Uart_send_(uchar); void Uart_send(uchar *); void Uart_send_int(int,uchar); /*=================================*/ void serial_port_init() { SCON=0x50; //串口控制第七第六方式控制,第四位允许接收 PCON|=0x80; //第七位SMOD AUXR=0x11; //第四位置位允许独立波特率发生器运行 BRT=Reload; //独立波特率发生器赋值 AUXR1=0x80; //选择串口位置 ES=1; //允许串口中断 // EA=1; //允许全局中断 } /*========================================*/ void Uart_send(uchar *str) { while(*str)Uart_send_(*str++); } /*========================================*/ void Uart_send_int(int num_send,uchar n_cnt) { uchar num_str[7],num_cnt=0,neg_flags=0; if(num_send<0){num_send=-num_send;neg_flags=1;} do{ num_str[num_cnt++]=num_send%10+0x30; num_send/=10; }while(num_send); while(num_cnt< n_cnt)num_str[num_cnt++]=0x30; if(neg_flags)num_str[num_cnt++]='-'; while(num_cnt--)Uart_send_(num_str[num_cnt]); } /*========================================*/ void Uart_send_flt(float flt,uchar n) { long int num; uchar counter=0,num_str[10],n_temp; bit small_1=0,neg_flags=0; n_temp=n; if(flt< 1.0)small_1=1; while(n_temp){n_temp--;flt*=10;} num=flt; if(num< 0){num=-num,neg_flags=1;} do{ num_str[counter++]=num%10+0x30; num/=10; if(counter==n)num_str[counter++]='.'; }while(num!=0); if(small_1)num_str[counter++]=0x30; if(neg_flags){num_str[counter++]='-';} while(counter--)(Uart_send_(num_str[counter])); } /*========================================*/ void Uart_send_(uchar dat) { ES=0; TI=0; SBUF=dat; while(!TI); TI=0; ES=1; } /*========================================*/ #endif
上一篇:单片机连续按键检测C51程序
下一篇:单片机液晶显示器程序
推荐阅读最新更新时间:2024-03-16 13:11
设计资源 培训 开发板 精华推荐
- Littelfuse|一个芯片实现过压、过流、短路和浪涌保护
- 了解 PI 全新 PowiGaN 开关电源 IC ,答题赢好礼!
- TI汽车方案拼图,你敢挑战吗?
- ADI & 世健 带您了解 化学分析和分析仪器应用方案
- Molex紧凑型Type-C连接器 为您的设计节约宝贵空间!下载好礼送!
- Silicon Labs EFM32PG22开发套件 传你所思 创你所想!申请进行时!
- TI E2E™ 中文社区8周年,8本技术合集送给热爱技术的你!
- 有奖直播:基于DLP® Pico™技术的TI桌面级DLP 3D打印、3D扫描及工业显示应用
- 观看TI无线连接专场研讨会 下载TI无线电子书有礼
- 索取海报:泰克矢量网络分析仪基础知识