单片机实验(三)

前言

实验一:利用定时器T1的中断控制P1.7引脚输出音频信号,启动蜂鸣器发出一段熟悉的与众不同的具有10个音节的音乐音频。

实验二:使用定时器/计数器来实现一个LCD显示年、月、日、星期 、时、分、秒的电子表,要求时和分可以方便设置。液晶显示器采用LCD 1602.

参考链接

LED数码管的静态显示与动态显示(Keil+Proteus)-CSDN博客

定时器/计数器的应用-CSDN博客

基于51单片机的7键电子琴音乐播放器proteus_c51电子琴程序_BT-BOX的博客-CSDN博客​​​​​​

笔记:C51单片机——音乐播放,模拟钢琴键。_c51音乐_c-tion的博客-CSDN博客

51单片机学习--蜂鸣器播放音乐_51单片机蜂鸣器音乐代码-CSDN博客

闰年(历法中的名词)_百度百科 (baidu.com)

蜂鸣器播放音乐_哔哩哔哩_bilibili

【51单片机】蜂鸣器播放音乐 - 知乎 (zhihu.com)【51单片机】蜂鸣器播放音乐 - 知乎 (zhihu.com)

独立键盘接口设计(Keil+Proteus)-CSDN博客

如何看懂音乐简谱_简谱怎么看-CSDN博客

【蓝桥杯——单片机学习笔记】十六.蜂鸣器播放音乐(STC15F2K60S2)_有源蜂鸣器可以播放音乐吗-CSDN博客

简谱_百度百科 (baidu.com)

[11-2] 蜂鸣器播放提示音&音乐_哔哩哔哩_bilibili

实验一

Keil

这个找了半天,不是付费的就是代码不全的,然后是从视频上面找到一个简单的敲了一下,还是没怎么弄懂。

#include<reg51.h>
#define uchar unsigned char
#define uint unsigned intsbit speaker=P1^7;//生日快乐歌的音符频率表,不同频率由不同的延时来决定
uchar code SONG_TONE[]={212,212,190,212,159,169,212,212,190,212,142,159,212,212,106,126,159,169,190,119,119,126,159,142,159,0};//生日快乐歌节奏节拍表,节拍决定每个音符的演奏长短
uchar code SONG_LONG[]={9,3,12,12,12,24,9,3,12,12,24,9,3,12,12,12,12,12,9,3,12,12,12,24,0};//延时函数
void delay(uint x){uchar t;while(x--)for(t=0;t<120;t++);
}
//播放音乐
void PlayMusic(){uint i=0,j,k;for(i=0;i<26;i++){for(j=0;j<SONG_LONG[i]*20;j++){speaker=~speaker;for(k=0;k<SONG_TONE[i]/3;k++);}delay(100);}
}void main(){while(1){PlayMusic();}
}

Proteus

原理图到没什么需要更改的,主要是不同频率的声音应该怎么发出来。

拓展 

其实主要还是代码部分,原理图不需要修改,需要先找一份简谱,然后需要知道怎么看音乐乐谱。

实验需要把握的就是“音的高低”“音的长短”。

在目前的代码中不考虑倍低音和低音,自己低音、中音、高音对应的代码。

音的高低部分需要知道,在基本音符上方加一个“.”,表示该音升高一个八度,称为高音;在基本音符下方加一个“.”,表示该音降低一个八度,称为低音。这里只需要知道区分低中高音即可。

知道了音的高低接下来就是音的长短部分,其实就是表示那个音延迟的时间,这里就只需要记住在基本音符下方加记一条短横线,表示缩短原音符时值的一半。

这两个问题都解决了就可以直接编写程序了,其实也不需要很复杂的乐理知识,甚至基本音符是什么都可以不用知道,就是看着简谱修改成自己对应的即可,对于特殊的符号还需要自己查资料看表示什么。

基本音符 

1234567
DoReMiFaSolLaSi

 《平凡之路》的部分简谱

 

 缺点对应倍低音的地方无法编写程序,因为这只有低音、中音、高音频率的代码,然后音也有点不准吧,但是调子感觉也差不多,可以听出来大致的调子。

