蓝桥杯模块显示部分Part2:数码管
二、
发布日期:2021-07-01 03:12:44
浏览次数:2
分类:技术文章
本文共 8047 字,大约阅读时间需要 26 分钟。
九层妖塔 起于垒土
蓝桥杯模块显示部分Part2:数码管
数码管
一、原理图:
二、while(1)
死循环扫描写法
1、Template数码管初始化:
//----------------------------------数码管初始化--------------------------//void SEG_Init(void) //关闭熄灭所有数码管{ P2&=0X1F; //关573 P0=0X00; //预送数据 P2|=0XC0; P2&=0XDF; //开数码管位选573 P0=0X00; //位选,全部不选 P2&=0X1F; //关573 P2&=0X1F; //关573 P0=0XFF; //预送数据 P2|=0XE0; P2&=0XFF; //开数码管段选573 P0=0XFF; P2&=0X1F; //关573}
Notes:
●关闭熄灭所有数码管,直接从开发板初始化函数中复制即可。2、Template动态数码管驱动:
//----------------------------------数码管显示--------------------------//void SEG_Display(){ //-------第一个数码管---------// P2&=0X1F; //关573 P0=SEG_Array[yi]; //预送段选数据 P2|=0XE0; //开数码管段选573 P0=SEG_Array[yi]; //送段选数据 P2&=0X1F; //关573 P0=0X00; //消影 P2|=0XC0; //开数码管位选573 P0=0X01; //送位选数据 P2&=0X1F; //关573 Delay1ms(); //-------第二个数码管---------// P2&=0X1F; //关573 P0=SEG_Array[er]; //预送段选数据 P2|=0XE0; //开数码管段选573 P0=SEG_Array[er]; //送段选数据 P2&=0X1F; //关573 P0=0X00; //消影 P2|=0XC0; //开数码管位选573 P0=0X02; //送位选数据 P2&=0X1F; //关573 Delay1ms();//------------------------------------动态数码管驱动------第三四个---------// //-------第三个数码管---------// P2&=0X1F; //关573 P0=SEG_Array[san]; //预送段选数据 P2|=0XE0; //开数码管段选573 P0=SEG_Array[san]; //送段选数据 P2&=0X1F; //关573 P0=0X00; //消影 P2|=0XC0; //开数码管位选573 P0=0X04; //送位选数据 P2&=0X1F; //关573 Delay1ms(); //-------第四个数码管---------// P2&=0X1F; //关573 P0=SEG_Array[si]; //预送段选数据 P2|=0XE0; //开数码管段选573 P0=SEG_Array[si]; //送段选数据 P2&=0X1F; //关573 P0=0X00; //消影 P2|=0XC0; //开数码管位选573 P0=0X08; //送位选数据 P2&=0X1F; //关573 Delay1ms();//------------------------------------动态数码管驱动------第五六个---------// //-------第五个数码管---------// P2&=0X1F; //关573 P0=SEG_Array[wu]; //预送段选数据 P2|=0XE0; //开数码管段选573 P0=SEG_Array[wu]; //送段选数据 P2&=0X1F; //关573 P0=0X00; //消影 P2|=0XC0; //开数码管位选573 P0=0X10; //送位选数据 P2&=0X1F; //关573 Delay1ms(); //-------第六个数码管---------// P2&=0X1F; //关573 P0=SEG_Array[liu]; //预送段选数据 P2|=0XE0; //开数码管段选573 P0=SEG_Array[liu]; //送段选数据 P2&=0X1F; //关573 P0=0X00; //消影 P2|=0XC0; //开数码管位选573 P0=0X20; //送位选数据 P2&=0X1F; //关573 Delay1ms();//------------------------------------动态数码管驱动------第七八个---------// //-------第七个数码管---------// P2&=0X1F; //关573 P0=SEG_Array[qi]; //预送段选数据 P2|=0XE0; //开数码管段选573 P0=SEG_Array[qi]; //送段选数据 P2&=0X1F; //关573 P0=0X00; //消影 P2|=0XC0; //开数码管位选573 P0=0X40; //送位选数据 P2&=0X1F; //关573 Delay1ms(); //-------第八个数码管---------// P2&=0X1F; //关573 P0=SEG_Array[ba]; //预送段选数据 P2|=0XE0; //开数码管段选573 P0=SEG_Array[ba]; //送段选数据 P2&=0X1F; //关573 P0=0X00; //消影 P2|=0XC0; //开数码管位选573 P0=0X80; //送位选数据 P2&=0X1F; //关573 Delay1ms();}
3、Template部分主程序:
//-----数码管段码数组---//uchar SEG_Array[]={ 0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90, //数字0~9无点,索引值0~9 0x40,0x79,0x24,0x30,0x19,0x12,0x02,0x78,0x00,0x10, //数字0~9有点,索引值(0~9)+10 0XFF, //灭灯,索引值20 0X86, //字母大写E,索引值21 0XC8, //字母小写n,索引值22 0XBF //中间一横,索引值23};uchar yi,er,san,si,wu,liu,qi,ba; //传递段选数据的全局变量void All_Init(void); //开发板初始化void Delay1ms(); //@12.000MHzvoid SEG_Init(void); //数码管初始化,关闭熄灭所有数码管void SEG_Display(); //数码管显示void main(void){ All_Init(); //开发板初始化 while(1) { yi=******; er=******; san=******; si=******; wu=******; liu=******; qi=******; ba=******; SEG_Display(); //数码管显示 SEG_Init(); //数码管初始化,熄灭所有数码管 } }
Notes:
●段码[MSB…LSB]
即[P07…P00]
对应码顺序为[dp,g,f,e,d,c,b,a]
。(可由原理图读出) ●将数码管显示函数写在了一个SEG_Display()
函数里边;通过uchar yi,er,san,si,wu,liu,qi,ba;
八个全局变量来传递段选数据。 ●主函数中将八个全局变量放在数码管显示函数SEG_Display()
之前提前计算完,而没有使用形参在子函数中计算,可以保证在八个数码管的显示时间相对的绝对相等。 ●SEG_Init();
数码管初始化,熄灭所有数码管。其实在此只熄灭了最后一个点亮的第八个数码管。确保每个数码管扫描时间都为1ms。避免因主程序过长造成最后一个数码管扫描时间长更亮;或者按键松手检测时,只亮最后一个数码管。 ● 若全局变量和某个函数中的局部变量同名,则在该函数中,此全局变量被屏蔽,在该函数内,访问的是局部变量,与同名的全局变量不发生任何关系。
●大科的写法,定义了yi ~ ba八个全局变量,再使用相同形参名yi ~ ba定义了八个形参。咋一看貌似是带参的子函数。可实际运行却是,数码管段码提前计算好,赋值给八个全局变量,再由八个全局变量作为实参将段码值传递给八个形参,子函数读取的是子函数内形参的值。
三、定时器定时扫描
1、Template数码管段码缓冲数组处理函数:
//--------------------------数码管段码缓冲数组处理函数-----------------------------//void SEG_Process(uchar key_value) { sprintf(SEG_Buf,"%04u",(uint)key_value);}
2、Template数码管显示转换:
//------------------------------------动态数码管显示转换-------------------//void SEG_Tran(uchar *SEG_Buf,uchar *SEG_Code){ uchar i; uchar j=0; uchar temp; for(i=0;i<8;i++,j++) { switch(SEG_Buf[j]) { case '0': temp = 0xc0; break; case '1': temp = 0xf9; break; case '2': temp = 0xa4; break; case '3': temp = 0xb0; break; case '4': temp = 0x99; break; case '5': temp = 0x92; break; case '6': temp = 0x82; break; case '7': temp = 0xf8; break; case '8': temp = 0x80; break; case '9': temp = 0x90; break; case 'A': temp = 0x88; break; case 'b': temp = 0x83; break; case 'C': temp = 0xC6; break; case 'd': temp = 0xA1; break; case 'E': temp = 0x86; break; case 'F': temp = 0x8E; break; case 'H': temp = 0x89; break; case 'L': temp = 0xC7; break; case 'N': temp = 0xC8; break; case 'P': temp = 0x8C; break; case 'U': temp = 0xC1; break; case '-': temp = 0xBF; break; case ' ': temp = 0xFF; break; default: temp = 0XFF; } if(SEG_Buf[j+1] == '.') //小数点 { temp = temp&0X7F; j++; } SEG_Code[i] = temp; // 数码管段码数组 }}
Notes:
●将数码管段码缓冲数组SEG_Buf[9];
中的数据转化为数码管的段码,并保存到SEG_Code[8];
中。 ●数码管段码缓冲数组SEG_Buf[9];
为一维字符数组,元素为字符常量,即为字符串。 3、Template数码管显示:
//------------------------------------动态数码管显示---------------------//void SEG_Disp(uchar *SEG_Code,uchar SEG_Position){ P2&=0X1F; //关573 P0=SEG_Code[SEG_Position]; //预送段选数据 P2|=0XE0; //开数码管段选573 P0=SEG_Code[SEG_Position];//送段选数据 P2&=0X1F; //关573 P2&=0X1F; //关573 P0=0X00; //消影 P2|=0XC0; //开数码管位选573 P0=1<
4、Template定时器初始化及定时器服务函数:
//------------------------------------定时器1初始化,数码管------------//void SEG_Timer1Init(void) //1毫秒@12.000MHz{ TR1 = 0; //关定时器1 AUXR |= 0x40; //定时器时钟1T模式 TMOD &= 0x0F; //设置定时器模式 TL1 = 0x20; //设置定时初值 TH1 = 0xD1; //设置定时初值 TF1 = 0; //清除TF1标志 //TR1 = 1; //定时器1开始计时}//----------------------------定时器1服务函数,数码管显示----------------//void SEG_Timer1_Service() { TR1 = 1; //定时器1开始计时 if(TF1 == 1) { SEG_Timer1Init(); //关定时器;重装初值;清除TF1标志 SEG_Disp(SEG_Code,SEG_Position); //动态数码管显示 if(++SEG_Position == 8) SEG_Position=0; TR1 = 1; //定时器1开始计时 }}
Notes:
●将STC-ISP生成的定时器初始化函数中,删除最后一行开定时器;在第一行增加关定时器。 ●只要TR1 = 1
定时器就在不停计数,无论溢出标志位是否清除,无论若是中断查询是否响应。所以定时器溢出后,先关定时器,重新写入初值,再清除标志位。 ●数码管显示函数SEG_Disp()
的形参为SEG_Position
与全局变量SEG_Position
重名,因此在SEG_Disp()
中全局变量被屏蔽。 ●if(++SEG_Position == 8) SEG_Position=0;
中的SEG_Position
为全局变量,起着在各个函数间传递位选数据的作用。因此不能将SEG_Timer1_Service()
与SEG_Disp()
两个函数合写。 ●定时器没有按照国信长天例程使用中断查询法: ○数码管扫描的1ms时间要求并不严格,不需要1ms定时时间到的时候立即去响应。 ○若用中断查询法,可能会影响其他模块的显示或者通信。 5、Template部分主程序:
#includeuchar SEG_Buf[9]; //数码管段码缓冲数组。字符数组,元素为字符常量,即为字符串uchar SEG_Code[8]; //数码管段码uchar SEG_Position; //数码管位选void SEG_Timer1Init(void); //1毫秒@12.000MHz定时器1初始化—_数码管void SEG_Timer1_Service(); //定时器1服务函数,数码管显示void SEG_Tran(uchar *SEG_Buf,uchar *SEG_Code); //动态数码管显示转换void SEG_Disp(uchar *SEG_Code,uchar SEG_Position); //动态数码管显示void main(void){ All_Init(); SEG_Timer1Init(); //1毫秒@12.000MHz定时器1初始化_数码管 while(1) { sprintf(SEG_Buf,"%08u",(uint)num); SEG_Tran(SEG_Buf,SEG_Code); //动态数码管显示转换 SEG_Timer1_Service(); //定时器1服务函数,数码管显示 }}
Notes:
●SEG_Buf[9]
是用一位字符数组来存放字符串的,本质上是一个字符串。 ● SEG_Buf[9]
为数码管面向其他模块的接口。 在其他模块程序中,可以使用sprintf
将需要显示的数据写入字符串(即数组SEG_Buf[]
)中。(buffer-缓冲) ●子函数中的形参uchar *SEG_Buf
是用指针法来访问一个数组,形参从实参(实参为数组名)那儿得到数组的起始地址。(此外还有下标法,详见谭浩强237页) ●在程序执行过程中,不能使用赋值语句给(字符)数组整体赋值。 ●在程序执行过程中,可以给数组元素逐个赋字符值,最后认为加入串结束标志\0
。 ●数组名为一地址常量。 ●字符串以字符\0
作为字符串结束标志。\0
作为标志占用存储空间,但不计入字符串的实际长度。或者说sprintf
会在字符串后面添加\0
所以SEG_Buf[]
的实际长度为9;若字符串包含小数点(浮点型)SEG_Buf[]
的长度应为10。 转载地址:https://methadone-no1.blog.csdn.net/article/details/108098014 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!
发表评论
最新留言
路过按个爪印,很不错,赞一个!
[***.219.124.196]2024年04月20日 09时09分38秒
关于作者
喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!
推荐文章
Servlet_快速入门
2019-05-01
Servlet_生命周期方法
2019-05-01
Servlet_体系结构
2019-05-01
Request_原理
2019-05-01
Request_继承体系
2019-05-01
前端权限控制:获取用户信息接口构造数据
2019-05-01
有状态服务和无状态服务
2019-05-01
七牛云存储:断点续传
2019-05-01
字节流复制文本文件【应用】
2019-05-01
字节流复制图片
2019-05-01
私钥加密私钥解密
2019-05-01
锁的释放流程-ReentrantLock.unlock
2019-05-01
Java判断字符串是否为数字(浮点类型也包括)
2019-05-01
ubuntu opencv-python 安装很慢问题
2019-05-01
MySQL5.7版本修改了my.ini配置文件后mysql服务无法启动问题
2019-05-01
【大数据开发】Java基础 -总结21-Hashmap和HashTable的区别
2019-05-01
Azkaban体系结构
2019-05-01
机器学习之重头戏-特征预处理
2019-05-01