c语言逐行扫描打印图案,用逐行扫描法读取4X4键盘矩阵,不能扫描出第一列按键?...

建议找成熟稳定的程序抄。以下程序来自好书《51单片机轻松入门-基于STC15W4K系列》

#include "STC15W4K.H"      // 注意宏定义后面没分号

#include "KEY.H"

#include "Dynamic_Display.H"   // 动态显示相关

extern           unsigned char DispBuf[6];   // 6字节的显示缓冲区,DispBuf[0]是最低位;

#define         Hidden         16               // 高位消隐码在数据表中的位置,DispTab[15]=0xff

void delay10ms(void)          // 22.1184MHz

{

unsigned char i,j,k;

for(i=2;i>0;i--)       // 注意后面没分号

for(j=114;j>0;j--)     // 注意后面没分号

for(k=241;k>0;k--);    // 注意后面有分号

}

unsigned char  KeyScan()

{

unsigned char key=0xff;     // 无键按下时,key=0xff;

P2=0x0f;                                // 在IO口由输出方式变为输入方式时要延迟一个时钟周期。

P2=0x0f;                                // 采取写2次的方法延时。

if (P2!=0x0f)

{

delay10ms();                // 键盘消抖,延时10MS

if (P2!=0x0f)                         // 有键按下

{

P2=0xef;                                   // 扫描第1列(逐列扫描开始)

P2=0xef;

switch (P2)

{

case 0xe7:key=0x00;break;

case 0xeb:key=0x04;break;

case 0xed:key=0x08;break;

case 0xee:key=0x0c;break;

}

//说明:本switch语句执行结束后会接着执行下面的语句,但由于P2口输出的扫描码

//      发生变化,程序不会进入后面的switch语句,也就不会多次修改key值。

P2=0xdf;                                   // 扫描第2列(逐列扫描开始)

P2=0xdf;

switch (P2)

{

case 0xd7:key=0x01;break;

case 0xdb:key=0x05;break;

case 0xdd:key=0x09;break;

case 0xde:key=0x0d;break;

}

P2=0xbf;                                   // 扫描第3列(逐列扫描开始)

P2=0xbf;

switch (P2)

{

case 0xb7:key=0x02;break;

case 0xbb:key=0x06;break;

case 0xbd:key=0x0a;break;

case 0xbe:key=0x0e;break;

}

P2=0x7f;                                   // 扫描第4列(逐列扫描开始)

P2=0x7f;

switch (P2)

{

case 0x77:key=0x03;break;

case 0x7b:key=0x07;break;

case 0x7d:key=0x0b;break;

case 0x7e:key=0x0f;break;

}

P2=0x0f;                                   // 在IO口由输出方式变为输入方式时要延迟一个时钟周期。

P2=0x0f;                                   // 采取写2次的方法延时。

while(P2!=0x0f);                   // 等待按键释放

}

}

return(key);

}

void KeyHandle(unsigned char KeyValue)