#include<reg51.h>
#define uchar unsigned char#define L1  1
#define L1_ 2
#define L2  3
#define L2_ 4
#define L3  5
#define L4  6
#define L4_ 7
#define L5  8
#define L5_ 9
#define L6  10
#define L6_ 11
#define L7  12
#define M1  13
#define M1_ 14
#define M2  15
#define M2_ 16
#define M3  17
#define M4  18
#define M4_ 19
#define M5  20
#define M5_ 21
#define M6  22
#define M6_ 23
#define M7  24
#define H1  25
#define H1_ 26
#define H2  27
#define H2_ 28
#define H3  29
#define H4  30
#define H4_ 31
#define H5  32
#define H5_ 33
#define H6  34
#define H6_ 35
#define H7  36#define ClockSpeed 12000000 //时钟频率,Hz
#define SongSpeed 330       //ms,八分音符
sbit beepIO = P1^7;         //定义蜂鸣器IO口unsigned char freq_select;//音阶频率表
unsigned int code freq_table[]={0,61714 ,61928 ,62131 ,62322 ,62502 ,62673 ,62833 ,62985 ,63128 ,63263 ,63391 ,63511, //低音63628 ,63731 ,63835 ,63928 ,64021 ,64103 ,64185 ,64260 ,64331 ,64400 ,64463 ,64524, //中音64580 ,64633 ,64684 ,64732 ,64777 ,64820 ,64860 ,64898 ,64934 ,64968 ,65000 ,65030 //高音};				//平凡之路
uchar code song[]={M1,2, M3,1, M1,1, L7,1, M1,1, M2,1, L5,1, 0,1, M3,1, M3,1, M3,1, M6,1, M6,1, M6,1, M1,1, M2,1, M3,1, M3,1, 0,2, 0,2, 0,2,0,1, M3,1, M3,1, M6,1, M6,1, M6,3, M5,1, M5,2, M4,1, M3,3, 0,2, 0,2, 0,1, M3,1, M3,1, M6,1, M6,1, M1,1, M2,1, M3,1,M3,2, 0,2, 0,2, 0,2, 0,1, M3,1, M3,1, M1,1, M4,1, M4,1, M4,1, M4,1, M3,1, M1,2, 0,2, 0,2,  40};void timer0_initial()
{beepIO = 0;TH0   = 0xFD;	TL0   = 0x09;TMOD  = 0x01;  //选择定时器0,工作方式1ET0   = 1;     //允许定时器0中断EA    = 1;     //CPU开放中断TF0   = 0;     //溢出标志位清0TR0   = 1;     //开启定时器0
}void BeepTimer0() interrupt 1	//中断函数
{beepIO = !beepIO;   //蜂鸣器IO口高低电平转换TH0 = freq_table[freq_select]/256 ;TL0 = freq_table[freq_select]%256 ;
}void delay_ms(unsigned int x) //延时函数
{unsigned char t;while(x--) for(t=0;t<120;t++);
}void main()
{unsigned char select;timer0_initial();while(song[select]!= 40)        //判断歌曲是否结束{freq_select=song[select];		if(freq_select)         //判断是否是休止符0{select++;delay_ms(song[select]*SongSpeed);TR0 = 0;   //关闭蜂鸣器一段时间再打开,模拟按键抬手动作。delay_ms(10);TR0 = 1;select++;}else{			TR0 = 0;select++;delay_ms(song[select]*SongSpeed);TR0 = 1;select++;}}
}

然后就是此次的作业《倔强》

 这个写我还是比较满意,就是中间会断掉然后重新开始,推测是因为储存不了这么多的数据,不能完整播放整首歌,但是也还不错,还是比较有成就感的。

#include<reg51.h>
#define uchar unsigned char#define L1  1
#define L1_ 2
#define L2  3
#define L2_ 4
#define L3  5
#define L4  6
#define L4_ 7
#define L5  8
#define L5_ 9
#define L6  10
#define L6_ 11
#define L7  12
#define M1  13
#define M1_ 14
#define M2  15
#define M2_ 16
#define M3  17
#define M4  18
#define M4_ 19
#define M5  20
#define M5_ 21
#define M6  22
#define M6_ 23
#define M7  24
#define H1  25
#define H1_ 26
#define H2  27
#define H2_ 28
#define H3  29
#define H4  30
#define H4_ 31
#define H5  32
#define H5_ 33
#define H6  34
#define H6_ 35
#define H7  36#define ClockSpeed 12000000 //时钟频率,Hz
#define SongSpeed 330       //ms,八分音符
sbit beepIO = P1^7;         //定义蜂鸣器IO口uchar freq_select;//音阶频率表
unsigned int code freq_table[]={0,61714 ,61928 ,62131 ,62322 ,62502 ,62673 ,62833 ,62985 ,63128 ,63263 ,63391 ,63511, //低音63628 ,63731 ,63835 ,63928 ,64021 ,64103 ,64185 ,64260 ,64331 ,64400 ,64463 ,64524, //中音64580 ,64633 ,64684 ,64732 ,64777 ,64820 ,64860 ,64898 ,64934 ,64968 ,65000 ,65030 //高音};				//倔强
uchar code song[]={M1,3, M1,1, M1,1, L7,1, M1,1, M2,1, M2,1, L7,1,M1,3, M1,1, M1,1, L7,1, M1,1, M2,1, M2,1, M2,1, M3,1, M1,1,M1,3, M1,1, M1,1, M1,1, L5,1, M1,1, M1,1, M2,1, M4,1, M3,1, M2,1, M1,1, M3,1, M2,1, M2,2,M1,3, M1,1, M1,1, L7,1, M1,1, M2,1, M2,1, L7,1,M1,3, M1,1, M1,1, L7,1, M1,1, M2,1, M2,1, M2,1, M3,1, M1,1,M1,3, M1,1, M1,1, M1,1, L5,1, M1,1, M1,1, M2,1, M4,1, M3,1, M2,1, M1,1, M3,1, M2,1, M2,1, 0,1,M1,1, L7,1, M1,1, M1,1, L7,1, M1,1, M1,1, 0,2, M1,1, L7,1, M1,1, M1,1, M2,1, M3,1, M3,2, M1,1, L7,1, M1,1, M1,1, M2,1, M4,1, M4,1, M3,1, M2,1, M1,1, M1,1, L7,1, M1,1, M3,1, M6,1, M5,1, M5,1,M3,1, M2_,1, M3,1, M3,1, M4,1, M5,1, M5,1, M4,1, M3,1, M3,1, M2,1, M1,1, L7,1, M1,1, M1,1, M2,1, M3,1, M3,1, M2,1, M1,1, L7,1,M1,1, L7,1, M1,1, M1,1, M6,1, M7,1, M7,1, M6,1, M5,1, M5,1, M3,1, M5,1, M6,1, M1,1, M1,1, M1,3, M6,1, M7,1, M3,1, M5,1, M5,1, 0,1,M3,1, M3,1, M3,1, M3,1, M4,1, M5,1, M5,1, M4,1, M3,1, M3,1, M3,1, M2,1, M1,1, M1,1, M1,1, M1,1, M2,1, M3,1, M2,1, M1,1, L7,1, M1,1, L6,1, M1,1, M1,1, M6,1, M7,1, M6,1, M5,1, M3,1, M6,1, M6,1, M1,1, M1,1, M1,3, M6,1, M7,1, M6,1, M5,1, M5,1, L6,1, M1,1, M1,2, 40
};void timer0_initial()
{beepIO = 0;TH0   = 0xFD;	TL0   = 0x09;TMOD  = 0x01;  //选择定时器0,工作方式1ET0   = 1;     //允许定时器0中断EA    = 1;     //CPU开放中断TF0   = 0;     //溢出标志位清0TR0   = 1;     //开启定时器0
}void BeepTimer0() interrupt 1	//中断函数
{beepIO = !beepIO;   //蜂鸣器IO口高低电平转换TH0 = freq_table[freq_select]/256 ;TL0 = freq_table[freq_select]%256 ;
}void delay_ms(unsigned int x) //延时函数
{uchar t;while(x--) for(t=0;t<120;t++);
}void main()
{uchar select;timer0_initial();while(song[select]!= 40)        //判断歌曲是否结束{freq_select=song[select];		if(freq_select)         //判断是否是休止符0{select++;delay_ms(song[select]*SongSpeed);TR0 = 0;   //关闭蜂鸣器一段时间再打开,模拟按键抬手动作。delay_ms(10);TR0 = 1;select++;}else{			TR0 = 0;select++;delay_ms(song[select]*SongSpeed);TR0 = 1;select++;}}
}

实验二

Keil

代码还是和之前的差不多,然后就是需要增加年月日星期的处理,星期常见的就是用三个字母来表示,年月日都是用数字,有了前面的基础就很容易确定LCD 1602的显示以及时钟的进位,这里需要考虑的是根据不同的年份以及月份,他的天数是不一致的,所以这个需要考虑进去,经过在程序中这里也没有体现到,可以通过设置时间的节点来去验证他是否满足要求。

0123456789ABCDEF
80yyyy.MM.ddww
C0TIMEhh:mm:ss

我这里犯了一个很严重的错误就是对于年份在用LCD输出的时候用到取余和整除,年份的类型是字符型,这就导致输出的数不对,后面还是通过控制变量法来缩小他的范围最后是想到uchar最大是到255,这个年份都是4位数的,所以会导致越界计算出来的数值就不正确,后面是改成了整型。 

#include<reg51.h>
#define uchar unsigned char
#define uint unsigned int
sbit RS=P2^0;
sbit RW=P2^1;
sbit E=P2^2;//202140200126
uint int_time;//定义中断次数变量(最多到200)
uchar code date[]=" 2022.12.31 SAT ";//LCD 第1行显示的内容
uchar code time[]=" TIME  23:59:55 ";//第2行显示的内容
uchar second=55,minute=59,hour=23;//秒,分钟,小时
uint year=2022;//年
uchar month=12,day=31,week=6;//月,日,星期
void delay(uint i);//延时函数
void write_com(uchar com);//写入命令数据到LCD
void write_data(uchar date);//写入字符显示数据到LCD
void init1602();//LCD1602初始化设定
void write_sfm(uchar add,uchar date);//向指定地址写入数据
void write_sfm_year(uchar add,uint year);//写年
void write_sfm_week(uchar add,uchar week);//写星期
void clock_init();//对时钟进行初始化
void clock_write(uint s,uint m,uint h,uint d,uint w,uint M,uint y);//写数据
int isleap(uint year);//判断是否是闰年void main(){int_time=0;//中断次数、秒、分钟、小时,年,月,日,星期变量进行清零second=55;minute=59;hour=23;year=2022;month=12;day=31;week=6;init1602();//lcd的初始化clock_init();//时钟的初始化TMOD=0x01;//设置定时器T0为方式1定时EA=1;//总中断开ET0=1;//允许T0中断TH0=0xEE;//对T0进行初始化TH1=0x00;TR0=1;while(1){clock_write(second,minute,hour,day,week,month,year);}
}void delay(uint j){//延时函数uchar i=110;for(;j>0;j--){while(i--);i=110;}
}void write_com(uchar com){//写入命令数据到LCDRW=0;RS=0;P0=com;delay(5);E=1;delay(5);E=0;
}void write_data(uchar date){//写入字符显示数据到LCDRW=0;RS=1;P0=date;delay(5);E=1;delay(5);E=0;
}void init1602(){//LCD1602初始化设定RW=0;E=0;write_com(0x3C);write_com(0x0C);//开整体显示,光标关,无闪烁write_com(0x06);//光标右移,写入一个字符后地址指针加1write_com(0x01);//清屏write_com(0x80);//字符输入地址,字符的第一位
}void write_sfm(uchar add,uchar date){//向指定地址写入数据uchar shi,ge;shi=date/10;//十位ge=date%10;//个位write_com(add);write_data(0x30+shi);//0x30表示48,让数字变成字符write_data(0x30+ge);
}//这里头脑不清醒了一下,uchar是一个字节,最大是127,而年份是四位数的,所以需要用uint类型才可以uint是两个字节有(65535)
void write_sfm_year(uchar add,uint date){//向指定地址写入数据(年份是四位的,所以单独处理)uchar ge,shi,bai,qian;qian=date/1000;//千位bai=date/100%10;//百位shi=date/10%10;//十位ge=date%10;//个位write_com(add);write_data(0x30+qian);//0x30表示48,让数字变成字符write_data(0x30+bai);//0x30表示48,让数字变成字符write_data(0x30+shi);//0x30表示48,让数字变成字符write_data(0x30+ge);//0x30表示48,让数字变成字符
}void write_sfm_week(uchar add,uchar date){//向指定地址写入数据(星期是用三个字母来表示)write_com(add);switch(date){case 0:write_data('S');write_data('U');write_data('N');break;//周日case 1:write_data('M');write_data('O');write_data('N');break;//周一case 2:write_data('T');write_data('U');write_data('E');break;//周二case 3:write_data('W');write_data('E');write_data('D');break;//周三case 4:write_data('T');write_data('H');write_data('U');break;//周四case 5:write_data('F');write_data('R');write_data('I');break;//周五case 6:write_data('S');write_data('A');write_data('T');break;//周六}
}void clock_init(){//对时钟进行初始化uchar i,j;//写第一行write_com(0x80);for(i=0;i<16;i++){write_data(date[i]);}//写第二行write_com(0xC0);for(j=0;j<16;j++){write_data(time[j]);}
}void clock_write(uint s,uint m,uint h,uint d,uint w,uint M,uint y){//向指定位置写数据//写年write_sfm_year(0x81,y);//写月write_sfm(0x86,M);//写星期write_sfm_week(0x8C,w);//写日write_sfm(0x89,d);//写小时write_sfm(0xC7,h);//写分钟write_sfm(0xCa,m);//写秒write_sfm(0xCd,s);
}//判断是否是闰年
int isleap(uint year){//能被400整除或者能被4整除且不能被100整除为闰年if((year%400==0)||(year%4==0 && year %100!=0)){return 1;}else{return 0;}
}//中断服务函数
void timer0(void) interrupt 1{uchar leap=isleap(year);//判断今年是否为闰年TR0=0;//停止计时,避免给计时造成误差(需要有这个,不然就一直在中断,导致时间不准确)TH0=0xEE;//对T0重新赋初值TH1=0x00;int_time++;//记录中断次数if(int_time==200){//中断次数满200次5ms*200=1sint_time=0;//中断次数变量清零second++;//加1秒}if(second==60){//60秒为一分钟second=0;minute++;}if(minute==60){//60分钟为一小时minute=0;hour++;}if(hour==24){//一天是24小时hour=0;week++;day++;}if(week==7){//0表示周日,week=7表示新的一周开始了week=0;}if(month==2){//是闰年(2月有29天)if(leap==1){if(day>29){day=0;month++;}}else{//不是闰年(2月有28天)if(day>28){day=0;month++;}}}else if(month==1 || month==3 || month==5 || month==7 || month==8 || month==10 || month==12){if(day>31){day=0;month++;}}else if(month==4 || month==6 || month==9 || month==11){if(day>30){day=0;month++;}}if(month==13){month=0;year++;}TR0=1;
}

Proteus

原理图和之前是保持一致的。

 在线过年,我们一起包饺子!!!

 

拓展

 

咱就说这个应不应该希望。这里大致的思路就是添加几个按钮,来控制增加或者减少年月日时分, 秒就不需要添加了,时钟一直在动这个调也不好调。

这里我一直在想这个调分钟这个进位怎么调,主要开始对于每月的天数,我突然想到我以前的电子表就是每个位置是独立的,调分钟不会影响其他的,就是分钟从0-59变化,抱着这样的想法,时钟就可以很快确定了,不就是求余吗,大家都会。

为了简单我又做了一个假设,当设置的时候每个月都是31天,而且分开设置的话,天数和星期就不是保持同步的了。

目前就是在之前的基础上增加了一个独立键盘的扫描,以及对应的求余功能,对于设置直接导致进位和借位的功能还没实现,目前先将简单实现的完成。扫描的时候尽量不和RS,RW,E用一个口,会导致不好查询,程序我不知道怎么编写,如果影响了那三根引脚的输出会导致LCD一直在查询发出警告。

这个我昨天想到了一个东西,我感觉突然就悟了一样,就是对于时间小于0前面要退一位吗,因为都是无符号数所以不好来设置,然后我就想到了一个方法。假设是要判断分钟minute,可以先给分钟加上一个60,然后整除60,如果他的结果是0,就表示之前的值是小于60的,如果他的值是等于1,就表示他的值在[0,60)之间,如果他的值等于2,就表示他大于等于60了,因为数值是实时更新的,所以这里就排除了数是一个很大的正数,或者是一个很小的负数的情况,主要是考虑-1,和60这两种情况怎么让小时的位置上面的数发生相应的变化。

#include<reg51.h>
#define uchar unsigned char
#define uint unsigned int
sbit RS=P2^0;
sbit RW=P2^1;
sbit E=P2^2;//分别对应12个按钮
sbit S1=P1^0;
sbit S2=P1^1;
sbit S3=P1^2;
sbit S4=P1^3;
sbit S5=P1^4;
sbit S6=P1^5;
sbit S7=P1^6;
sbit S8=P1^7;
sbit S9=P3^04;
sbit S10=P3^5;
sbit S11=P3^6;
sbit S12=P3^7;uint int_time;//定义中断次数变量(最多到200)
uchar code date[]=" 2022.12.31 SAT ";//LCD 第1行显示的内容
uchar code time[]=" TIME  23:59:55 ";//第2行显示的内容
uint second=55,minute=59,hour=23;//秒,分钟,小时
uint year=2022;//年
uchar month=12,day=31,week=6;//月,日,星期
uchar keyval;//定义键盘储存变量单元void delay(uint i);//延时函数
void write_com(uint com);//写入命令数据到LCD
void write_data(uint date);//写入字符显示数据到LCD
void init1602();//LCD1602初始化设定
void write_sfm(uint add,uint date);//向指定地址写入数据
void write_sfm_year(uint add,uint year);//写年
void write_sfm_week(uint add,uint week);//写星期
void clock_init();//对时钟进行初始化
void clock_write(uint s,uint m,uint h,uint d,uint w,uint M,uint y);//写数据
int isleap(uint year);//判断是否是闰年
void key_scan(void);//扫描键盘void main(){int_time=0;//中断次数、秒、分钟、小时,年,月,日,星期变量进行清零second=55;minute=59;hour=23;year=2022;month=12;day=31;week=6;keyval=0;//键值初始化为0init1602();//lcd的初始化clock_init();//时钟的初始化TMOD=0x01;//设置定时器T0为方式1定时EA=1;//总中断开ET0=1;//允许T0中断TH0=0xEE;//对T0进行初始化TH1=0x00;TR0=1;while(1){key_scan();switch(keyval){case 1://每年year++;year=(year+10000)%10000;break;case 2:year--;year=(year+10000)%10000;break;case 3://每月month++;month=(month+11)%12+1;break;//要让0表示12则需要先减1求余之后加1case 4:month--;month=(month+11)%12+1;break;//要让0表示12则需要先减1求余之后加1case 5://每周week++;week=(week+7)%7;break;case 6:week--;week=(week+7)%7;break;case 7://天数day++;day=(day+30)%31+1;break;case 8:day--;day=(day+30)%31+1;break;case 9://小时hour++;hour=(hour+60)%60;break;case 10:hour--;hour=(hour+60)%60;break;case 11://分钟minute++;minute=(minute+60)%60;break;case 12:minute--;minute=(minute+60)%60;break;}clock_write(second,minute,hour,day,week,month,year);keyval=0;}
}//延时函数
void delay(uint j){uchar i=110;for(;j>0;j--){while(i--);i=110;}
}//写入命令数据到LCD
void write_com(uint com){RW=0;RS=0;P0=com;delay(5);E=1;delay(5);E=0;
}//写入字符显示数据到LCD
void write_data(uint date){RW=0;RS=1;P0=date;delay(5);E=1;delay(5);E=0;
}//LCD1602初始化设定
void init1602(){RW=0;E=0;write_com(0x3C);write_com(0x0C);//开整体显示,光标关,无闪烁write_com(0x06);//光标右移,写入一个字符后地址指针加1write_com(0x01);//清屏write_com(0x80);//字符输入地址,字符的第一位
}//向指定地址写入数据
void write_sfm(uint add,uint date){uchar shi,ge;shi=date/10;//十位ge=date%10;//个位write_com(add);write_data(0x30+shi);//0x30表示48,让数字变成字符write_data(0x30+ge);
}//这里头脑不清醒了一下,uchar是一个字节,最大是127,而年份是四位数的,所以需要用uint类型才可以uint是两个字节有(65535)
void write_sfm_year(uint add,uint date){//向指定地址写入数据(年份是四位的,所以单独处理)uchar ge,shi,bai,qian;qian=date/1000;//千位bai=date/100%10;//百位shi=date/10%10;//十位ge=date%10;//个位write_com(add);write_data(0x30+qian);//0x30表示48,让数字变成字符write_data(0x30+bai);//0x30表示48,让数字变成字符write_data(0x30+shi);//0x30表示48,让数字变成字符write_data(0x30+ge);//0x30表示48,让数字变成字符
}void write_sfm_week(uint add,uint date){//向指定地址写入数据(星期是用三个字母来表示)write_com(add);switch(date){case 0:write_data('S');write_data('U');write_data('N');break;//周日case 1:write_data('M');write_data('O');write_data('N');break;//周一case 2:write_data('T');write_data('U');write_data('E');break;//周二case 3:write_data('W');write_data('E');write_data('D');break;//周三case 4:write_data('T');write_data('H');write_data('U');break;//周四case 5:write_data('F');write_data('R');write_data('I');break;//周五case 6:write_data('S');write_data('A');write_data('T');break;//周六}
}//对时钟进行初始化
void clock_init(){uchar i,j;//写第一行write_com(0x80);for(i=0;i<16;i++){write_data(date[i]);}//写第二行write_com(0xC0);for(j=0;j<16;j++){write_data(time[j]);}
}//向指定位置写数据
void clock_write(uint s,uint m,uint h,uint d,uint w,uint M,uint y){//写年write_sfm_year(0x81,y);//写月write_sfm(0x86,M);//写星期write_sfm_week(0x8C,w);//写日write_sfm(0x89,d);//写小时write_sfm(0xC7,h);//写分钟write_sfm(0xCa,m);//写秒write_sfm(0xCd,s);
}//判断是否是闰年
int isleap(uint year){//能被400整除或者能被4整除且不能被100整除为闰年if((year%400==0)||(year%4==0 && year %100!=0)){return 1;}else{return 0;}
}//键盘扫描
void key_scan(void){P1=0xFF;P3=0xFF;//不能给P2赋值,不然会导致LCD一直警告if((P1&0xFF)!=0xFF||(P3&0xF0)!=0xF0){delay(10);//适当延迟,怕到时候误判if(S1==0)keyval=1;if(S2==0)keyval=2;if(S3==0)keyval=3;if(S4==0)keyval=4;if(S5==0)keyval=5;if(S6==0)keyval=6;if(S7==0)keyval=7;if(S8==0)keyval=8;if(S9==0)keyval=9;if(S10==0)keyval=10;if(S11==0)keyval=11;if(S12==0)keyval=12;}
}//中断服务函数
void timer0(void) interrupt 1{uchar leap=isleap(year);//判断今年是否为闰年TR0=0;//停止计时,避免给计时造成误差(需要有这个,不然就一直在中断,导致时间不准确)TH0=0xEE;//对T0重新赋初值TH1=0x00;int_time++;//记录中断次数if(int_time==200){//中断次数满200次5ms*200=1sint_time=0;//中断次数变量清零second++;//加1秒}if(second==60){//60秒为一分钟second=0;minute++;}if(minute==60){//60分钟为一小时minute=0;hour++;}if(hour==24){//一天是24小时hour=0;week++;day++;}if(week==7){//0表示周日,week=7表示新的一周开始了week=0;}if(month==2){//是闰年(2月有29天)if(leap==1){if(day>29){day=0;month++;}}else{//不是闰年(2月有28天)if(day>28){day=0;month++;}}}else if(month==1 || month==3 || month==5 || month==7 || month==8 || month==10 || month==12){if(day>31){day=0;month++;}}else if(month==4 || month==6 || month==9 || month==11){if(day>30){day=0;month++;}}if(month==13){month=0;year++;}TR0=1;
}

