視頻鏈接:https://www.bilibili.com/video/BV1at4y117Eq
#include
#define uchar unsigned char
#define uint unsigned int
#define LCD1602_DATAPINS P0
sbit LCD1602_E=P2^2;
sbit LCD1602_RW=P2^1;
sbit LCD1602_RS=P2^0;
sbit KEY = P3^0;
sbit h1 = P1^0;
sbit h2 = P1^1;
sbit h3 = P1^2;
sbit h4 = P1^3;
sbit l1 = P1^4;
sbit l2 = P1^5;
sbit l3 = P1^6;
sbit l4 = P1^7;
/*******************************************************************************
* 函 數 名 : Lcd1602_Delay1ms
* 函數功能 : 延時函數,延時1ms
* 輸 入 : c
* 輸 出 : 無
* 說 名 : 該函數是在12MHZ晶振下,12分頻單片機的延時。
*******************************************************************************/
void Lcd1602_Delay1ms(uint c) //誤差 0us
{
uchar a,b;
for (; c>0; c--)
{
for (b=199;b>0;b--)
{
for(a=1;a>0;a--);
}
}
}
/*******************************************************************************
* 函 數 名 : LcdWriteCom
* 函數功能 : 向LCD寫入一個字節的命令
* 輸 入 : com
* 輸 出 : 無
*******************************************************************************/
void LcdWriteCom(uchar com) //寫入命令
{
LCD1602_E = 0; //使能
LCD1602_RS = 0; //選擇發送命令
LCD1602_RW = 0; //選擇寫入
LCD1602_DATAPINS = com; //放入命令
Lcd1602_Delay1ms(1); //等待數據穩定
LCD1602_E = 1; //寫入時序
Lcd1602_Delay1ms(5); //保持時間
LCD1602_E = 0;
}
/*******************************************************************************
* 函 數 名 : LcdWriteData
* 函數功能 : 向LCD寫入一個字節的數據
* 輸 入 : dat
* 輸 出 : 無
*******************************************************************************/
void LcdWriteData(uchar dat) //寫入數據
{
LCD1602_E = 0; //使能清零
LCD1602_RS = 1; //選擇輸入數據
LCD1602_RW = 0; //選擇寫入
LCD1602_DATAPINS = dat; //寫入數據
Lcd1602_Delay1ms(1);
LCD1602_E = 1; //寫入時序
Lcd1602_Delay1ms(5); //保持時間
LCD1602_E = 0;
}
/*******************************************************************************
* 函 數 名 : LcdInit()
* 函數功能 : 初始化LCD屏
* 輸 入 : 無
* 輸 出 : 無
*******************************************************************************/
void LcdInit() //LCD初始化子程序
{
LcdWriteCom(0x38); //開顯示
LcdWriteCom(0x0c); //開顯示不顯示光標
LcdWriteCom(0x06); //寫一個指針加1
LcdWriteCom(0x01); //清屏
LcdWriteCom(0x80); //設置數據指針起點
}
//按指定位置顯示一個字符
void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData)
{
Y &= 0x1;
X &= 0xF; //限制X不能大于15,Y不能大于1
if (Y) X |= 0x40; //當要顯示第二行時地址碼+0x40;
X |= 0x80; // 算出指令碼
LcdWriteCom(X); //這里不檢測忙信號,發送地址碼
LcdWriteData(DData);
}
//按指定位置顯示一串字符
void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData)
{
unsigned char ListLength;
ListLength = 0;
Y &= 0x1;
X &= 0xF; //限制X不能大于15,Y不能大于1
while (DData[ListLength]>=0x20) //若到達字串尾則退出
{
if (X <= 0xF) //X坐標應小于0xF
{
DisplayOneChar(X, Y, DData[ListLength]); //顯示單個字符
ListLength++;
X++;
}
}
}
void KEY_scan()
{
if(KEY==0)
{
Lcd1602_Delay1ms(20);
if(KEY==0)
{
while(KEY==0);
}
}
}
void key_scan1()
{
h1=1;
h2=1;
h3=1;
h4=1;
l1=1;
l2=1;
l3=1;
l4=1;
h1=0;
if(l1==0 && h1==0 && h2==1 && l2==1 && l3==1 && l4==1 ) //1
{
while(l1==0);
DisplayOneChar(0,0,'0');
}else if(l1==1 && h1==0 && h2==1 && l2==0 && l3==1 && l4==1 ) //2
{
while(l2==0);
DisplayOneChar(0,0,'1');
}else if(l1==1 && h1==0 && h2==1 && l2==1 && l3==0 && l4==1)//3
{
while(l3==0);
DisplayOneChar(0,0,'2');
}else if(l1==1 && h1==0 && h2==1 && l2==1 && l3==1 && l4==0)//4
{
while(l4==0);
DisplayOneChar(0,0,'3');
}
else
{
h1=1;
h2=0;
if(l1==0 && h1==1 && h2==0 && l2==1 && l3==1 && l4==1 )//6
{
while(l1==0);
}else if(l1==1 && h1==1 && h2==0 && l2==0 && l3==1 && l4==1 )//7
{
while(l2==0);
}else if(l1==1 && h1==1 && h2==0 && l2==1 && l3==0 && l4==1)//8
{
while(l3==0);
}else if(l1==1 && h1==1 && h2==0 && l2==1 && l3==1 && l4==0)//9
{
while(l4==0);
}else
{
h1=1;
h2=1;
h3=0;
if(l1==0 && h1==1 && h2==1 && h3==0 && l2==1 && l3==1 && l4==1)//11
{
while(l1==0);
}else if(l1==1 && h1==1 && h2==1 && h3==0 && l2==0 && l3==1 && l4==1)//12
{
while(l2==0);
}else if(l1==1 && h1==1 && h2==1 && h3==0 && l2==1 && l3==0 && l4==1)//13
{
while(l3==0);
}else if(l1==1 && h1==1 && h2==1 && h3==0 && l2==1 && l3==1 && l4==0 )//14
{
while(l4==0);
}
else
{
h3=1;
h4=0;
if(l1==0 && h1==1 && h2==1 && h3==4 && l2==1 && l3==1 && l4==1)//11
{
while(l1==0);
}else if(l1==1 && h1==1 && h2==1 && h3==4 && l2==0 && l3==1 && l4==1)//12
{
while(l2==0);
}else if(l1==1 && h1==1 && h2==1 && h3==4 && l2==1 && l3==0 && l4==1)//13
{
while(l3==0);
}else if(l1==1 && h1==1 && h2==1 && h3==4 && l2==1 && l3==1 && l4==0 )//14
{
while(l4==0);
}
}
}
}
}
void main()
{
LcdInit();
EA=1; //全局中斷開
EX0=1; //外部中斷0開() EX1 為外部中斷1 (P3^3)
IT0=1; //低電平觸發 IT1為中斷1
EX1=1; //外部中斷0開() EX1 為外部中斷1 (P3^3)
IT1=1; //低電平觸發 IT1為中斷1
/*
DisplayOneChar(0,0,(char)(datt/100+'0'));
DisplayOneChar(1,0,(char)(datt/10%10+'0'));
DisplayOneChar(2,0,(char)(datt%10+'0'));
*/
while(1)
{
key_scan1();
}
}
void Key_INT1( ) interrupt 0 // INT0 1 INT1 2
{
DisplayOneChar(0,0,'A');
}
void Key_INT2( ) interrupt 2 // INT0 1 INT1 2
{
DisplayOneChar(0,0,'B');
}
審核編輯:湯梓紅
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。
舉報投訴
-
Proteus
+關注
關注
76文章
1688瀏覽量
105749 -
中斷
+關注
關注
5文章
884瀏覽量
41063 -
矩陣
+關注
關注
0文章
407瀏覽量
34311 -
按鍵
+關注
關注
4文章
221瀏覽量
57392
發布評論請先 登錄
相關推薦
掃描矩陣鍵盤的三種方法:行列掃描法,反轉法,狀態機法
本帖最后由 默默地努力 于 2013-11-26 15:08 編輯
實現矩陣鍵盤掃描的三種方法(代碼見附件):1.行列掃描法2.反轉法3.狀態機法(結合定時器中斷)第一種和第二
發表于 11-26 15:04
proteus矩陣鍵盤仿真電路
×4矩陣鍵盤原理1.逐行掃描2.行列掃描(二)8×8矩陣鍵盤設計1.proteus電路圖設計2.程序設計3.矩陣鍵盤仿真效果(三)總結(一)
發表于 07-15 08:27
STM32F767+STM32CubeMX I2C通信讀寫EEPROM數據(采用輪詢、DMA、中斷三種方式)精選資料推薦
STM32F767+STM32CubeMX I2C通信讀寫EEPROM數據(采用輪詢、DMA、中斷三種方式)摘要-前言作為一名STM32的初學者,在學習過程中會遇到很多問題,解決過程中
發表于 08-23 09:08
單片機三種常用的軟件架構
一個合適的軟件架構不僅結構清晰,而且可以便于開發與維護。便于幫助開發者把握整個工程的框架三種常用的軟件架構:1.順序執行的前后臺系統2.時間片輪詢系統3.多任務操作系統順序執行的前后臺系統程序以變量
發表于 01-07 08:29
【STM32】4*4矩陣鍵盤掃描程序(中斷方式)
上一篇雜記寫了個掃描方式的鍵盤程序,但是最近在做一個小游戲對CPU響應速度有要求,于是再弄個簡單的鍵盤中斷檢測程序吧。 總體思路:中斷線為4行連接的GPIO口,先初始化矩陣的4行輸
發表于 11-26 11:21
?17次下載
Proteus中以中斷方式實現矩陣鍵盤的掃描
前言最近做單片機的課程設計用到矩陣鍵盤,在此做個記錄。1 矩陣鍵盤的掃描方式使用矩陣鍵盤時,首先要判斷是否有按鍵按下,這個過程稱為
發表于 11-26 12:21
?9次下載
評論