{

DispBuf[0] = KeyValue;                        // 个位显示按键值

DispBuf[5] = Hidden;                        // 十万位消隐

DispBuf[4] = Hidden;                        // 万位消隐

DispBuf[3] = Hidden;                        // 千位消隐

DispBuf[2] = Hidden;                        // 百位消隐

DispBuf[1] = Hidden;                        // 十位消隐

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
51单片机实现矩阵键盘行列扫描. #define DataPort P0 //定义数据端口 程序中遇到DataPort 则用P0 替换 #define P2 P1 sbit LATCH1=P2^2;//定义锁存使能端口 段锁存 sbit LATCH2=P2^3;// 位锁存 unsigned char code dofly_DuanMa[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f, 0x77,0x7c,0x39,0x5e,0x79,0x71};// 显示段码值0~F unsigned char code dofly_WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分别对应相应的数码管点亮,即位码 unsigned char TempData[10]; //存储显示值的全局变量 void DelayUs2x(unsigned char t);//us级延时函数声明 void delay(unsigned char t); //ms级延时 void Display(unsigned char FirstBit,unsigned char Num);//数码管显示函数 unsigned char KeyScan(void);//键盘扫描 unsigned char KeyPro(void); void Init_Timer0(void);//定时器初始化 /*------------------------------------------------ 主函数 ------------------------------------------------*/ void main (void) { unsigned char num,i,j; unsigned char temp[8]; Init_Timer0(); while (1) //主循环 { num=KeyPro(); if(num!=0xff) { if(i<8) { temp[i]=dofly_DuanMa[num]; for(j=0;j<=i;j++) TempData[7-i+j]=temp[j]; } i++; if(i==9)//多一个按键输入为了清屏 原本应该为8 { i=0; for(j=0;j<8;j++)//清屏 TempData[j]=0; } } //Display(0,8); //显示全部8位 //主循环中添加其他需要一直工作的程序 } } /*------------------------------------------------ uS延时函数,含有输入参数 unsigned char t,无返回值 unsigned char 是定义无符号字符变量,其值的范围是 0~255 这里使用晶振12M,精确延时请使用汇编,大致延时 长度如下 T=tx2+5 uS ------------------------------------------------*/ void DelayUs2x(unsigned char t) { while(--t); } /*------------------------------------------------ mS延时函数,含有输入参数 unsigned char t,无返回值 unsigned char 是定义无符号字符变量,其值的范围是 0~255 这里使用晶振12M,精确延时请使用汇编 ------------------------------------------------*/ void delay(unsigned char t) { while(t--) { //大致延时1mS DelayUs2x(245); DelayUs2x(245); } } /*------------------------------------------------ 显示函数,用于动态扫描数码管 输入参数 FirstBit 表示需要显示的第一位,如赋值2表示从第三个数码管开始显示 如输入0表示从第一个显示。 Num表示需要显示的位数,如需要显示99两位数值则该值输入2 ------------------------------------------------*/ void Display(unsigned char FirstBit,unsigned char Num) { static unsigned char i=0; DataPort=0; //清空数据,防止有交替重影 LATCH1=1; //段锁存 LATCH1=0; DataPort=dofly_WeiMa[i+FirstBit]; //取位码 LATCH2=1; //位锁存 LATCH2=0; DataPort=TempData[i]; //取显示数据,段码 LATCH1=1; //段锁存 LATCH1=0; i++; if(i==Num) i=0; } /*------------------------------------------------ 定时器初始化子程序 ------------------------------------------------*/ void Init_Timer0(void) { TMOD |= 0x01; //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响 //TH0=0x00; //给定初值 //TL0=0x00; EA=1; //总中断打开 ET0=1; //定时器中断打开 TR0=1; //定时器开关打开 } /*------------------------------------------------ 定时器中断子程序 ------------------------------------------------*/ void Timer0_isr(void) interrupt 1 { TH0=(65536-2000)/256; //重新赋值 2ms TL0=(65536-2000)%6; Display(0,8); // 调用数码管扫描 } /*------------------------------------------------ 按键扫描函数,返回扫描键值 ------------------------------------------------*/ uchar keyscan() //键盘扫描函数,使用行列逐级扫描 { unsigned char Val; P2=0xf0;//高四位置高,低四位拉低 if(P2!=0xf0)//表示有按键按下 { delay(10); //去抖 if(P2!=0xf0) { //表示有按键按下 P2=0xfe; //检测第一行 if(P2!=0xfe) { Val=C&0xf0; Val+=0x0e; while(P2!=0xfe); delay(10); //去抖 while(P2!=0xfe); return Val; } P2=0xfd; //检测第二行 if(P2!=0xfd) { Val=P2&0xf0; Val+=0x0d; while(P2!=0xfd); delay(10); //去抖 while(P2!=0xfd); return Val; } P2=0xfb; //检测第三行 if(P2!=0xfb) { Val=P2&0xf0; Val+=0x0b; while(P2!=0xfb); delay(10); //去抖 while(P2!=0xfb); return Val; } P2=0xf7; //检测第四行 if(P2!=0xf7) { Val=P2&0xf0; Val+=0x07; while(P2!=0xf7); delay(10); //去抖 while(P2!=0xf7); return Val; } } } return 0xff; } /*------------------------------------------------ 按键值处理函数,返回键值 ------------------------------------------------*/ unsigned char KeyPro(void) { switch(KeyScan()) { case 0x7e:return 0;break;//0 按下相应的键显示相对应的码值 case 0x7d:return 1;break;//1 case 0x7b:return 2;break;//2 case 0x77:return 3;break;//3 case 0xbe:return 4;break;//4 case 0xbd:return 5;break;//5 case 0xbb:return 6;break;//6 case 0xb7:return 7;break;//7 case 0xde:return 8;break;//8 case 0xdd:return 9;break;//9 case 0xdb:return 10;break;//a case 0xd7:return 11;break;//b case 0xee:return 12;break;//c case 0xed:return 13;break;//d case 0xeb:return 14;break;//e case 0xe7:return 15;break;//f default:return 0xff;break; } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值