为了更好的表示按钮的含义这里用到了写文字的模块,如果直接用中文命名器件容易导致程序崩溃。

综合 

怎么好看怎么来吧,就是有点小卡。

总结

认真想下来其实还是很有意思的,但是也是很耗时间,有时间再来尝试一些新的功能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/208030.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vmware17虚拟机安装windows10系统

不要去什么系统之家之类的下载镜像&#xff0c;会不好安装&#xff0c;镜像被魔改过了&#xff0c;适合真实物理机上的系统在PE里安装系统&#xff0c;建议下载原版系统ISO文件 安装vmware17pro 下载地址https://dwangshuo.jb51.net/202211/tools/VMwareplayer17_855676.rar 解…

04.PostgreSQL是如何实现隔离级别的?

PostgreSQL是如何实现隔离级别的&#xff1f; 事务有哪些特性&#xff1f; 事务看起来感觉简单&#xff0c;但是要实现事务必须要遵守 4 个特性&#xff0c;分别如下&#xff1a; 原子性&#xff08;Atomicity&#xff09;&#xff1a;一个事务中的所有操作&#xff0c;要么…

Elasticsearch(ES)概述

文章目录 一.什么是Elasticsearch?1.正向索引和倒排索引2.Mysql和ES的概念对比3.安装elasticsearch、kibana 二.IK分词器三.索引库操作四.文档操作五.RestClient操作索引库1.初始化RestClient2.创建索引库3.删除索引库4.判断索引库是否存在 六.RestClient操作文档1.新增文档2.…

