利用AT89C51单片机控制字符型LCD显示器制作一个简易的倒数计数器,可用来煮方便面、煮开水或小睡片刻等。先进行一小段时间倒计数,当倒计数为0时,则发出一段音乐声响,通知倒计数时间到,去做该做的事。 定时闹钟采用宇字符型LCD(16×2)显示器,显示格式为“TIME分分:秒秒”。 程序运行后LCD上显示倒计数的时间为“30:00”分钟,此时按一下K5即可开始倒计时。如果要改变为其他的倒计时时间,直接可按一下其中一个按键设定一个固定的倒计时时间: K2--设置倒计数的时间为5分钟,显示“05:00”。 K3--设置倒计数的时间为10分钟,显示“10:00”。 K4--设置倒计数的时间为20分钟,显示“20:00”。 注意:只能按一下其中一个按键,设定一次,然后再按一下K5,即开始倒计时。 也可在LCD上显示倒计数的时间为“30:00”分钟的基础上进行增1分钟或减1分钟的倒计时时间调整,即在程序运行后,先按一下K1,再按一下K2(增1分钟)或按一下K3(减1分钟),直到所设定的倒计时时间,然后按一下K5,即开始倒计时。可调整的倒计数的时间范围为1∽60分钟。 倒计时工作时,指示灯LED闪动,表示倒计时运行。 根据上面描述设计一个代码

时间: 2024-02-27 14:51:47 浏览: 21
很好,下面是对应的代码实现,代码中注释可能会有帮助: ```c #include <reg51.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #define uchar unsigned char #define uint unsigned int #define LCD_RS P2_0 #define LCD_RW P2_1 #define LCD_EN P2_2 #define LCD_DATA P0 #define K1 P3_2 #define K2 P3_3 #define K3 P3_4 #define K4 P3_5 #define K5 P3_6 uchar code music[] = {0x00, 0x00, 0x3C, 0x00, 0x7E, 0x00, 0xFF, 0x00, 0x7E, 0x00, 0x3C, 0x00, 0x00, 0x00, 0x00, 0x00}; uchar time_min = 30, time_sec = 0; // 倒计时初始时间为30:00 uchar set_time_min = 0, set_time_sec = 0; // 手动设置的倒计时时间 uchar set_flag = 0; // 标志是否设置了倒计时时间 uchar adjust_flag = 0; // 标志是否正在进行时间调整 uchar adjust_min = 0; // 调整倒计时的分钟数 sbit LED = P1_0; /* 音乐发声函数 */ void music_play() { uint i, j; int t = 10000; // 调整音乐速度的参数 for(i = 0; i < 16; i++) { for(j = 0; j < t; j++); // 调整音乐速度 P1 = music[i]; for(j = 0; j < t; j++); // 调整音乐速度 } } /* LCD写入命令 */ void lcd_write_cmd(uchar cmd) { LCD_RS = 0; LCD_RW = 0; LCD_DATA = cmd; LCD_EN = 1; LCD_EN = 0; } /* LCD写入数据 */ void lcd_write_data(uchar dat) { LCD_RS = 1; LCD_RW = 0; LCD_DATA = dat; LCD_EN = 1; LCD_EN = 0; } /* LCD初始化 */ void lcd_init() { lcd_write_cmd(0x38); // 16x2显示,5x7点阵,8位数据接口 lcd_write_cmd(0x0c); // 显示器开,光标关闭,不闪烁 lcd_write_cmd(0x06); // 光标右移,字符不移动 lcd_write_cmd(0x01); // 清屏 } /* LCD显示字符 */ void lcd_display_char(uchar x, uchar y, uchar data) { uchar addr; if(y == 0) { addr = 0x80 + x; } else { addr = 0xc0 + x; } lcd_write_cmd(addr); lcd_write_data(data); } /* LCD显示字符串 */ void lcd_display_str(uchar x, uchar y, uchar *str) { uchar addr; if(y == 0) { addr = 0x80 + x; } else { addr = 0xc0 + x; } lcd_write_cmd(addr); while(*str != '\0') { lcd_write_data(*str++); } } /* 延时函数 */ void delay(uint ms) { uint i, j; for(i = 0; i < ms; i++) { for(j = 0; j < 1141; j++); } } /* 读取按键 */ uchar key_scan() { if(K1 == 0) { delay(10); // 延时去抖动 if(K1 == 0) { while(!K1); // 等待按键释放 return 1; } } if(K2 == 0) { delay(10); if(K2 == 0) { while(!K2); return 2; } } if(K3 == 0) { delay(10); if(K3 == 0) { while(!K3); return 3; } } if(K4 == 0) { delay(10); if(K4 == 0) { while(!K4); return 4; } } if(K5 == 0) { delay(10); if(K5 == 0) { while(!K5); return 5; } } return 0; } /* 显示倒计时时间 */ void display_time() { uchar min_high = time_min / 10; uchar min_low = time_min % 10; uchar sec_high = time_sec / 10; uchar sec_low = time_sec % 10; lcd_display_char(0, 1, 'T'); lcd_display_char(1, 1, 'I'); lcd_display_char(2, 1, 'M'); lcd_display_char(3, 1, 'E'); lcd_display_char(4, 1, min_high + '0'); lcd_display_char(5, 1, min_low + '0'); lcd_display_char(6, 1, ':'); lcd_display_char(7, 1, sec_high + '0'); lcd_display_char(8, 1, sec_low + '0'); } /* 手动设置倒计时时间 */ void set_time() { uchar key; while(1) { key = key_scan(); if(key == 2) { set_time_min = 5; set_time_sec = 0; break; } if(key == 3) { set_time_min = 10; set_time_sec = 0; break; } if(key == 4) { set_time_min = 20; set_time_sec = 0; break; } } set_flag = 1; } /* 调整倒计时时间 */ void adjust_time() { uchar key; while(1) { key = key_scan(); if(key == 2) { adjust_min++; if(adjust_min > 59) { adjust_min = 59; } } if(key == 3) { adjust_min--; if(adjust_min < 0) { adjust_min = 0; } } if(key == 5) { break; } set_time_min = time_min + adjust_min; set_time_sec = time_sec; if(set_time_min > 59) { set_time_min = 59; } if(set_time_min < 1) { set_time_min = 1; } display_time(); } } /* 倒计时主程序 */ void countdown() { uint i; uchar key; while(1) { // 每秒钟更新一次倒计时时间 for(i = 0; i < 1000; i++) { delay(1); key = key_scan(); if(key == 5) { return; } if(key == 1) { adjust_flag = 1; adjust_time(); adjust_flag = 0; break; } } if(adjust_flag) { continue; } if(time_min == 0 && time_sec == 0) { music_play(); LED = 1; return; } if(time_sec == 0) { time_min--; time_sec = 59; } else { time_sec--; } display_time(); LED = ~LED; // LED灯闪烁 } } /* 主函数 */ void main() { lcd_init(); display_time(); while(1) { uchar key = key_scan(); if(key == 2 || key == 3 || key == 4) { set_time(); display_time(); } if(key == 5) { if(set_flag) { time_min = set_time_min; time_sec = set_time_sec; set_flag = 0; countdown(); display_time(); } } } } ``` 希望这个代码对你有帮助。

