一、使用proteus繪製簡單的電路圖,用於後續仿真
二、編寫程序
/********************************************************************************************************************
---- @Project: LED-74HC595
---- @File: main.c
---- @Edit: ZHQ
---- @Version: V1.0
---- @CreationTime: 20200528
---- @ModifiedTime: 20200529
---- @Description: 第1個至第8個LED燈一直不亮。在第9個至第16個LED燈,依次逐個亮燈並且每次只能亮一個燈。按一次獨立按鍵S1,將會更改跑馬燈的運動方向。
---- 單片機:AT89C52
********************************************************************************************************************/
#include "reg52.h"
/*——————宏定義——————*/
#define FOSC 11059200L
#define T1MS (65536-FOSC/12/1000) /*1ms timer calculation method in 12Tmode*/
#define const_time_level_09_16 600
#define const_voice_short 80 /*蜂鳴器短叫的持續時間*/
#define const_key_time1 40
/*——————變量函數定義及聲明——————*/
/*定義74HC595*/
sbit Hc595_Sh = P2^3;
sbit Hc595_St = P2^4;
sbit Hc595_Ds = P2^5;
/*定義蜂鳴器*/
sbit Beep = P2^7;
/*定義按鍵S1*/
sbit Key_S1 = P0^0;
/*定義按鍵模擬地*/
sbit Key_Gnd = P0^4;
unsigned char ucLED1 = 0; /*代表16個燈的亮滅狀態,0代表滅,1代表亮*/
unsigned char ucLED2 = 0;
unsigned char ucLED3 = 0;
unsigned char ucLED4 = 0;
unsigned char ucLED5 = 0;
unsigned char ucLED6 = 0;
unsigned char ucLED7 = 0;
unsigned char ucLED8 = 0;
unsigned char ucLED9 = 0;
unsigned char ucLED10 = 0;
unsigned char ucLED11 = 0;
unsigned char ucLED12 = 0;
unsigned char ucLED13 = 0;
unsigned char ucLED14 = 0;
unsigned char ucLED15 = 0;
unsigned char ucLED16 = 0;
unsigned char ucLed_update = 0; /*刷新變量。每次更改LED燈的狀態都要更新一次。*/
unsigned char ucLedStep_01_08 = 0; /*第1個至第8個LED跑馬燈的步驟變量*/
unsigned int uiTimeCnt_01_08 = 0; /*第1個至第8個LED跑馬燈的統計定時中斷次數的延時計數器*/
unsigned char ucLedStep_09_16 = 0; /*第9個至第16個LED跑馬燈的步驟變量*/
unsigned int uiTimeCnt_09_16 = 0; /*第9個至第16個LED跑馬燈的統計定時中斷次數的延時計數器*/
unsigned char ucLedStatus16_09 = 0; /*代表底層74HC595輸出狀態的中間變量*/
unsigned char ucLedStatus08_01 = 0; /*代表底層74HC595輸出狀態的中間變量*/
unsigned char ucKeySec = 0; /*被觸發的按鍵編號*/
unsigned int uiKeyTimeCnt1 = 0; /*按鍵去抖動延時計數器*/
unsigned char ucKeyLock1 = 0; /*按鍵觸發後自鎖的變量標誌*/
unsigned int uiVoiceCnt = 0; /*蜂鳴器鳴叫的持續時間計數器*/
unsigned char ucLedDirFlag = 0; /*方向變量,把按鍵與跑馬燈關聯起來的核心變量,0代表正方向,1代表反方向*/
/**
* @brief 定時器0初始化函數
* @param 無
* @retval 初始化T0
**/
void Init_T0(void)
{
TMOD = 0x01; /*set timer0 as mode1 (16-bit)*/
TL0 = T1MS; /*initial timer0 low byte*/
TH0 = T1MS >> 8; /*initial timer0 high byte*/
}
/**
* @brief 外圍初始化函數
* @param 無
* @retval 初始化外圍
**/
void Init_Peripheral(void)
{
ET0 = 1;/*允許定時中斷*/
TR0 = 1;/*啓動定時中斷*/
EA = 1;/*開總中斷*/
}
/**
* @brief 初始化函數
* @param 無
* @retval 初始化單片機
**/
void Init(void)
{
Key_Gnd = 0;
Beep = 1;
Init_T0();
}
/**
* @brief 延時函數
* @param 無
* @retval 無
**/
void Delay_Long(unsigned int uiDelayLong)
{
unsigned int i;
unsigned int j;
for(i=0;i<uiDelayLong;i++)
{
for(j=0;j<500;j++) /*內嵌循環的空指令數量*/
{
; /*一個分號相當於執行一條空語句*/
}
}
}
/**
* @brief 延時函數
* @param 無
* @retval 無
**/
void Delay_Short(unsigned int uiDelayShort)
{
unsigned int i;
for(i=0;i<uiDelayShort;i++)
{
; /*一個分號相當於執行一條空語句*/
}
}
/**
* @brief 掃描按鍵
* @param 無
* @retval 放在定時中斷裏
**/
void Key_Scan(void)
{
if(Key_S1 == 1) /*IO是高電平,說明按鍵沒有被按下,這時要及時清零一些標誌位*/
{
ucKeyLock1 = 0;
uiKeyTimeCnt1 = 0;
}
else if(ucKeyLock1 == 0) /*有按鍵按下,且是第一次被按下*/
{
uiKeyTimeCnt1 ++; /*累加定時中斷次數*/
if(uiKeyTimeCnt1 > const_key_time1)
{
uiKeyTimeCnt1 = 0;
ucKeyLock1 = 1; /*自鎖按鍵置位,避免一直觸發*/
ucKeySec = 1;
}
}
}
/**
* @brief 按鍵服務的應用程序
* @param 無
* @retval 無
**/
void Key_Service(void)
{
switch(ucKeySec) /*按鍵服務狀態切換*/
{
case 1: /*改變跑馬燈方向的按鍵,對應S1*/
if(ucLedDirFlag == 0) /*通過中間變量改變跑馬燈的方向*/
{
ucLedDirFlag = 1;
}
else
{
ucLedDirFlag = 0;
}
uiVoiceCnt = const_voice_short; /*按鍵聲音觸發,滴一聲就停。*/
ucKeySec = 0; /*響應按鍵服務處理程序後,按鍵編號清零,避免一致觸發*/
break;
}
}
/**
* @brief 595驅動函數
* @param 無
* @retval * 兩個聯級74HC595的工作過程:
* 每個74HC595內部都有一個8位的寄存器,兩個聯級起來就有兩個寄存器。ST引腳就相當於一個刷新
* 信號引腳,當ST引腳產生一個上升沿信號時,就會把寄存器的數值輸出到74HC595的輸出引腳並且鎖存起來,
* DS是數據引腳,SH是把新數據送入寄存器的時鐘信號。也就是說,SH引腳負責把數據送入到寄存器裏,ST引腳
* 負責把寄存器的數據更新輸出到74HC595的輸出引腳上並且鎖存起來。
**/
void HC595_Drive(unsigned char ucLedStatusTemp16_09, unsigned char ucLedStatusTemp08_01)
{
unsigned char i;
unsigned char ucTempData;
Hc595_Sh = 0;
Hc595_St = 0;
ucTempData = ucLedStatusTemp16_09; /*先送高8位*/
for(i = 0; i < 8; i ++)
{
if(ucTempData >= 0x80)
{
Hc595_Ds = 1;
}
else
{
Hc595_Ds = 0;
}
Hc595_Sh = 0; /*SH引腳的上升沿把數據送入寄存器*/
Delay_Short(15);
Hc595_Sh = 1;
Delay_Short(15);
ucTempData = ucTempData <<1;
}
ucTempData = ucLedStatusTemp08_01; /*再先送低8位*/
for(i = 0; i < 8; i ++)
{
if(ucTempData >= 0x80)
{
Hc595_Ds = 1;
}
else
{
Hc595_Ds = 0;
}
Hc595_Sh = 0; /*SH引腳的上升沿把數據送入寄存器*/
Delay_Short(15);
Hc595_Sh = 1;
Delay_Short(15);
ucTempData = ucTempData <<1;
}
Hc595_St = 0; /*ST引腳把兩個寄存器的數據更新輸出到74HC595的輸出引腳上並且鎖存起來*/
Delay_Short(15);
Hc595_St = 1;
Delay_Short(15);
Hc595_Sh = 0; /*拉低,抗干擾就增強*/
Hc595_St = 0;
Hc595_Ds = 0;
}
/**
* @brief LED更新函數
* @param 無
* @retval
* 把74HC595驅動程序翻譯成類似單片機IO口直接驅動方式的過程。
* 每次更新LED輸出,記得都要把ucLed_update置1表示更新。
**/
void LED_Update()
{
if(ucLed_update == 1)
{
ucLed_update = 0; /*及時清零,讓它產生只更新一次的效果,避免一直更新。*/
if(ucLED1 == 1)
{
ucLedStatus08_01 = ucLedStatus08_01 | 0x01;
}
else
{
ucLedStatus08_01 = ucLedStatus08_01 & 0xfe;
}
if(ucLED2 == 1)
{
ucLedStatus08_01 = ucLedStatus08_01 | 0x02;
}
else
{
ucLedStatus08_01 = ucLedStatus08_01 & 0xfd;
}
if(ucLED3 == 1)
{
ucLedStatus08_01 = ucLedStatus08_01 | 0x04;
}
else
{
ucLedStatus08_01 = ucLedStatus08_01 & 0xfb;
}
if(ucLED4 == 1)
{
ucLedStatus08_01 = ucLedStatus08_01 | 0x08;
}
else
{
ucLedStatus08_01 = ucLedStatus08_01 & 0xf7;
}
if(ucLED5 == 1)
{
ucLedStatus08_01 = ucLedStatus08_01 | 0x10;
}
else
{
ucLedStatus08_01 = ucLedStatus08_01 & 0xef;
}
if(ucLED6 == 1)
{
ucLedStatus08_01 = ucLedStatus08_01 | 0x20;
}
else
{
ucLedStatus08_01 = ucLedStatus08_01 & 0xdf;
}
if(ucLED7 == 1)
{
ucLedStatus08_01 = ucLedStatus08_01 | 0x40;
}
else
{
ucLedStatus08_01 = ucLedStatus08_01 & 0xbf;
}
if(ucLED8 == 1)
{
ucLedStatus08_01 = ucLedStatus08_01 | 0x80;
}
else
{
ucLedStatus08_01 = ucLedStatus08_01 & 0x7f;
}
if(ucLED9 == 1)
{
ucLedStatus16_09 = ucLedStatus16_09 | 0x01;
}
else
{
ucLedStatus16_09 = ucLedStatus16_09 & 0xfe;
}
if(ucLED10 == 1)
{
ucLedStatus16_09 = ucLedStatus16_09 | 0x02;
}
else
{
ucLedStatus16_09 = ucLedStatus16_09 & 0xfd;
}
if(ucLED11 == 1)
{
ucLedStatus16_09 = ucLedStatus16_09 | 0x04;
}
else
{
ucLedStatus16_09 = ucLedStatus16_09 & 0xfb;
}
if(ucLED12 == 1)
{
ucLedStatus16_09 = ucLedStatus16_09 | 0x08;
}
else
{
ucLedStatus16_09 = ucLedStatus16_09 & 0xf7;
}
if(ucLED13 == 1)
{
ucLedStatus16_09 = ucLedStatus16_09 | 0x10;
}
else
{
ucLedStatus16_09 = ucLedStatus16_09 & 0xef;
}
if(ucLED14 == 1)
{
ucLedStatus16_09 = ucLedStatus16_09 | 0x20;
}
else
{
ucLedStatus16_09 = ucLedStatus16_09 & 0xdf;
}
if(ucLED15 == 1)
{
ucLedStatus16_09 = ucLedStatus16_09 | 0x40;
}
else
{
ucLedStatus16_09 = ucLedStatus16_09 & 0xbf;
}
if(ucLED16 == 1)
{
ucLedStatus16_09 = ucLedStatus16_09 | 0x80;
}
else
{
ucLedStatus16_09 = ucLedStatus16_09 & 0x7f;
}
HC595_Drive(ucLedStatus16_09, ucLedStatus08_01);
}
}
///**
//* @brief LED服務函數
//* @param 無
//* @retval 無
//**/
//void LED_Flicker_01_08(void)
//{
// switch(ucLedStep_01_08)
// {
// case 0:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED1 = 1;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 1;
// }
// break;
// case 1:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED2 = 1;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 2;
// }
// break;
// case 2:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED3 = 1;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 3;
// }
// break;
// case 3:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
// ucLED4 = 1;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 4;
// }
// break;
// case 4:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED5 = 1;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 5;
// }
// break;
// case 5:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
// ucLED6 = 1;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 6;
// }
// break;
// case 6:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED7 = 1;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 7;
// }
// break;
// case 7:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED8 = 1;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 8;
// }
// break;
// case 8:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED8 = 0;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 9;
// }
// break;
// case 9:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED7 = 0;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 10;
// }
// break;
// case 10:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED6 = 0;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 11;
// }
// break;
// case 11:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED5 = 0;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 12;
// }
// break;
// case 12:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED4 = 0;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 13;
// }
// break;
// case 13:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED3 = 0;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 14;
// }
// break;
// case 14:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED2 = 0;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 15;
// }
// break;
// case 15:
// if(uiTimeCnt_01_08 >= const_time_level_01_08)
// {
// uiTimeCnt_01_08 = 0; /*時間計數器清零*/
//
// ucLED1 = 0;
//
// ucLed_update = 1; /*更新顯示*/
// ucLedStep_01_08 = 0;
// }
// break;
// }
//}
/**
* @brief LED服務函數
* @param 無
* @retval 無
**/
void LED_Flicker_09_16(void)
{
switch(ucLedStep_09_16)
{
case 0:
if(uiTimeCnt_09_16 >= const_time_level_09_16)
{
uiTimeCnt_09_16 = 0; /*時間計數器清零*/
if(ucLedDirFlag == 0) /*正方向*/
{
ucLED16 = 0;
ucLED9 = 1;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 1;
}
else
{
ucLED15 = 1;
ucLED16 = 0;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 7;
}
}
break;
case 1:
if(uiTimeCnt_09_16 >= const_time_level_09_16)
{
uiTimeCnt_09_16 = 0;
if(ucLedDirFlag == 0) /*正方向*/
{
ucLED9 = 0;
ucLED10 = 1;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 2;
}
else
{
ucLED16 = 1;
ucLED9 = 0;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 0;
}
}
break;
case 2:
if(uiTimeCnt_09_16 >= const_time_level_09_16)
{
uiTimeCnt_09_16 = 0; /*時間計數器清零*/
if(ucLedDirFlag == 0)
{
ucLED10 = 0;
ucLED11 = 1;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 3;
}
else
{
ucLED9 = 1;
ucLED10 = 0;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 1;
}
}
break;
case 3:
if(uiTimeCnt_09_16 >= const_time_level_09_16)
{
uiTimeCnt_09_16 = 0; /*時間計數器清零*/
if(ucLedDirFlag == 0)
{
ucLED11 = 0;
ucLED12 = 1;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 4;
}
else
{
ucLED10 = 1;
ucLED11 = 0;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 2;
}
}
break;
case 4:
if(uiTimeCnt_09_16 >= const_time_level_09_16)
{
uiTimeCnt_09_16 = 0; /*時間計數器清零*/
if(ucLedDirFlag == 0)
{
ucLED12 = 0;
ucLED13 = 1;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 5;
}
else
{
ucLED11 = 1;
ucLED12 = 0;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 3;
}
}
break;
case 5:
if(uiTimeCnt_09_16 >= const_time_level_09_16)
{
uiTimeCnt_09_16 = 0; /*時間計數器清零*/
if(ucLedDirFlag == 0)
{
ucLED13 = 0;
ucLED14 = 1;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 6;
}
else
{
ucLED12 = 1;
ucLED13 = 0;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 4;
}
}
break;
case 6:
if(uiTimeCnt_09_16 >= const_time_level_09_16)
{
uiTimeCnt_09_16 = 0; /*時間計數器清零*/
if(ucLedDirFlag == 0)
{
ucLED14 = 0;
ucLED15 = 1;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 7;
}
else
{
ucLED13 = 1;
ucLED14 = 0;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 5;
}
}
break;
case 7:
if(uiTimeCnt_09_16 >= const_time_level_09_16)
{
uiTimeCnt_09_16 = 0; /*時間計數器清零*/
if(ucLedDirFlag == 0)
{
ucLED15 = 0;
ucLED16 = 1;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 0;
}
else
{
ucLED14 = 1;
ucLED15 = 0;
ucLed_update = 1; /*更新顯示*/
ucLedStep_09_16 = 6;
}
}
break;
}
}
/**
* @brief 定時器0中斷函數
* @param 無
* @retval 無
**/
void ISR_T0(void) interrupt 1
{
TF0 = 0; /*清除中斷標誌*/
TR0 = 0; /*關中斷*/
// if(uiTimeCnt_01_08 < 0xffff) /*設定這個條件,防止uiTimeCnt超範圍。*/
// {
// uiTimeCnt_01_08 ++;
// }
if(uiTimeCnt_09_16 < 0xffff) /*設定這個條件,防止uiTimeCnt超範圍。*/
{
uiTimeCnt_09_16 ++;
}
Key_Scan();
if(uiVoiceCnt != 0)
{
uiVoiceCnt--; /*每次進入定時中斷都自減1,直到等於零爲止。才停止鳴叫*/
Beep=0; /*蜂鳴器是PNP三極管控制,低電平就開始鳴叫。*/
}
else
{
; /*此處多加一個空指令,想維持跟if括號語句的數量對稱,都是兩條指令。不加也可以。*/
Beep=1; /*蜂鳴器是PNP三極管控制,高電平就停止鳴叫。*/
}
TL0 = T1MS; /*initial timer0 low byte*/
TH0 = T1MS >> 8; /*initial timer0 high byte*/
TR0 = 1; /*開中斷*/
}
/*——————主函數——————*/
/**
* @brief 主函數
* @param 無
* @retval 實現LED燈閃爍
**/
void main()
{
/*單片機初始化*/
Init();
/*延時,延時時間一般是0.3秒到2秒之間,等待外圍芯片和模塊上電穩定*/
Delay_Long(100);
/*單片機外圍初始化*/
Init_Peripheral();
while(1)
{
/*LED服務函數*/
LED_Flicker_09_16();
/*LED更新函數*/
LED_Update();
/*按鍵服務的應用程序*/
Key_Service();
}
}
三、仿真實現
51單片機實現兩片聯級74HC595控制跑馬燈,獨立按鍵控制跑馬燈的方向