Elasticsearch 优化查询中获取字段内容的方式,性能提升5倍!

1、背景 集群配置为&#xff1a;8 个 node 节点&#xff0c;16 核 32G&#xff0c;索引 4 分片 1 副本。应用程序的查询逻辑是按经纬度排序后找前 200 条文档。 1、应用对查询要求比较高&#xff0c;search 没有慢查询的状态。 2、集群压测性能不能上去&#xff0c;cpu 使用未打…

Nat. Rev. Chem. | 一份关于用机器学习研究化学问题的评估指导

今天为大家介绍的是来自Tiago Rodrigues团队的一篇论文。机器学习&#xff08;ML&#xff09;有望解决化学领域的重大挑战。尽管ML工作流程的适用性极广&#xff0c;但人们通常发现评估研究设计多种多样。目前评估技术和指标的异质性导致难以&#xff08;或不可能&#xff09;比…

java设计模式学习之【单例模式】

文章目录 引言单例模式简介定义与用途实现方式&#xff1a;饿汉式懒汉式 UML 使用场景优势与劣势单例模式在spring中的应用饿汉式实现懒汉式实现数据库连接示例代码地址 引言 单例模式是一种常用的设计模式&#xff0c;用于确保在一个程序中一个类只有一个实例&#xff0c;并且…

