看此篇博文之前建议先看博主的上一篇博文: 51单片机教程:8*8 点阵显示字符、数字、简单汉字
取走点赞哦~
资料链接:点阵16乘16.rar 一、点阵原理 市面上有很多种类的点阵,但是最常用的还是这种8乘8点阵 内部结构如下图 可以发现点阵就是由led构成,并且有一排是阳极,另一排是阴极,有一排是控制行,另一排是控制列。 打开proteus搜索MATRIX,看到各种型号的点阵,选择8乘8即可 任意选择一个颜色的点阵,我们发现引脚没有做任何标记,默认上排:阴极、控制列;下排:阳极、控制行 (一定要自己测一下,proteus仿真布置的点阵每次方向都可能发生改变) 什么意思呢,也就是跟下图对应关系,只不过引脚位置发生改变
二、4个 “ 2乘2 ” 点阵拼成一个 “ 4乘4” 点阵 我们先从简单开始理解,首先画出4个 “2乘2” 点阵 然后把上下方向的阴极控制列连接起来 把左右方向的阳极控制行连接起来 就拼成了一个4*4点阵 二、4个 “ 8乘8 ” 点阵拼成一个 “ 16乘16” 点阵 注意:proteus的点阵真的很难用,找不到方向,得自己测 测量两个内容:控制列还是控制行、阴极还是阳极
三、仿真:依次显示 “化作尘”,间隔为1秒
这里使用了74hc138与74hc595,不了解原理的自行百度,不做讲解 对应代码:
/* hc595与hc138控制8*8点阵 一开始写这个程序的时候,程序运行占用存储空间超过了128b 错误: 点阵(四)16乘16.C(98): error C249: 'DATA': SEGMENT TOO LARGE 导致程序无法运行,原因是数组占用了大量的内存 后来在数组名前加了code解决了这个问题 意思是把数组存储到程序存储区4kb而不是 data区128b */ #include <regx51.h> sbit SH_CP = P3^0; sbit DS = P3^1; sbit ST_CP = P3^2; //不需要改变的数组保存到code程序存储区 unsigned char code b[3][32] = { /*-- 文字: 化 --*/ /*-- 宋体12; 此字体下对应的点阵为:宽x高=16x16 --*/ 0x10,0x01,0x10,0x01,0x10,0x21,0x08,0x11,0x08,0x09,0x0C,0x05,0x0C,0x03,0x0A,0x01, 0x89,0x01,0x48,0x01,0x28,0x01,0x08,0x41,0x08,0x41,0x08,0x41,0x08,0x7E,0x08,0x00, /*-- 文字: 作 --*/ /*-- 宋体12; 此字体下对应的点阵为:宽x高=16x16 --*/ 0x90,0x00,0x90,0x00,0x90,0x00,0x88,0x7F,0x48,0x01,0x4C,0x01,0x2C,0x01,0x0A,0x1F, 0x09,0x01,0x08,0x01,0x08,0x01,0x08,0x3F,0x08,0x01,0x08,0x01,0x08,0x01,0x08,0x01, /*-- 文字: 尘 --*/ /*-- 宋体12; 此字体下对应的点阵为:宽x高=16x16 --*/ 0x80,0x00,0x80,0x00,0x90,0x04,0x90,0x08,0x88,0x10,0x84,0x20,0x82,0x20,0x00,0x00, 0x80,0x00,0x80,0x00,0xFC,0x1F,0x80,0x00,0x80,0x00,0x80,0x00,0xFF,0x7F,0x00,0x00, }; void delayms(unsigned int m) { int i,j; for(i=0; i<m; i++) for(j=0; j<120; j++); } void hc_595(unsigned int temp,unsigned int temp1) //595段码程序 { char t; bit val; for(t=0; t<8; t++) { val = temp & 0x80; //第一个数据传到第一个595输出 DS = val; SH_CP = 0;; SH_CP = 1; temp = temp<<1; } for(t=0; t<8; t++) { val = temp1 & 0x80; //第二个数据传输到第二个595输出 DS = val; SH_CP = 0;; SH_CP = 1; temp1 = temp1<<1; } ST_CP = 0;; ST_CP = 1; } void hc_138() { unsigned int n,t,i; for(i=0; i<3; i++) //显示3个汉字 { for(t=0; t<30; t++) //用来延时1秒 { for(n=0; n<16; ++n) //一次显示一个汉字 { if(n<8) //第一个138的高电平向下移动 { P3_6 = 1; //第一个138使能,第二个138不工作 P2 = n; } else //第二个138的高电平向下移动 { P3_6 = 0; //第二个138使能,第一个138不工作 P2 = n-8; } hc_595(b[i][2*n],b[i][2*n+1]); //发送段码 delayms(2); } } } } void main(void) { while(1) { hc_138(); } }