3.7 51单片机-LED 16*16点阵

Posted DS小龙哥

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了3.7 51单片机-LED 16*16点阵相关的知识,希望对你有一定的参考价值。

3.7 LED 16*16点阵

3.7.1 点阵原理图

图3-7-1

LED点阵使用连线说明:P595_A接J17、P595_B接J18、JP595跳线帽需要接上、JP1302断开、J11(P3.3)断开

图3-7-2 

图3-7-3 8*8的点阵原理图

从上面LED点阵原理图中看出,点阵里的每个LED的阴端接在行上面,阳端接在列上面

 图3-7-4 二极管

 图3-7-5 发光二极管

图3-7-6 二极管电路图

点亮一个LED二极管只需要根据正负极给对应的电源即可发光。

根据实验板的16*16点阵原理图来看,只要给横向()16个IO输出低电平,纵向(列)16个IO口输出高电平,即可点亮点阵上面的所有LED

3.7.2 点阵原理介绍

实验板上的16*16点阵是由4个8*8LED点阵组成的,一个 8*8 的点阵是由 64个LED小灯组成。

16*16点阵分别由4片74HC595控制,4片74HC595采用级联的方式连接。

图3-7-7 16*16LED点阵

3.7.3 74HC595缓存器

74HC595是一个8位串行输入、平行输出的位移缓存器:平行输出为三态输出。

在SCK的上升沿,单行数据由SDL输入到内部的8位位移缓存器,并由Q7‘输出,而平行输出则是在LCK的上升沿将在8位位移缓存器的数据存人到8位平行输出缓存器。

当串行数据输人端OE的控制信号为低使能时,平行输出端的输出值等于平行输出缓存器所存储的值。而当OE为高电位,也就是输出关闭时,平行输出端会维持在高阻抗状态。

74HC595支持级联,当多个595级联到一起时,通过数据线发送的一个数据最终会移位给最后的一个595

因为级联数据是被挤出到下一级的,所以先发送的数据最后是到最后一级的595。

每次向74HC595发送一个字节,74HC595最先收到的一位数据是高位,也就是最先收到的数据就到达Q7脚,如果我们先输出数据高位的话,最高位在8个脉冲后就会跑到Q7脚(数据脚最高位)。这就像我们排队一样,一个寄存器里面有8个位置,每次给一个脉冲就好比一次呼叫:“大家可以往前移一位了!”就这样,队伍不断得往前移。

这个过程就像在向前挤一样:

图3-7-8 

 

图3-7-9 

 

图3-7-10 HC595管脚功能图 

 

图3-7-11 HC595真值表

通过真值表得知:

发送一个数据0的时序代码如下:

HC595_DATA=0;HC595_SCK=0;HC595_SCK=1;

发送一个数据1的时序代码如下:

HC595_DATA=1;HC595_SCK=0;HC595_SCK=1;

3.7.4 示例代码: LED点阵单点扫描显示

下面代码实现,点阵依次从左上角点亮一个点,等待一段时间,再关闭所有点,再点亮第二个点,等待一段时间,再关闭所有点,从左到右,从上到下,依次把所有点亮都点亮一遍。

#include <reg51.h>
//定义HC595使用的IO口
sbit HC595_DATA=P3^4; //串行数据输出
sbit HC595_RCK=P3^5;  //存储寄存器时钟
sbit HC595_SCK=P3^6;  //移位寄存器时钟

/*
通过HC595发送一个字节数据
先发送高位,在HC959上对应的是低位(也就是LED点阵的最左边)
*/
void HC595_Send_Byte(u8 byte)
{
    u8 i;
    for(i=0;i<8;i++)
    {
        if(byte&0x80)HC595_DATA=1;
        else HC595_DATA=0;
        HC595_SCK=0;
        HC595_SCK=1;
        byte<<=1; //依次发送高位
    }
}

/*
将HC959存储器里的数据输出到总线上
*/
void HC595_DataOut(void)
{
    //将移位存储器的输出并行输出到总线上
    HC595_RCK=0;
    HC595_RCK=1;
    HC595_RCK=0;
}

/*
HC595的3、4级是控制点阵的列 (低电平)
HC595的1、2级是控制点阵的行 (高电平)
给HC595第1级发送的数据会移动到最后一个HC595。
*/
int main()
{
    u8 i,j;
    u16 Row; //行
    u16 col; //列
    while(1)
    {
        for(i=0;i<16;i++)
        {
            Row=0x0000;
            Row|=1<<i; //行
            for(j=0;j<16;j++)
            {
                col=0xFFFF;
                col&=~(1<<j); //列
                HC595_Send_Byte(col>>8);
                HC595_Send_Byte(col);
                HC595_Send_Byte(Row>>8);
                HC595_Send_Byte(Row);
                HC595_DataOut();  
                DelayMs(100); //延时一段时间,再关闭所有显示
                HC595_Send_Byte(0xFF);
                HC595_Send_Byte(0xFF);
                HC595_Send_Byte(0x00);
                HC595_Send_Byte(0x00);
                HC595_DataOut(); 
            }
        }
    }
}