计算机组成学习-数据的表示和运算总结

1、进制与编码 1.1 进位计数法 常用的进位计数法有十进制、二进制、八进制、十六进制等。十六进制每个 数位可取0〜9、A、B、C、D、E、F中的任意一个&#xff0c;其中A、B、C、D、E、F分别表示 10〜15。 八进制数字通常以前缀 "0"&#xff08;零&#xff09;加上数…

Ubuntu 2204 安装libimobiledevice

libimobiledevice是一个开源的软件&#xff0c;它可以直接使用系统原生协议和IOS设备进行通信&#xff0c;类似iMazing&#xff0c;iTunes&#xff0c;libimobiledevice不依赖IOS的私有库&#xff0c;并且连接IOS设备时用的都是原生协议&#xff0c;IOS无需越狱就能实现设备信息…

同旺科技 USB TO SPI / I2C --- 调试W5500_读写网关地址

所需设备&#xff1a; 内附链接 1、USB转SPI_I2C适配器(专业版); 首先&#xff0c;连接W5500模块与同旺科技USB TO SPI / I2C适配器&#xff0c;如下图&#xff1a; 这里的网关地址设置为192.168.1.1 先将网关地址写入寄存器&#xff0c;然后再读取出来&#xff1a;

Echarts 设备状态 甘特图

