【121.1 可調參數的數碼管倒計時。】

上圖121.1.1 數碼管

上圖121.1.2 獨立按鍵

上圖121.1.3 有源蜂鳴器
上節講如何設置數據,本節講“數據”如何關聯“某種功能”,本節的“可調參數”就是“數據”,“倒計時”就是“某種功能”。程序功能如下:
(1)倒計時范圍從0.00秒到99.99秒,范圍可調。開機默認是:10.00秒。
(2)K1[設置鍵]。當數碼管“沒有閃爍”時,“長按”K1鍵則進入“閃爍模式”,某位數碼管開始閃爍,閃爍的位代表可修改的位數據,此時再“短按”K1按鍵可以使數碼管在位之間切換閃爍。當數碼管處于“閃爍模式”時,“長按”K1按鍵,代表數據修改完成并停止閃爍。
(3)K2[加鍵]與[復位健]。當數碼管某位正在閃爍時,此時K2是[加鍵],按K2會使位數據“自加1”。當數碼管“沒有閃爍”時,此時K2是[復位鍵],按K2會使當前倒計時數據恢復“設置值”。
(4)K3[減鍵]與[開始健]。當數碼管某位正在閃爍時,此時K3是[減鍵],按K3會使位數據“自減1”。當數碼管“沒有閃爍”時,此時K3是[開始鍵],按K3開始倒計時。
代碼如下:
\#include "REG52.H"
\#define KEY\_FILTER\_TIME 25 //按鍵的“短按”兼“濾波”的“穩定時間”
\#define KEY\_LONG\_TIME 500 //按鍵的“長按”兼“濾波”的“穩定時間”
\#define SCAN\_TIME 1
\#define VOICE\_TIME 50
\#define BLINK\_TIME 250 //數碼管閃爍跳動的時間的間隔
void T0\_time();
void SystemInitial(void) ;
void Delay(unsigned long u32DelayTime) ;
void PeripheralInitial(void) ;
void KeyScan(void);
void KeyTask(void);
void VoiceScan(void);
void DisplayScan(void);
void DisplayTask(void); //上層顯示的任務函數
void RunTask(void); //倒計時的應用程序
void Wd1(void); //窗口1顯示函數。用來設置參數。
void Wd2(void); //窗口2顯示函數。倒計時的運行顯示窗口
void PartUpdate(unsigned char u8Part); //局部選擇對應的某個局部變量更新顯示輸出
void BeepOpen(void);
void BeepClose(void);
sbit KEY\_INPUT1=P2^2;
sbit KEY\_INPUT2=P2^1;
sbit KEY\_INPUT3=P2^0;
sbit P1\_0=P1^0;
sbit P1\_1=P1^1;
sbit P1\_2=P1^2;
sbit P1\_3=P1^3;
sbit P3\_4=P3^4;
//數碼管轉換表
code unsigned char Cu8DigTable\[\]=
{
0x3f, //0 序號0
0x06, //1 序號1
0x5b, //2 序號2
0x4f, //3 序號3
0x66, //4 序號4
0x6d, //5 序號5
0x7d, //6 序號6
0x07, //7 序號7
0x7f, //8 序號8
0x6f, //9 序號9
0x00, //不顯示 序號10
0x40, //橫杠- 序號11
};
volatile unsigned char vGu8ScanTimerFlag=0;
volatile unsigned int vGu16ScanTimerCnt=0;
volatile unsigned char vGu8BeepTimerFlag=0;
volatile unsigned int vGu16BeepTimerCnt=0;
volatile unsigned char vGu8BlinkTimerFlag=0; //數碼管閃爍跳動的定時器
volatile unsigned int vGu16BlinkTimerCnt=0;
//倒計時的軟件定時器,注意,這里是unsigned long類型,范圍是0到4294967295毫秒
volatile unsigned char vGu8CountdownTimerFlag=0;
volatile unsigned long vGu32CountdownTimerCnt=10000; //當前倒計時的計時值
unsigned long Gu32SetData\_Countdown=10000; //倒計時的設置值
//數碼管上層每10ms就定時刷新一次顯示的軟件定時器。用于及時更新顯示秒表當前的實時數值
volatile unsigned char vGu8UpdateTimerFlag=0;
volatile unsigned int vGu16UpdateTimerCnt=0;
unsigned char Gu8RunStart=0; //應用程序的總啟動
unsigned char Gu8RunStep=0; //應用程序的總運行步驟。建議跟vGu8RunStart成雙成對出現
unsigned char Gu8RunStatus=0; //當前倒計時的狀態。0代表停止,1代表正在工作中
unsigned char Gu8EditData\_4=0; //對應顯示右起第4位數碼管的“位”數據,是中間變量。
unsigned char Gu8EditData\_3=0; //對應顯示右起第3位數碼管的“位”數據,是中間變量。
unsigned char Gu8EditData\_2=0; //對應顯示右起第2位數碼管的“位”數據,是中間變量。
unsigned char Gu8EditData\_1=0; //對應顯示右起第1位數碼管的“位”數據,是中間變量。
unsigned char Gu8Wd=1; //窗口選擇變量。人機交互程序框架的支點。初始化開機后顯示第1個窗口。
unsigned char Gu8WdUpdate=1; //整屏更新變量。初始化為1開機后整屏更新一次顯示。
unsigned char Gu8Part=0; //局部選擇變量。0代表當前窗口下沒有數據被選中。
unsigned char Gu8PartUpdate\_1=0; //局部1的更新變量,
unsigned char Gu8PartUpdate\_2=0; //局部2的更新變量
unsigned char Gu8PartUpdate\_3=0; //局部3的更新變量,
unsigned char Gu8PartUpdate\_4=0; //局部4的更新變量
volatile unsigned char vGu8Display\_Righ\_4=1; //顯示“1”,跟vGu32CountdownTimerCnt高位一致
volatile unsigned char vGu8Display\_Righ\_3=0;
volatile unsigned char vGu8Display\_Righ\_2=0;
volatile unsigned char vGu8Display\_Righ\_1=0;
volatile unsigned char vGu8Display\_Righ\_Dot\_4=0;
volatile unsigned char vGu8Display\_Righ\_Dot\_3=1; //開機默認保留顯示2個小數點
volatile unsigned char vGu8Display\_Righ\_Dot\_2=0;
volatile unsigned char vGu8Display\_Righ\_Dot\_1=0;
volatile unsigned char vGu8KeySec=0;
void main()
{
SystemInitial();
Delay(10000);
PeripheralInitial();
while(1)
{
KeyTask(); //按鍵的任務函數
DisplayTask(); //數碼管顯示的上層任務函數
RunTask(); //倒計時的應用程序
}
}
void PartUpdate(unsigned char u8Part) //局部選擇對應的某個局部變量更新顯示輸出
{
switch(u8Part)
{
case 1:
Gu8PartUpdate\_1=1;
break;
case 2:
Gu8PartUpdate\_2=1;
break;
case 3:
Gu8PartUpdate\_3=1;
break;
case 4:
Gu8PartUpdate\_4=1;
break;
}
}
void RunTask(void) //倒計時的應用程序
{
if(0==Gu8RunStart)
{
return; // 如果總開關處于停止狀態,則直接退出當前函數,不執行該函數以下的其它代碼
}
switch(Gu8RunStep)
{
case 0: //在這個步驟里,主要用來初始化一些參數
vGu8UpdateTimerFlag=0;
vGu16UpdateTimerCnt=10; //每10ms更新顯示一次當前倒計時的時間
vGu8UpdateTimerFlag=1;
Gu8RunStep=1; //跳轉到每10ms更新顯示一次的步驟里
break;
case 1: //每10ms更新一次顯示,確保實時顯示當前倒計時的時間
if(0==vGu16UpdateTimerCnt) //每10ms更新顯示一次當前倒計時的時間
{
vGu8UpdateTimerFlag=0;
vGu16UpdateTimerCnt=10; //重置定時器,為下一個10ms更新做準備
vGu8UpdateTimerFlag=1;
Gu8WdUpdate=1; //整屏更新一次顯示當前倒計時的時間
if(0==vGu32CountdownTimerCnt) //如果倒計時的時間到,則跳轉到結束的步驟
{
Gu8RunStep=2; //跳轉到倒計時結束的步驟
}
}
break;
case 2: //倒計時結束的步驟
//Gu8RunStatus=0; //這行代碼注釋掉,讓每次新啟動之前都必須按一次K1復位按鍵才有效
Gu8RunStart=0; //倒計時的運行步驟的停止
Gu8RunStep=0; //總運行步驟歸零。建議跟vGu8RunStart成雙成對出現
vGu8BeepTimerFlag=0;
vGu16BeepTimerCnt=VOICE\_TIME; //蜂鳴器發出“滴”一聲
vGu8BeepTimerFlag=1;
Gu8WdUpdate=1; //整屏更新一次顯示當前倒計時的時間
break;
}
}
void KeyTask(void) //按鍵的任務函數
{
if(0==vGu8KeySec)
{
return;
}
if(0!=Gu8RunStatus) //在“非停止”狀態下,用return來攔截一些“不該響應”的按鍵
{
if(2==vGu8KeySec) //在“非停止”狀態下,只響應\[復位\]這個按鍵
{
; //這里沒有return語句,表示可以繼續往下掃描本函數余下的代碼,沒有被攔截。
}
else
{
return; //其余的按鍵則攔截退出
}
}
switch(vGu8KeySec)
{
case 1: //按鍵K1的“短按”。切換數碼管閃爍的位。
switch(Gu8Wd) //在某個窗口下
{
case 1: //在窗口1下
if(0!=Gu8Part) //處于“閃爍模式”的時候,是“切換局部”
{
PartUpdate(Gu8Part); //切換之前的局部進行更新。
Gu8Part++; //切換局部
if(Gu8Part>4)
{
Gu8Part=1;
}
PartUpdate(Gu8Part); //切換之后的局部進行更新。
vGu8BeepTimerFlag=0;
vGu16BeepTimerCnt=VOICE\_TIME; //蜂鳴器發出“滴”一聲
vGu8BeepTimerFlag=1;
}
break;
}
vGu8KeySec=0;
break;
case 2: //按鍵K2[加鍵]與[復位健]
if(0!=Gu8Part) //處于“閃爍模式”的時候,是[加鍵]
{
switch(Gu8Wd) //在某個窗口下
{
case 1: //在窗口1下
switch(Gu8Part) //二級支點的局部選擇
{
case 1: //局部1被選中,代表右起第4位數碼管被選中。
if(Gu8EditData\_4<9)
{
Gu8EditData\_4++; //編輯“千位”個體的中間變量
}
PartUpdate(Gu8Part); //當前局部更新顯示輸出到數碼管
break;
case 2: //局部2被選中,代表右起第3位數碼管被選中。
if(Gu8EditData\_3<9)
{
Gu8EditData\_3++; //編輯“百位”個體的中間變量
}
PartUpdate(Gu8Part); //當前局部更新顯示輸出到數碼管
break;
case 3: //局部3被選中,代表右起第2位數碼管被選中。
if(Gu8EditData\_2<9)
{
Gu8EditData\_2++; //編輯“十位”個體的中間變量
}
PartUpdate(Gu8Part); //當前局部更新顯示輸出到數碼管
break;
case 4: //局部4被選中,代表右起第1位數碼管被選中。
if(Gu8EditData\_1<9)
{
Gu8EditData\_1++; //編輯“個位”個體的中間變量
}
PartUpdate(Gu8Part); //當前局部更新顯示輸出到數碼管
break;
}
break;
}
}
else //處于“沒有閃爍”的時候,是[復位健]
{
Gu8EditData\_4=Gu32SetData\_Countdown/10000%10; //分解成“十秒”個體
Gu8EditData\_3=Gu32SetData\_Countdown/1000%10; //分解成“個秒”個體
Gu8EditData\_2=Gu32SetData\_Countdown/100%10; //分解成“百毫秒”個體
Gu8EditData\_1=Gu32SetData\_Countdown/10%10; //分解成“十毫秒”個體
Gu8RunStatus=0; //倒計時返回停止的狀態
Gu8RunStart=0; //倒計時的運行步驟的停止
Gu8RunStep=0; //總運行步驟歸零。建議跟vGu8RunStart成雙成對出現
Gu8Wd=1; //返回設置數據的窗口
Gu8WdUpdate=1; //整屏更新一次顯示
}
vGu8BeepTimerFlag=0;
vGu16BeepTimerCnt=VOICE\_TIME; //蜂鳴器發出“滴”一聲
vGu8BeepTimerFlag=1;
vGu8KeySec=0;
break;
case 3: //按鍵K3[減鍵]與[開始健]
if(0!=Gu8Part) //處于“閃爍模式”的時候,是[減鍵]
{
switch(Gu8Wd) //在某個窗口下
{
case 1: //在窗口1下
switch(Gu8Part) //二級支點的局部選擇
{
case 1: //局部1被選中,代表右起第4位數碼管被選中。
if(Gu8EditData\_4>0)
{
Gu8EditData\_4--; //編輯“十秒”個體的中間變量
}
PartUpdate(Gu8Part); //當前局部更新顯示輸出到數碼管
break;
case 2: //局部2被選中,代表右起第3位數碼管被選中。
if(Gu8EditData\_3>0)
{
Gu8EditData\_3--; //編輯“個秒”個體的中間變量
}
PartUpdate(Gu8Part); //當前局部更新顯示輸出到數碼管
break;
case 3: //局部3被選中,代表右起第2位數碼管被選中。
if(Gu8EditData\_2>0)
{
Gu8EditData\_2--; //編輯“百毫秒”個體的中間變量
}
PartUpdate(Gu8Part); //當前局部更新顯示輸出到數碼管
break;
case 4: //局部4被選中,代表右起第1位數碼管被選中。
if(Gu8EditData\_1>0)
{
Gu8EditData\_1--; //編輯“十毫位”個體的中間變量
}
PartUpdate(Gu8Part); //當前局部更新顯示輸出到數碼管
break;
}
break;
}
}
else //處于“沒有閃爍”的時候,是[開始健]
{
if(0==Gu8RunStatus) //在停止狀態下
{
vGu8CountdownTimerFlag=0;
vGu32CountdownTimerCnt=Gu32SetData\_Countdown; //從“設置值”開始倒計時
vGu8CountdownTimerFlag=1; //允許倒計時的軟件定時器的啟動
Gu8RunStatus=1; //倒計時處于工作狀態(并且,這一瞬間才正式啟動倒計時)
Gu8RunStart=1; //倒計時的運行步驟的總開關開啟
Gu8RunStep=0; //總運行步驟歸零。建議跟vGu8RunStart成雙成對出現
Gu8Wd=2; //進入倒計時運行的窗口
Gu8WdUpdate=1; //整屏更新一次顯示,確保在啟動的時候能顯示到最新的數據
}
}
vGu8BeepTimerFlag=0;
vGu16BeepTimerCnt=VOICE\_TIME; //蜂鳴器發出“滴”一聲
vGu8BeepTimerFlag=1;
vGu8KeySec=0;
break;
case 4: //K1按鍵的“長按”,具有進入和退出“閃爍模式”的功能。“退出”隱含“確定”
switch(Gu8Wd) //在某個窗口下
{
case 1: //在窗口1下
if(0==Gu8Part) //處于“沒有閃爍”的時候,將進入“閃爍模式”
{
Gu8EditData\_4=Gu32SetData\_Countdown/10000%10; //分解成“十秒”個體
Gu8EditData\_3=Gu32SetData\_Countdown/1000%10; //分解成“個秒”個體
Gu8EditData\_2=Gu32SetData\_Countdown/100%10; //分解成“百毫秒”個體
Gu8EditData\_1=Gu32SetData\_Countdown/10%10; //分解成“十毫秒”個體
Gu8Part=1; //進入“閃爍模式”,從“局部1”開始閃爍
}
else //處于“閃爍模式”的時候,將退出到“沒有閃爍”,隱含“確定”功能
{
//把個體合并還原成數據
Gu32SetData\_Countdown=Gu8EditData\_4\*10000+Gu8EditData\_3\*1000;
Gu32SetData\_Countdown=Gu32SetData\_Countdown+Gu8EditData\_2\*100;
Gu32SetData\_Countdown=Gu32SetData\_Countdown+Gu8EditData\_1\*10;
Gu8Part=0; //退出“閃爍模式”
Gu8WdUpdate=1; //整屏更新
}
break;
}
vGu8BeepTimerFlag=0;
vGu16BeepTimerCnt=VOICE\_TIME; //蜂鳴器發出“滴”一聲
vGu8BeepTimerFlag=1;
vGu8KeySec=0;
break;
}
}
void DisplayTask(void) //數碼管顯示的上層任務函數
{
switch(Gu8Wd) //以窗口選擇Gu8Wd為支點,去執行對應的窗口顯示函數。又一次用到switch語句
{
case 1:
Wd1(); //窗口1顯示函數。用來設置參數。
break;
case 2:
Wd2(); //窗口2顯示函數。倒計時的運行顯示窗口。
break;
}
}
void Wd1(void) //窗口1顯示函數。用來設置參數。
{
//需要借用的中間變量,用來拆分數據位。
static unsigned char Su8Temp\_4,Su8Temp\_3,Su8Temp\_2,Su8Temp\_1; //需要借用的中間變量
static unsigned char Su8BlinkFlag=0; //兩種狀態的切換判斷的中間變量
if(1==Gu8WdUpdate) //如果需要整屏更新
{
Gu8WdUpdate=0; //及時清零,只更新一次顯示即可,避免一直進來更新顯示
//屬于靜態數據,起“裝飾”作用,切換窗口后只掃描一次的代碼。
vGu8Display\_Righ\_Dot\_4=0;
vGu8Display\_Righ\_Dot\_3=1; //保留顯示2位小數點
vGu8Display\_Righ\_Dot\_2=0;
vGu8Display\_Righ\_Dot\_1=0;
Gu8PartUpdate\_1=1; //局部1更新顯示
Gu8PartUpdate\_2=1 ;//局部2更新顯示
Gu8PartUpdate\_3=1 ;//局部3更新顯示
Gu8PartUpdate\_4=1 ;//局部4更新顯示
}
if(1==Gu8PartUpdate\_1) //局部1更新顯示
{
Gu8PartUpdate\_1=0; //及時清零,只更新一次顯示即可,避免一直進來更新顯示
if(Gu32SetData\_Countdown<10000)
{
Su8Temp\_4=10; //顯示“無”
}
else
{
Su8Temp\_4=Gu8EditData\_4; //顯示“十秒”的臨時中間個體,屬于動態數據。
}
vGu8Display\_Righ\_4=Su8Temp\_4; //過渡需要顯示的數據到底層驅動變量
}
if(1==Gu8PartUpdate\_2) //局部2更新顯示
{
Gu8PartUpdate\_2=0; //及時清零,只更新一次顯示即可,避免一直進來更新顯示
Su8Temp\_3=Gu8EditData\_3; //顯示“個秒”的臨時中間個體,屬于動態數據。
vGu8Display\_Righ\_3=Su8Temp\_3; //過渡需要顯示的數據到底層驅動變量
}
if(1==Gu8PartUpdate\_3) //局部3更新顯示
{
Gu8PartUpdate\_3=0; //及時清零,只更新一次顯示即可,避免一直進來更新顯示
Su8Temp\_2=Gu8EditData\_2; //顯示“百毫秒”的臨時中間個體,屬于動態數據。
vGu8Display\_Righ\_2=Su8Temp\_2; //過渡需要顯示的數據到底層驅動變量
}
if(1==Gu8PartUpdate\_4) //局部4更新顯示
{
Gu8PartUpdate\_4=0; //及時清零,只更新一次顯示即可,避免一直進來更新顯示
Su8Temp\_1=Gu8EditData\_1; //顯示“十毫秒”的臨時中間個體,屬于動態數據。
vGu8Display\_Righ\_1=Su8Temp\_1; //過渡需要顯示的數據到底層驅動變量
}
if(0==vGu16BlinkTimerCnt) //某位被選中的數碼管跳動閃爍的定時器
{
vGu8BlinkTimerFlag=0;
vGu16BlinkTimerCnt=BLINK\_TIME; //重設定時器的定時時間
vGu8BlinkTimerFlag=1;
switch(Gu8Part) //某個局部被選中,則閃爍跳動
{
case 1:
if(0==Su8BlinkFlag) //兩種狀態的切換判斷
{
Su8BlinkFlag=1;
Su8Temp\_4=10; //右起第4個顯示“不顯示”(10代表不顯示)
}
else
{
Su8BlinkFlag=0;
Su8Temp\_4=Gu8EditData\_4; //顯示“十秒”的臨時中間個體,屬于動態數據。
}
break;
case 2:
if(0==Su8BlinkFlag) //兩種狀態的切換判斷
{
Su8BlinkFlag=1;
Su8Temp\_3=10; //右起第3個顯示“不顯示”(10代表不顯示)
}
else
{
Su8BlinkFlag=0;
Su8Temp\_3=Gu8EditData\_3; //顯示“個秒”的臨時中間個體,屬于動態數據。
}
break;
case 3:
if(0==Su8BlinkFlag) //兩種狀態的切換判斷
{
Su8BlinkFlag=1;
Su8Temp\_2=10; //右起第2個顯示“不顯示”(10代表不顯示)
}
else
{
Su8BlinkFlag=0;
Su8Temp\_2=Gu8EditData\_2; //顯示“百毫秒”的臨時中間個體,屬于動態數據。
}
break;
case 4:
if(0==Su8BlinkFlag) //兩種狀態的切換判斷
{
Su8BlinkFlag=1;
Su8Temp\_1=10; //右起第1個顯示“不顯示”(10代表不顯示)
}
else
{
Su8BlinkFlag=0;
Su8Temp\_1=Gu8EditData\_1; //顯示“十毫秒”的臨時中間個體,屬于動態數據。
}
break;
default: //都沒有被選中的時候
if(Gu32SetData\_Countdown<10000)
{
Su8Temp\_4=10; //顯示“無”
}
else
{
Su8Temp\_4=Gu8EditData\_4; //顯示“十秒”的臨時中間個體,屬于動態數據。
}
Su8Temp\_3=Gu8EditData\_3; //顯示“個秒”的臨時中間個體,屬于動態數據。
Su8Temp\_2=Gu8EditData\_2; //顯示“百毫秒”的臨時中間個體,屬于動態數據。
Su8Temp\_1=Gu8EditData\_1; //顯示“十毫秒”的臨時中間個體,屬于動態數據。
break;
}
vGu8Display\_Righ\_4=Su8Temp\_4; //過渡需要顯示的數據到底層驅動變量
vGu8Display\_Righ\_3=Su8Temp\_3; //過渡需要顯示的數據到底層驅動變量
vGu8Display\_Righ\_2=Su8Temp\_2; //過渡需要顯示的數據到底層驅動變量
vGu8Display\_Righ\_1=Su8Temp\_1; //過渡需要顯示的數據到底層驅動變量
}
}
void Wd2(void) //窗口2顯示函數。倒計時的運行顯示窗口。
{
//需要借用的中間變量,用來拆分數據位。
static unsigned char Su8Temp\_4,Su8Temp\_3,Su8Temp\_2,Su8Temp\_1; //需要借用的中間變量
if(1==Gu8WdUpdate) //如果需要整屏更新
{
Gu8WdUpdate=0; //及時清零,只更新一次顯示即可,避免一直進來更新顯示
//先分解數據,注意,這里分解的時候,“先整除后求余”必須用一行代碼一氣呵成,不能拆
//分成兩行代碼,否則會有隱患會有bug。除非,把四個臨時變都改成unsigned long類型。
//Su8Temp\_4提取“十秒”位。
Su8Temp\_4=vGu32CountdownTimerCnt/10000%10; //實際精度是0.001秒,但顯示精度是0.01秒
//Su8Temp\_3提取“個秒”位。
Su8Temp\_3=vGu32CountdownTimerCnt/1000%10; //實際精度是0.001秒,但顯示精度是0.01秒
//Su8Temp\_2提取“百毫秒”位。
Su8Temp\_2=vGu32CountdownTimerCnt/100%10; //實際精度是0.001秒,但顯示精度是0.01秒
//Su8Temp\_1提取“十毫秒”位。
Su8Temp\_1=vGu32CountdownTimerCnt/10%10; //實際精度是0.001秒,但顯示精度是0.01秒
//判斷數據范圍,來決定最高位數碼管是否需要顯示。
if(vGu32CountdownTimerCnt<10000) //10.000秒。實際4位數碼管最大只能顯示99.99秒
{
Su8Temp\_4=10; //在數碼管轉換表里,10代表一個“不顯示”的數據
}
//上面先分解數據之后,再過渡需要顯示的數據到底層驅動變量里,讓過渡的時間越短越好
vGu8Display\_Righ\_4=Su8Temp\_4; //過渡需要顯示的數據到底層驅動變量
vGu8Display\_Righ\_3=Su8Temp\_3;
vGu8Display\_Righ\_2=Su8Temp\_2;
vGu8Display\_Righ\_1=Su8Temp\_1;
vGu8Display\_Righ\_Dot\_4=0;
vGu8Display\_Righ\_Dot\_3=1; //保留顯示2位小數點
vGu8Display\_Righ\_Dot\_2=0;
vGu8Display\_Righ\_Dot\_1=0;
}
}
void KeyScan(void) //按鍵底層的驅動掃描函數,放在定時中斷函數里
{
static unsigned char Su8KeyShortFlag=0; //按鍵“短按”觸發的標志
static unsigned char Su8KeyLock1;
static unsigned int Su16KeyCnt1;
static unsigned char Su8KeyLock2;
static unsigned int Su16KeyCnt2;
static unsigned char Su8KeyLock3;
static unsigned int Su16KeyCnt3;
//需要詳細分析以下這段“短按”與“長按”代碼的朋友,請參考第96節。
if(0!=KEY\_INPUT1)
{
Su8KeyLock1=0;
Su16KeyCnt1=0;
if(1==Su8KeyShortFlag)
{
Su8KeyShortFlag=0;
vGu8KeySec=1; //觸發K1的“短按”
}
}
else if(0==Su8KeyLock1)
{
Su16KeyCnt1++;
if(Su16KeyCnt1>=KEY\_FILTER\_TIME)
{
Su8KeyShortFlag=1;
}
if(Su16KeyCnt1>=KEY\_LONG\_TIME)
{
Su8KeyLock1=1;
Su8KeyShortFlag=0;
vGu8KeySec=4; //觸發K1的“長按”
}
}
if(0!=KEY\_INPUT2)
{
Su8KeyLock2=0;
Su16KeyCnt2=0;
}
else if(0==Su8KeyLock2)
{
Su16KeyCnt2++;
if(Su16KeyCnt2>=KEY\_FILTER\_TIME)
{
Su8KeyLock2=1;
vGu8KeySec=2;
}
}
if(0!=KEY\_INPUT3)
{
Su8KeyLock3=0;
Su16KeyCnt3=0;
}
else if(0==Su8KeyLock3)
{
Su16KeyCnt3++;
if(Su16KeyCnt3>=KEY\_FILTER\_TIME)
{
Su8KeyLock3=1;
vGu8KeySec=3;
}
}
}
void DisplayScan(void) //數碼管底層的驅動掃描函數,放在定時中斷函數里
{
static unsigned char Su8GetCode;
static unsigned char Su8ScanStep=1;
if(0==vGu16ScanTimerCnt)
{
P0=0x00;
P1\_0=1;
P1\_1=1;
P1\_2=1;
P1\_3=1;
switch(Su8ScanStep)
{
case 1:
Su8GetCode=Cu8DigTable\[vGu8Display\_Righ\_1\];
if(1==vGu8Display\_Righ\_Dot\_1)
{
Su8GetCode=Su8GetCode|0x80;
}
P0=Su8GetCode;
P1\_0=0;
P1\_1=1;
P1\_2=1;
P1\_3=1;
break;
case 2:
Su8GetCode=Cu8DigTable\[vGu8Display\_Righ\_2\];
if(1==vGu8Display\_Righ\_Dot\_2)
{
Su8GetCode=Su8GetCode|0x80;
}
P0=Su8GetCode;
P1\_0=1;
P1\_1=0;
P1\_2=1;
P1\_3=1;
break;
case 3:
Su8GetCode=Cu8DigTable\[vGu8Display\_Righ\_3\];
if(1==vGu8Display\_Righ\_Dot\_3)
{
Su8GetCode=Su8GetCode|0x80;
}
P0=Su8GetCode;
P1\_0=1;
P1\_1=1;
P1\_2=0;
P1\_3=1;
break;
case 4:
Su8GetCode=Cu8DigTable\[vGu8Display\_Righ\_4\];
if(1==vGu8Display\_Righ\_Dot\_4)
{
Su8GetCode=Su8GetCode|0x80;
}
P0=Su8GetCode;
P1\_0=1;
P1\_1=1;
P1\_2=1;
P1\_3=0;
break;
}
Su8ScanStep++;
if(Su8ScanStep>4)
{
Su8ScanStep=1;
}
vGu8ScanTimerFlag=0;
vGu16ScanTimerCnt=SCAN\_TIME;
vGu8ScanTimerFlag=1;
}
}
void VoiceScan(void) //蜂鳴器的驅動函數
{
static unsigned char Su8Lock=0;
if(1==vGu8BeepTimerFlag&&vGu16BeepTimerCnt>0)
{
if(0==Su8Lock)
{
Su8Lock=1;
BeepOpen();
}
else
{
vGu16BeepTimerCnt--;
if(0==vGu16BeepTimerCnt)
{
Su8Lock=0;
BeepClose();
}
}
}
}
void BeepOpen(void)
{
P3\_4=0;
}
void BeepClose(void)
{
P3\_4=1;
}
void T0\_time() interrupt 1
{
VoiceScan(); //蜂鳴器的驅動函數
KeyScan(); //按鍵底層的驅動掃描函數
DisplayScan(); //數碼管底層的驅動掃描函數
if(1==vGu8ScanTimerFlag&&vGu16ScanTimerCnt>0)
{
vGu16ScanTimerCnt--; //遞減式的軟件定時器
}
if(1==vGu8BlinkTimerFlag&&vGu16BlinkTimerCnt>0) //數碼管閃爍跳動的定時器
{
vGu16BlinkTimerCnt--; //遞減式的軟件定時器
}
//每10ms就定時更新一次顯示的軟件定時器
if(1==vGu8UpdateTimerFlag&&vGu16UpdateTimerCnt>0)
{
vGu16UpdateTimerCnt--; //遞減式的軟件定時器
}
//倒計時實際走的時間的軟件定時器,注意,這里還附加了啟動狀態的條件“&&1==Gu8RunStatus”
if(1==vGu8CountdownTimerFlag&&vGu32CountdownTimerCnt>0&&1==Gu8RunStatus)
{
vGu32CountdownTimerCnt--; //遞減式的軟件定時器
}
TH0=0xfd; //此參數可根據具體的時間來修改,盡量確保每定時中斷一次接近1ms
TL0=0x40; //此參數可根據具體的時間來修改,盡量確保每定時中斷一次接近1ms
}
void SystemInitial(void)
{
P0=0x00;
P1\_0=1;
P1\_1=1;
P1\_2=1;
P1\_3=1;
TMOD=0x01;
TH0=0xfd; //此參數可根據具體的時間來修改,盡量確保每定時中斷一次接近1ms
TL0=0x40; //此參數可根據具體的時間來修改,盡量確保每定時中斷一次接近1ms
EA=1;
ET0=1;
TR0=1;
//上電初始化開機顯示的窗口
Gu8EditData\_4=Gu32SetData\_Countdown/10000%10; //分解成“十秒”個體
Gu8EditData\_3=Gu32SetData\_Countdown/1000%10; //分解成“個秒”個體
Gu8EditData\_2=Gu32SetData\_Countdown/100%10; //分解成“百毫秒”個體
Gu8EditData\_1=Gu32SetData\_Countdown/10%10; //分解成“十毫秒”個體
Gu8Wd=1; //返回設置數據的窗口
Gu8WdUpdate=1; //整屏更新一次顯示
}
void Delay(unsigned long u32DelayTime)
{
for(;u32DelayTime>0;u32DelayTime--);
}
void PeripheralInitial(void)
{
}
- 首頁
- 第一節:我的價值觀
- 第二節:初學者的疑惑
- 第三節:單片機最重要的一個特性
- 第四節:平臺軟件和編譯器軟件的簡介
- 第五節:用Keil2軟件關閉,新建,打開一個工程的操作流程
- 第六節:把.c源代碼編譯成.hex機器碼的操作流程
- 第七節:本節預留
- 第八節:把.hex機器碼程序燒錄到單片機的操作流程
- 第九節:本節預留
- 第十節:程序從哪里開始,要到哪里去?
- 第十一節:一個在單片機上練習C語言的模板程序
- 第十二節:變量的定義和賦值
- 【TODO】第十三節:賦值語句的覆蓋性
- 【TODO】第十四節:二進制與字節單位,以及常用三種變量的取值范圍
- 【TODO】第十五節:二進制與十六進制
- 【TODO】第十六節:十進制與十六進制
- 【TODO】第十七節:加法運算的5種常用組合
- 【TODO】第十八節:連加、自加、自加簡寫、自加1
- 【TODO】第十九節:加法運算的溢出
- 【TODO】第二十節:隱藏中間變量為何物?
- 【TODO】第二十一節:減法運算的5種常用組合。
- 【TODO】第二十二節:連減、自減、自減簡寫、自減1
- 【TODO】第二十三節:減法溢出與假想借位
- 【TODO】第二十四節:借用unsigned long類型的中間變量可以減少溢出現象
- 【TODO】第二十五節:乘法運算中的5種常用組合
- 【TODO】第二十六節:連乘、自乘、自乘簡寫,溢出
- 【TODO】第二十七節:整除求商
- 【TODO】第二十八節:整除求余
- 【TODO】第二十九節:“先余后商”和“先商后余”提取數據某位,哪家強?
- 【TODO】第三十節:邏輯運算符的“與”運算
- 【TODO】第三十一節:邏輯運算符的“或”運算
- 【TODO】第三十二節:邏輯運算符的“異或”運算
- 【TODO】第三十三節:邏輯運算符的“按位取反”和“非”運算
- 【TODO】第三十四節:移位運算的左移
- 【TODO】第三十五節:移位運算的右移
- 【TODO】第三十六節:括號的強制功能---改變運算優先級
- 【TODO】第三十七節:單字節變量賦值給多字節變量的疑惑
- 【TODO】第三十八節:第二種解決“運算過程中意外溢出”的便捷方法
- 【TODO】第三十九節:if判斷語句以及常量變量的真假判斷
- 【TODO】第四十節:關系符的等于“==”和不等于“!=”
- 【TODO】第四十一節:關系符的大于“>”和大于等于“>=”
- 【TODO】第四十二節:關系符的小于“<”和小于等于“<=”
- 【TODO】第四十三節:關系符中的關系符:與“&&”,或“||”
- 【TODO】第四十四節:小括號改變判斷優先級
- 【TODO】第四十五節: 組合判斷if...else if...else
- 【TODO】第四十六節: 一維數組
- 【TODO】第四十七節: 二維數組
- 【TODO】第四十八節: while循環語句
- 【TODO】第四十九節: 循環語句do while和for
- 【TODO】第五十節: 循環體內的continue和break語句
- 【TODO】第五十一節: for和while的循環嵌套
- 【TODO】第五十二節: 支撐程序框架的switch語句
- 【TODO】第五十三節: 使用函數的三要素和執行順序
- 【TODO】第五十四節: 從全局變量和局部變量中感悟“棧”為何物
- 【TODO】第五十五節: 函數的作用和四種常見書寫類型
- 【TODO】第五十六節: return在函數中的作用以及四個容易被忽略的功能
- 【TODO】第五十七節: static的重要作用
- 【TODO】第五十八節: const(./book/或code)在定義數據時的作用
- 【TODO】第五十九節: 全局“一鍵替換”功能的#define
- 【TODO】第六十節: 指針在變量(./book/或常量)中的基礎知識
- 【TODO】第六十一節: 指針的中轉站作用,地址自加法,地址偏移法
- 【TODO】第六十二節: 指針,大小端,化整為零,化零為整
- 【TODO】第六十三節: 指針“化整為零”和“化零為整”的“靈活”應用
- 【TODO】第六十四節: 指針讓函數具備了多個相當于return的輸出口
- 【TODO】第六十五節: 指針作為數組在函數中的入口作用
- 【TODO】第六十六節: 指針作為數組在函數中的出口作用
- 【TODO】第六十七節: 指針作為數組在函數中既“入口”又“出口”的作用
- 【TODO】第六十八節: 為函數接口指針“定向”的const關鍵詞
- 【TODO】第六十九節: 宏函數sizeof(./book/)
- 【TODO】第七十節: “萬能數組”的結構體
- 【TODO】第七十一節: 結構體的內存和賦值
- 【TODO】第七十二節: 結構體的指針
- 【TODO】第七十三節: 結構體數據的傳輸存儲和還原
- 【TODO】第七十四節: 結構體指針在函數接口處的頻繁應用
- 【TODO】第七十五節: 指針的名義(例:一維指針操作二維數組)
- 【TODO】第七十六節: 二維數組的指針
- 【TODO】第七十七節: 指針唯一的“單向輸出”通道return
- 【TODO】第七十八節: typedef和#define和enum
- 【TODO】第七十九節: 各種變量常量的命名規范
- 【TODO】第八十節: 單片機IO口驅動LED
- 【TODO】第八十一節: 時間和速度的起源(指令周期和晶振頻率)
- 【TODO】第八十二節: Delay“阻塞”延時控制LED閃爍
- 【TODO】第八十三節: 累計主循環的“非阻塞”延時控制LED閃爍
- 【TODO】第八十四節: 中斷與中斷函數
- 【TODO】第八十五節: 定時中斷的寄存器配置
- 【TODO】第八十六節: 定時中斷的“非阻塞”延時控制LED閃爍
- 【TODO】第八十七節: 一個定時中斷產生N個軟件定時器
- 【TODO】第八十八節: 兩大核心框架理論(四區一線,switch外加定時中斷)
- 【TODO】第八十九節: 跑馬燈的三種境界
- 【TODO】第九十節: 多任務并行處理兩路跑馬燈
- 【TODO】第九十一節: 蜂鳴器的“非阻塞”驅動
- 【TODO】第九十二節: 獨立按鍵的四大要素(自鎖,消抖,非阻塞,清零式濾波)
- 【TODO】第九十三節: 獨立按鍵鼠標式的單擊與雙擊
- 【TODO】第九十四節: 兩個獨立按鍵構成的組合按鍵
- 【TODO】第九十五節: 兩個獨立按鍵的“電腦鍵盤式”組合按鍵
- 【TODO】第九十六節: 獨立按鍵“一鍵兩用”的短按與長按
- 【TODO】第九十七節: 獨立按鍵按住不松手的連續均勻觸發
- 【TODO】第九十八節: 獨立按鍵按住不松手的“先加速后勻速”的觸發
- 【TODO】第九十九節: “行列掃描式”矩陣按鍵的單個觸發(原始版)
- 【TODO】第一百節: “行列掃描式”矩陣按鍵的單個觸發(優化版)
- 【TODO】第一百零一節: 矩陣按鍵鼠標式的單擊與雙擊
- 【TODO】第一百零二節: 兩個“任意行輸入”矩陣按鍵的“有序”組合觸發
- 【TODO】第一百零三節: 兩個“任意行輸入”矩陣按鍵的“無序”組合觸發
- 【TODO】第一百零四節: 矩陣按鍵“一鍵兩用”的短按與長按
- 【TODO】第一百零五節: 矩陣按鍵按住不松手的連續均勻觸發
- 【TODO】第一百零六節: 矩陣按鍵按住不松手的“先加速后勻速”觸發
- 【TODO】第一百零七節: 開關感應器的識別與軟件濾波
- 【TODO】第一百零八節: 按鍵控制跑馬燈的啟動和暫停和停止
- 【TODO】第一百零九節: 按鍵控制跑馬燈的方向
- 【TODO】第一百一十節: 按鍵控制跑馬燈的速度
- 第一百一十一節: 工業自動化設備的開關信號的運動控制
- 【TODO】第一百一十二節: 數碼管顯示的基礎知識
- 【TODO】第一百一十三節: 動態掃描的數碼管顯示數字
- 【TODO】第一百一十四節: 動態掃描的數碼管顯示小數點
- 【TODO】第一百一十五節: 按鍵控制數碼管的秒表
- 【TODO】第一百一十六節: 按鍵控制數碼管的倒計時
- 【TODO】第一百一十七節: 按鍵切換數碼管窗口來設置參數
- 【TODO】第一百一十八節: 按鍵讓某位數碼管閃爍跳動來設置參數
- 【TODO】第一百一十九節: 一個完整的人機界面的程序框架的脈絡
- 【TODO】第一百二十節: 按鍵切換窗口切換局部來設置參數
- 【TODO】第一百二十一節: 可調參數的數碼管倒計時
- 【TODO】第一百二十二節: 利用定時中斷做的“時分秒”數顯時鐘
- 【TODO】第一百二十三節: 一種能省去一個lock自鎖變量的按鍵驅動程序
- 【TODO】第一百二十四節: 數顯儀表盤顯示“速度、方向、計數器”的跑馬燈
- 【TODO】第一百二十五節: “雙線”的肢體接觸通信
- 【TODO】第一百二十六節: “單線”的肢體接觸通信
- 【TODO】第一百二十七節: 單片機串口接收數據的機制
- 【TODO】第一百二十八節: 接收“固定協議”的串口程序框架
- 【TODO】第一百二十九節: 接收帶“動態密匙”與“累加和”校驗數據的串口程序框架
- 【TODO】第一百三十節: 接收帶“動態密匙”與“異或”校驗數據的串口程序框架
- 【TODO】第一百三十一節: 靈活切換各種不同大小“接收內存”的串口程序框架
- 【TODO】第一百三十二節:“轉發、透傳、多種協議并存”的雙緩存串口程序框架
- 【TODO】第一百三十三節:常用的三種串口發送函數
- 【TODO】第一百三十四節:“應用層半雙工”雙機串口通訊的程序框架