51单片机控制数码管动态显示

首先打开proteus,导入8位数码管和89c51。

然后如图连线,分清断码和位码, 断码就是一个数码管的7个LED灯。

位码:就是第几位显示,由于是共阴极,所以哪位接地就显示哪位。

下面通过改变位码的接线就可以看出不同的效果

下面就编写程序,从第1位到第8位显示从0到7的八个数字。也就是位码的第一位为低电平,然后断码的除了g灯不亮其他都亮也就是0x3f,这样第一位就显示了0,后面就依次类推。

下面写c程序,之后编译。可复制后直接享用!

#include"reg51.h"

unsigned char dat[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
unsigned char wei[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};
//  点        g        f        e        d        c        b        a
// 显示0    0        1        1        1        1        1        1   除了点和g不亮其他的灯都亮就是0。  十六进制就是0x3f
// 显示1    0        0        0        0        1        1        0   只有b和c亮,其他的都不亮。        十六进制就是0x06
// 按照上面的方式依次类推得到 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f


void delay(unsigned int n){       //延时函数   不是很精确
    unsigned int i=0, j=0;
    for(i=0; i<n; i++){
        for(j=0; j<120; j++);
    }
}

void seg(){                       // 自己命名的子函数,proteus中数码管叫seg,我这里就取名叫seg好了
    unsigned int i;
    for(i=0; i<8; i++){          //  循环10次,得到i的值从0到7都能取到
        P3=~wei[i];             //  位码从第0位到第7位
        P2=dat[i];                // 断码从0到7显示
        delay(5);                  // 延时不能太长了,太长了人眼就能捕捉到有灯灭掉了,就不是持续点亮的状态了。
    }
}

void main(){                      // 主函数   加一个无限循环  只调用了子函数seg
    while(1){
        seg();
    }
}

把编译的hex文件导入后就可以看到0到7都显示出来了,而且也不闪动,如果有闪动请降低延时数。

由于这样接线太占用IO端口了,所以我们选用一个译码器来减少IO口的使用,用3个口控制8个口,这个译码器叫74ls138。

如上图所示,这个芯片本省就带取反功能,也就是Y0输出1的条件是输入000 P端口十六进制为0x00

Y1输出1的条件是输入001 P端口十六进制为0x01

Y2输出1的条件是输入010 P端口十六进制为0x02

Y3输出1的条件是输入011 P端口十六进制为0x03

Y4输出1的条件是输入100 P端口十六进制为0x04

Y5输出1的条件是输入101 P端口十六进制为0x05

Y6输出1的条件是输入110 P端口十六进制为0x06

Y7输出1的条件是输入111 P端口十六进制为0x07

根据这样的条件就可以用3个IO口控制数码管的8位位码了,节省5个IO口。如下图所示:

下面编写c程序并编译。

#include"reg51.h"

unsigned char dat[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
unsigned char wei[]={0x00,0x01,0x02,0x03,0x04,0x05,0x06,0x07};
//  点        g        f        e        d        c        b        a
// 显示0    0        1        1        1        1        1        1   除了点和g不亮其他的灯都亮就是0。  十六进制就是0x3f
// 显示1    0        0        0        0        1        1        0   只有b和c亮,其他的都不亮。        十六进制就是0x06
// 按照上面的方式依次类推得到 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f


void delay(unsigned int n){       //延时函数   不是很精确
    unsigned int i=0, j=0;
    for(i=0; i<n; i++){
        for(j=0; j<120; j++);
    }
}

void seg(){                       // 自己命名的子函数,proteus中数码管叫seg,我这里就取名叫seg好了
    unsigned int i;
    for(i=0; i<8; i++){          //  循环10次,得到i的值从0到7都能取到
        P3=wei[i];             //  位码从第0位到第7位输出给译码器,同样也能控制第0位到第七位。
        P2=dat[i];                // 断码从0到7显示
        delay(5);                  // 延时不能太长了,太长了人眼就能捕捉到有灯灭掉了,就不是持续点亮的状态了。
    }
}

void main(){                      // 主函数   加一个无限循环  只调用了子函数seg
    while(1){
        seg();
    }
}

proteus导入编译好的hex文件后模拟运行,就可以看到数码管从0到7显示出来了。

好了,这就是本文的主要内容。

物联沃分享整理
物联沃-IOTWORD物联网 » 51单片机控制数码管动态显示

发表评论