在做工厂智能化生产看板时&#xff0c;绝对会有设备状态看板&#xff0c;展示设备当天或者当前状态&#xff0c;设备状态数据一般是有mes 系统设备管理模块对设备信息进行采集&#xff0c;一般包括过站数据&#xff0c;设备当前状态&#xff0c;是否在线是否故障、检修、待生产…

Linux CentOS7 联网配置 | 安装中文输入法

参考视频&#xff1a;保姆式教学虚拟机联网liunx(centos)_哔哩哔哩_bilibili 配置网络&#xff1a;解决上网问题 第一步&#xff1a;选择网络模式 第二步&#xff1a;配置网卡命令&#xff1a;打开终端执行命令&#xff1a; 1、先切换到根目录下&#xff0c;防止在第执行cd …

css如何设置文本添加下划线

css文本添加下划线 text-decoration: underline;text-decoration相关属性参数 参数描述none默认。定义标准的文本。underline定义文本下的一条线。overline定义文本上的一条线。line-through定义穿过文本下的一条线。blink定义闪烁的文本。inherit规定应该从父元素继承 text-…

blue beacon rssi 指纹室内定位数据集

数据集是开展实验的基础&#xff0c;搜集并分享。如果你有关于室内定位的问题&#xff0c;请联系博主。 namedatesetpapercommentBLEBeacon: A Real-Subject Trial Dataset from Mobile Bluetooth Low Energy Beaconshttps://github.com/dimisik/BLEBeacon-Datasethttps://arxi…