相关推荐

最新推荐

recommend-type

基于AT89C51单片机的十进制计算器系统设计

本设计是基于AT89C51 单片机进行的十进制计算器系统设计,可以完成计算器的键盘输入,进行加、减、乘、除4 位无符号数字的简单四则运算,并在LED 上相应的显示结果。硬件方面从功能考虑,首先选择内部存储资源丰富的...
recommend-type

基于AT89C51单片机的交通灯控制系统设计与仿真

AT89C51单片机的交通灯控制系统是由AT89C51单片机、键盘电路、LED倒计时、交通灯显示等模块组成。系统除基本交通灯功能外,还具有通行时间手动设置、可倒计时显示、急车强行通过、交通特殊情况处理等相关功能,实验...
recommend-type

AT89C51制作的简单计数器

本制作的主要核心电路是用AT89C51组成的按键取值电路,S3S4分别控制计数值的加减。数码管使用的是共阳极通过3906控制。本计数器的计数范围为0-999999,最大的频率为50Hz,当然这些参数都可以根据需要调整。
recommend-type

基于AT89C51单片机的变频调速控制系统设计

本文中,设计变频调速控制系统时,控制芯片采用单片机AT89C51,采用SA8281作为正弦波发生器,用IR2110芯片来驱动,另外考虑到系统的稳定性,设计了系统的保护电路,这样整个系统有成本低廉,功能齐全的特点,并具有...
recommend-type

