51单片机基于Proteus的8X8点阵滚动显示心形

一、效果展示

注:制作动图时效果显示不好

二、所用材料

1、Proteus中8x8点阵

8x8点阵主要分为以下两种:

第一种为行共阴,第二种为行共阳

详细介绍可以参考这位大神说的:

http://www.51hei.com/bbs/dpj-31998-1.html

在Proteus中我选择的是绿色的8x8的点阵,其类型是行共阴

2、74hc595

详细介绍见我上一篇文章

https://blog.csdn.net/qq_41639829/article/details/82151706

3、stc89c52

可以选择其他单片机,我用的这个

三、原理介绍

1、点阵的 “静态显示”

这里所说的静态显示就是不滚动画面的意思,比如你用点阵只显示一个数字,或者只显示一个心形。

首先明白点阵显示的原理就相当于发光二极管一样,只不过是64个发光二极管罢了。

假如你想让第一行全亮,应该怎么做?

很简单,因为是行共阴,所以第一行首先接负极,其他行都接正极或者不接,然后每一列都接正极,这样就可以使第一行全亮。如果第二列不接正极,则第一行的第二列的灯就不会亮,就只亮七个灯。如果第二行这时接地,则第二行也会和第一行一样亮同样的七个灯(不懂得话看看上边的原理图)。

假如你想显示心形应该怎么办?

如果我们想显示上面的这个图形,可知,第一行和第八行是不需要接地的,其他的需要接地。

但是看列,每一列都能用到。如果按照上面说的接线就是下图这个样子(不懂看原理图)。

所以不能采取这种方法,我们采取“扫描”的方式,即一行一行的亮,也就是先第一行接地(其他行不接地),然后如果按照心形的图,就是所有都列都不接正极。然后接着第二行接地(其他行不接地),由心形图可知,需要第二、三、六、七列亮,则这几列接正。再往下是第三行,第三行再接地,然后第一、四、五、八列接正,依次往下。循环一圈后接着再第一行然后一直循环即可。

或许大家会疑惑,假如第二行接地,只有第二行的几个灯亮,那么其他行都是暗的,怎么会看到心形呢,其实这个每一行的切换时间需要很短,几毫秒左右,利用人眼的视觉暂留效应(不懂得可以查一下)来显示,正所谓 ”眼见不一定为实“ 啊。

2、点阵的滚动显示

本实验应为才有行扫描所以上下滚动显示,当然也可以左右,大家可以自己探索一下。

滚动显示和静态类似,只不过差别是,在一次循环后回到第一行时,静态情况下是显示第一行应有的列,而滚动是显示上次循环第二行对应的列,第二行就显示第三行对应的列,依次往下,这样就相当于图形往上移了一个格。然后每次循环都说是这样,这样就滚动显示了。不懂可以再看代码。

3、74hc595的原理

详细原理见我上一篇文章

https://blog.csdn.net/qq_41639829/article/details/82151706

四、取模软件

软件来源于网络(使用方法百度一下):

链接:https://pan.baidu.com/s/1fRfJ_QPTqsN8LM9Npya_lQ 密码:avhl

五、代码展示

#include<reg52.h>
#include<intrins.h>

//intrins.h函数,一般出现在C51单片机编程中,一般程序中需要使用到空指令_nop_();字符循环移位指令_crol_等时使用
//OE	第13脚	输出有效(低电平)
//MR	第10脚	主复位(低电平)

typedef unsigned char uchar;
typedef unsigned int uint;


/**********管脚说明********************/
#define P P1   //define 不用加;

sbit SER = P3^4;    //p3.4串行数据输入   DS
sbit SCK = P3^6;    //串行输入时钟,上升沿有效    SHCP	第11脚	数据输入时钟线
sbit RCK = P3^5;    //串行寄存器时钟,上升沿有效  STCP	第12脚	输出存储器锁存时钟线

/***********************************/

/**********函数声明********************/
void SendTo595(uchar byteData);
/***********************************/

/**********全局参数声明********************/
char shu=0xFE;
/***********************************/


void delay(unsigned int n)//延时函数
{
    unsigned int i=0,j=0;
    for(i=0;i<n;i++)
        for(j=0;j<123;j++);
}

void hang()//行扫描
{
	P=shu;
	shu=_crol_(shu,1);
}

char a[32]={0x3E,0x08,0x08,0x08,0x08,0x08,0x08,0x3E,
						0x00,0x66,0x99,0x81,0x42,0x24,0x18,0x00,
						0x42,0x42,0x42,0x42,0x42,0x42,0x3C,0x00,
					  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00};
void main()
{
  int i,b=0;
	while(1)
	{
			int num=12;
			while(num--)
			{
				for(i=0;i<8;i++)
				{
					hang();     //先位选再段选
					SendTo595(a[i+b]);			
					delay(3);
				}
				//P=0xFF;
				//SendTo595(0x00);
			}
		  b++;
			if(b>=24)b=0;
			//P=0xFF;
	 }	
}



//发送一个字节数据给595再并行输出
void SendTo595(uchar Data)
{
   int j=0;
   for(j;j<8;j++)
   {
		    SCK = 0; 
        SER=0x80&Data;//&为按位运算符,即全1为1,有0为0,上式也就是 (1000 0000)&(1111 1111)=1000 0000,若高位为1则是1高位为0则这个式子为0 
        Data=_crol_(Data,1); //左移一位 将高位补给低位,如果二进制数为01010101 那么_crol_(1) 为10101010
        SCK = 1;          //上升沿让串行输入时钟变成高电平 并延时两个时钟周期
        _nop_();

   }  
   
   /*位移寄存器完毕,转移到存储寄存器*/
	 
	
	 RCK = 0;
   RCK = 1;         //上升沿,存储寄存器变为高电平 延迟两个时钟周期
   _nop_();
	 RCK = 0;

}

猜你喜欢

转载自blog.csdn.net/qq_41639829/article/details/82222753