matlab基于线性二次调节器(LQR)法实现机器人路径规划可变轨迹跟踪

1、内容简介 略 可以交流、咨询、答疑 2、内容说明 基于线性二次调节器(LQR)法实现机器人路径规划可变轨迹跟踪 3、仿真分析 略 load path.mat %% 轨迹处理 % 定义参考轨迹 refPos_x path(:,1); refPos_y path(:,2); refPos [refPos_x, refPos_y];% 计算航向角和曲率 …

Elasticsearch:ES|QL 函数及操作符

如果你对 ES|QL 还不是很熟悉的话&#xff0c;请阅读之前的文章 “Elasticsearch&#xff1a;ES|QL 查询语言简介​​​​​​​”。ES|QL 提供了一整套用于处理数据的函数和运算符。 功能分为以下几类&#xff1a; 目录 ES|QL 聚合函数 AVG COUNT COUNT_DISTINCT 计数为近…

G1264 0.85 V 启动,12uA,同步 DC/DC 变换器

G1264 0.85 V 启动&#xff0c;12uA&#xff0c;同步 DC/DC 变换器 概述&#xff1a; G1264集成 PFM 模式同步升压变换器&#xff0c;只需要一个电感和两个电容。由于升压专有设计&#xff0c;它启动在非常低的输入电压下降到850毫伏&#xff0c;使其成为单电池碱性/镍氢电池操…