单片机(AT89C51)定时/计数器实验案例

继上篇的《单片机(AT89C51)定时/计数器详解及其实验案例》由于各种原因里面没有实验案例现在在此补上。 单片机(AT89C51)定时/计数器详解见上篇:...
recommend-type

RTL8188FU-Linux-v5.7.4.2-36687.20200602.tar(20765).gz

REALTEK 8188FTV 8188eus 8188etv linux驱动程序稳定版本, 支持AP,STA 以及AP+STA 共存模式。 稳定支持linux4.0以上内核。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

:YOLOv1目标检测算法:实时目标检测的先驱,开启计算机视觉新篇章

![:YOLOv1目标检测算法:实时目标检测的先驱,开启计算机视觉新篇章](https://img-blog.csdnimg.cn/img_convert/69b98e1a619b1bb3c59cf98f4e397cd2.png) # 1. 目标检测算法概述 目标检测算法是一种计算机视觉技术,用于识别和定位图像或视频中的对象。它在各种应用中至关重要,例如自动驾驶、视频监控和医疗诊断。 目标检测算法通常分为两类:两阶段算法和单阶段算法。两阶段算法,如 R-CNN 和 Fast R-CNN,首先生成候选区域,然后对每个区域进行分类和边界框回归。单阶段算法,如 YOLO 和 SSD,一次性执行检
recommend-type

ActionContext.getContext().get()代码含义

ActionContext.getContext().get() 是从当前请求的上下文对象中获取指定的属性值的代码。在ActionContext.getContext()方法的返回值上,调用get()方法可以获取当前请求中指定属性的值。 具体来说,ActionContext是Struts2框架中的一个类,它封装了当前请求的上下文信息。在这个上下文对象中,可以存储一些请求相关的属性值,比如请求参数、会话信息、请求头、应用程序上下文等等。调用ActionContext.getContext()方法可以获取当前请求的上下文对象,而调用get()方法可以获取指定属性的值。 例如,可以使用 Acti
recommend-type

c++校园超市商品信息管理系统课程设计说明书(含源代码) (2).pdf

校园超市商品信息管理系统课程设计旨在帮助学生深入理解程序设计的基础知识,同时锻炼他们的实际操作能力。通过设计和实现一个校园超市商品信息管理系统,学生掌握了如何利用计算机科学与技术知识解决实际问题的能力。在课程设计过程中,学生需要对超市商品和销售员的关系进行有效管理,使系统功能更全面、实用,从而提高用户体验和便利性。 学生在课程设计过程中展现了积极的学习态度和纪律,没有缺勤情况,演示过程流畅且作品具有很强的使用价值。设计报告完整详细,展现了对问题的深入思考和解决能力。在答辩环节中,学生能够自信地回答问题,展示出扎实的专业知识和逻辑思维能力。教师对学生的表现予以肯定,认为学生在课程设计中表现出色,值得称赞。 整个课程设计过程包括平时成绩、报告成绩和演示与答辩成绩三个部分,其中平时表现占比20%,报告成绩占比40%,演示与答辩成绩占比40%。通过这三个部分的综合评定,最终为学生总成绩提供参考。总评分以百分制计算,全面评估学生在课程设计中的各项表现,最终为学生提供综合评价和反馈意见。 通过校园超市商品信息管理系统课程设计,学生不仅提升了对程序设计基础知识的理解与应用能力,同时也增强了团队协作和沟通能力。这一过程旨在培养学生综合运用技术解决问题的能力,为其未来的专业发展打下坚实基础。学生在进行校园超市商品信息管理系统课程设计过程中,不仅获得了理论知识的提升,同时也锻炼了实践能力和创新思维,为其未来的职业发展奠定了坚实基础。 校园超市商品信息管理系统课程设计的目的在于促进学生对程序设计基础知识的深入理解与掌握,同时培养学生解决实际问题的能力。通过对系统功能和用户需求的全面考量,学生设计了一个实用、高效的校园超市商品信息管理系统,为用户提供了更便捷、更高效的管理和使用体验。 综上所述,校园超市商品信息管理系统课程设计是一项旨在提升学生综合能力和实践技能的重要教学活动。通过此次设计,学生不仅深化了对程序设计基础知识的理解,还培养了解决实际问题的能力和团队合作精神。这一过程将为学生未来的专业发展提供坚实基础,使其在实际工作中能够胜任更多挑战。