3.7.5 示例代码: 控制LED点阵显示指定效果

(1). 实现点阵按照指定时间间隔全亮和全灭

#include <reg51.h>
int main()
{
    u8 i,j;
    while(1)
    {
      //下面4行代码,开启点阵
      HC595_Send_Byte(0x00);//4列: 低位对应右边第8列开始,8~15
      HC595_Send_Byte(0x00);//3列: 低位对应左边第0列开始,0~7
      HC595_Send_Byte(0xFF);//2行: 低位对应上面第8行,依次8~15
      HC595_Send_Byte(0xFF);//1行: 低位对应上面第0行,依次0~7
      HC595_DataOut();  
      DelayMs(500);
      //下面4行代码,关闭点阵
      HC595_Send_Byte(0xFF);//4列: 低位对应右边第8列开始,8~15
      HC595_Send_Byte(0xFF);//3列: 低位对应左边第0列开始,0~7
      HC595_Send_Byte(0x00);//2行: 低位对应上面第8行,依次8~15
      HC595_Send_Byte(0x00);//1行: 低位对应上面第0行,依次0~7
      HC595_DataOut();  
      DelayMs(500);
}
}

(2). 实现点阵旋转效果,按照8*8为单位,点亮4个小方块

点亮的顺序,看下面代码的注释。

int main()
{

    while(1)
    {
        //点亮左上角8*8方块
        HC595_Send_Byte(0xFF);
        HC595_Send_Byte(0x00);
        HC595_Send_Byte(0x00);
        HC595_Send_Byte(0xFF);
        HC595_DataOut();
        DelayMs(100);
        
        //点亮右上角8*8方块
        HC595_Send_Byte(0x00);
        HC595_Send_Byte(0xFF);
        HC595_Send_Byte(0x00);
        HC595_Send_Byte(0xFF);
        HC595_DataOut();
        DelayMs(100);
        
        //点亮右下角8*8方块
        HC595_Send_Byte(0x00);
        HC595_Send_Byte(0xFF);
        HC595_Send_Byte(0xFF);
        HC595_Send_Byte(0x00);
        HC595_DataOut(); 
        DelayMs(100);  
        
        //点亮左下角8*8方块
        HC595_Send_Byte(0xFF);
        HC595_Send_Byte(0x00);
        HC595_Send_Byte(0xFF);
        HC595_Send_Byte(0x00);
        HC595_DataOut();
        DelayMs(100);      
}
}

3.7.6 示例代码: 控制LED点阵显示汉字

使用LED点阵显示汉字,需要用到汉字取模软件,本章节使用的取模软件“PCtoLCD2002”,软件的取模方式配置如下。

图3-7-12 

 

图3-7-13

示例代码:  在LED点阵上显示一个中文

#include <reg51.h>
//定义HC595使用的IO口
sbit HC595_DATA=P3^4; //串行数据输出
sbit HC595_RCK=P3^5;  //存储寄存器时钟
sbit HC595_SCK=P3^6;  //移位寄存器时钟
/*
通过HC595发送一个字节数据
先发送高位,在HC959上对应的是低位(也就是LED点阵的最左边)
*/
void HC595_Send_Byte(u8 byte)
{
    u8 i;
    for(i=0;i<8;i++)
    {
        if(byte&0x80)HC595_DATA=1;
        else HC595_DATA=0;
        HC595_SCK=0;
        HC595_SCK=1;
        byte<<=1; //依次发送高位
    }
}
/*
将HC959存储器里的数据输出到总线上
*/
void HC595_DataOut(void)
{
    //将移位存储器的输出并行输出到总线上
    HC595_RCK=0;
    HC595_RCK=1;
    HC595_RCK=0;
}
/*
在LED点阵上显示一个16*16数据
*/
void LED_DisplayData(u8 *p)
{
    u8 i;
    u16 Row; //控制行的状态,每次只点亮1行
    for(i=0;i<16;i++)
    {
        Row=0x0000;
        Row|=1<<i;
        HC595_Send_Byte(*(p+1)); //控制列右8个IO
        HC595_Send_Byte(*p);     //控制列左8个IO
        p+=2;
        HC595_Send_Byte(Row>>8); //控制行下8个IO
        HC595_Send_Byte(Row);    //控制行上8个IO
        HC595_DataOut();
    } 
}
u8 code font[] = {0x7F,0xFF,0x7F,0xFF,0x7F,0xFF,0x7F,0xFF,0x03,0xE0,0x7B,0xEF,0x7B,0xEF,0x7B,0xEF,0x7B,0xEF,0x7B,0xEF,0x03,0xE0,0x7B,0xEF,0x7F,0xFF,0x7F,0xFF,0x7F,0xFF,0x7F,0xFF};/*"中",0*/
int main()
{
    while(1)
    {
        LED_DisplayData(font); //显示中文
    }        
}