Flink Flink中的合流

一、Flink中的基本合流操作 在实际应用中&#xff0c;我们经常会遇到来源不同的多条流&#xff0c;需要将它们的数据进行联合处理。所以 Flink 中合流的操作会更加普遍&#xff0c;对应的 API 也更加丰富。 二、联合&#xff08;Union&#xff09; 最简单的合流操作&#xf…

深入Android S (12.0) 探索Framework之输入系统IMS的构成与启动

文章目录 前言一、输入系统的基本组成部分二、输入系统相关源码分析1、IMS 构建1.1、SystemServer # startOtherServices()1.2、InputManagerService1.3、NativeInputManager # nativeInit()1.4、NativeInputManager1.5、InputManager1.6、InputDispatcher1.7、InputReader1.8、…

io基础入门

压缩的封装 参考&#xff1a;https://blog.csdn.net/qq_29897369/article/details/120407125?utm_mediumdistribute.pc_relevant.none-task-blog-2defaultbaidujs_baidulandingword~default-0-120407125-blog-120163063.235v38pc_relevant_sort_base3&spm1001.2101.3001.…

6 新建工程——寄存器

文章目录 6.1 本地新建工程文件夹6.2 新建工程6.2.1 选择CPU型号6.2.2 在线添加库文件6.2.3 添加文件6.2.4 复制存储器分配文件6.2.5 配置选项卡6.2.5.1 Linker6.2.5.2 Target6.2.5.3 Output 选项卡6.2.5.4 Listing 选项卡6.2.6 下载器配置 版本说明&#xff1a;MDK5.24 6.1 本…