图3-7-14

3.7.7 示例代码: 控制LED点阵显示多个中文

下面代码实现,在LED点阵上循环显示多个文字或者图案,所有的功能都使用子函数封装,调用逻辑清晰。

#include <reg51.h>
//定义HC595使用的IO口
sbit HC595_DATA=P3^4; //串行数据输出
sbit HC595_RCK=P3^5;  //存储寄存器时钟
sbit HC595_SCK=P3^6;  //移位寄存器时钟

/*
通过HC595发送一个字节数据
先发送高位,在HC959上对应的是低位(也就是LED点阵的最左边)
*/
void HC595_Send_Byte(u8 byte)
{
    u8 i;
    for(i=0;i<8;i++)
    {
        if(byte&0x80)HC595_DATA=1;
        else HC595_DATA=0;
        HC595_SCK=0;
        HC595_SCK=1;
        byte<<=1; //依次发送高位
    }
}

/*
将HC959存储器里的数据输出到总线上
*/
void HC595_DataOut(void)
{
    //将移位存储器的输出并行输出到总线上
    HC595_RCK=0;
    HC595_RCK=1;
    HC595_RCK=0;
}

/*
在LED点阵上显示一个16*16数据
*/
void LED_DisplayData(u8 *p)
{
    u8 i;
    u16 Row; //控制行的状态,每次只点亮1行
    for(i=0;i<16;i++)
    {
        Row=0x0000;
        Row|=1<<i;
        HC595_Send_Byte(*(p+1)); //控制列右8个IO
        HC595_Send_Byte(*p);     //控制列左8个IO
        p+=2;
        HC595_Send_Byte(Row>>8); //控制行下8个IO
        HC595_Send_Byte(Row);    //控制行上8个IO
        HC595_DataOut();
    } 
}

/*
关闭点阵所有点
*/
void LED_Clear(void)
{
    HC595_Send_Byte(0xFF);//4列: 低位对应右边第8列开始,8~15
    HC595_Send_Byte(0xFF);//3列: 低位对应左边第0列开始,0~7
    HC595_Send_Byte(0x00);//2行: 低位对应上面第8行,依次8~15
    HC595_Send_Byte(0x00);//1行: 低位对应上面第0行,依次0~7
    HC595_DataOut(); 
}

code u8 buff[][32]=
{
    {0xFF,0xFF,0xFF,0xFF,0x00,0x80,0xDF,0xFF,0xDF,0xFF,0xDF,0xFF,0x1F,0xF0,0xDF,0xF7,0xDF,0xF7,0xEF,0xF7,0xEF,0xF7,0xF7,0xF7,0xF7,0xF7,0xFB,0xF7,0xFD,0xFA,0xFE,0xFD},/*"万",0*/
    {0xEF,0xFF,0xEF,0xC1,0xEF,0xDD,0x00,0xED,0xEF,0xED,0xEF,0xF5,0x81,0xED,0xEF,0xED,0xEF,0xDD,0x00,0xDD,0xEF,0xDD,0xF7,0xE9,0xF7,0xF5,0xFB,0xFD,0xFD,0xFD,0xFE,0xFD},/*"邦",1*/
    {0x0F,0xF0,0xEF,0xF7,0xEF,0xF7,0x0F,0xF0,0xEF,0xF7,0xEF,0xF7,0x0F,0xF0,0xDF,0xFF,0xEF,0xFF,0x07,0xC0,0xBB,0xDD,0xBD,0xDD,0xDF,0xDE,0xEF,0xDE,0x77,0xEB,0xBF,0xF7},/*"易",2*/
    {0x7F,0xFF,0x7B,0xEF,0x7B,0xEF,0x03,0xE0,0xFF,0xFB,0xBB,0xFB,0xBB,0x83,0x00,0xBD,0xBB,0xD6,0xBB,0xF7,0x83,0xF7,0xBB,0xF7,0xBB,0xEB,0x83,0xEB,0xBB,0xDD,0xFF,0xBE},/*"嵌",3*/
};

#define TIME 100 
int main()
{
    u8 i,j;
    while(1)
    {
        for(i=0;i<sizeof(buff)/sizeof(buff[0]);i++)
        {
            for(j=0;j<TIME;j++) //停留的时间
            {
               LED_DisplayData(buff[i]); //显示中文,需要循环调用
            }
            LED_Clear();  //清除显示
        }
}        
}

以上是关于3.7 51单片机-LED 16*16点阵的主要内容,如果未能解决你的问题,请参考以下文章

51单片机开发:通过74HC595控制LED点阵

51单片机 16*64LED单红点阵屏驱动测试

单片机LED16*16点阵显示实验

基于51单片机Proteus的8X8led点阵心滚动

基于单片机16点阵LED汉字滚动显示设计-全套资料

51单片机 16X16点阵动态显示+Proteus仿真