8051單片機搶答器C程序
/******************************************************************
本文引用地址:http://m.butianyuan.cn/article/201611/316477.htm* 自定義Macro
*******************************************************************/
//編碼的均為反向編碼
#define CLEAR 0x7f //定義清空的反碼
#define LED_BEGIN 0x01 // 定義開始時數碼管的顯示
#define LED_FOUL 0x38 // 犯規(guī)后顯示字母"F",數碼管編碼
#define LED_C 0x31 // 字母"C"的編碼
#define LED_L 0x71 // 字母"L"的編碼,兩個用來在主持人取消之后顯示"CL"--cancel
#define GET 1 // 這個是作為一個函數的參數來混的,就是成功搶答的意思
#define FOUL 0 // 和上面的參數一起混的,犯規(guī)---這兩個的用法在后面體現
#define READY 0x7e
/******************************************************************
* 自定義數據類型
*******************************************************************/
typedef unsigned char Byte; // 一個字節(jié)
typedef unsigned int Word; // 一個字,兩個字節(jié)
typedef bit Bool; // 模仿布爾型變量
//typedef sbit Port; // 本想用自定義一個端口類型的變量,比較方便,但是這句話步知道為何通不過編譯
/******************************************************************
* 定義MAX7219寄存器
*******************************************************************/
#define REG_NO_OP 0x00 // 定義空操作 register
#define DIG_1 0x01 // 定義數碼管1 register
#define DIG_2 0x02 // 定義數碼管2 register
#define DIG_3 0x03 // 定義數碼管3 register
#define DIG_4 0x04 // 定義數碼管4 register
#define DIG_5 0x05 // 定義數碼管5 register
#define DIG_6 0x06 // 定義數碼管6 register
#define DIG_7 0x07 // 定義數碼管7 register
#define DIG_8 0x08 // 定義數碼管8 register
#define REG_DECODE 0x09 // 定義解碼控制 register
#define REG_INTENSITY 0x0a // 定義顯示亮度 register
#define REG_SCAN_LIMIT 0x0b // 定義掃描限制 register
#define REG_SHUTDOWN 0x0c // 定義"shutdown"模式 register
#define REG_DISPLAY_TEST 0x0f // 定義"display test"模式 register
#define INTENSITY_MIN 0x00 // 定義最低顯示亮度
#define INTENSITY_MAX 0x0f // 定義最高顯示亮度
/*********************************************************************
* 定義硬件引腳連接
**********************************************************************/
sbit DA
sbit LOAD=P2^1; // MAX7219的鎖存端口
sbit CLK=P2^2; // MAX7219的時鐘端口
//sbit HOST_SWITCH=P0^0; // 主持人開關的接口
sbit HOST_START=P0^0; //主持人按鍵,用來重新開始的按鍵 start
sbit HOST_CANCEL=P0^1; //主持人用來取消搶答的按鍵 clear
sbit SWITCH1_3=P1^4; // 調節(jié)倒計時時間的撥碼開關,下劃線前面的號代表開關的序號,下劃線后面的號代表該開關的數值
sbit SWITCH2_2=P1^5; // 同上
sbit SWITCH3_2=P1^6; // 同上
sbit SWITCH4_1=P1^7; // 同上
sbit BEEP=P0^7; //定義蜂鳴器端口
sbit LS138_C=P2^4; //定義譯碼器輸入端
sbit LS138_B=P2^5; //同上
sbit LS138_A=P2^6; //同上
sbit LS138_E1=P2^7; //定義譯碼器使能端
/*********************************************************************
* 定義全局變量
**********************************************************************/
Byte da
Byte da
Byte da
Byte da
Bool da
Bool da
Bool da
Byte da
co
/***********************************************************************
* 共陰極七段數碼管顯示對應段查詢表(數字0-9分別對應co
***********************************************************************/
Byte co
{0x7e,0x30,0x6d,0x79,0x33,0x5b,0x5f,0x70,0x7f,0x7b};
Byte co
{0x01,0x4f,0x12,0x06,0x4c,0x24,0x20,0x0f,0x00,0x04};
/***********************************************************************
* 函數聲明
***********************************************************************/
void MAX7219_SendByte (Byte dataout);
void MAX7219_Write (Byte reg_number, Byte dataout);
void MAX7219_DisplayChar(Byte digit, Byte character);
void MAX7219_Clear (void);
void MAX7219_SetBrightness (Byte brightness);
void MAX7219_DisplayTestStart (void);
void MAX7219_DisplayTestStop (void);
void MAX7219_ShutdownStart (void);
void MAX7219_ShutdownStop (void);
void MAX7219_Init (void);
void Delay10ms(void);
Bool GetHostStartKey (void);
Bool GetHostCancelKey (void);
void GetCounter(void);
Byte GetPressed(Byte KeyState);
void IT0_Init(void);
void Timer0_Overflow();
void PressedHandle(Byte keyPressed);
void GetOrFoulHandle(Bool state);
void CancelHandle();
void SPEAKER_count (void); //聲明倒計時聲音函數
void SPEAKER_start(void); //聲明開始搶答聲音函數
void SPEAKER_get(void); //聲明搶到聲音函數
void SPEAKER_foul(void); // 聲明犯規(guī)聲音函數
/***********************************************************************
* MAX7219_SendByte()
*
* 描述: 向MAX7219傳送一個字節(jié)的數據
* Arguments : dataout = da
* Returns : none
*************************************************************************/
void MAX7219_SendByte (Byte dataout)
{
Byte i;
for (i=8;i>0;i--)
{
Byte mask=1<<(i-1);//mask是個掩碼,取位使用
CLK=0;//MAX7219的位傳入是在時鐘的上升沿之前,所以在每發(fā)一位之前都要變?yōu)榈碗娖?/p>
if (dataout&mask)
DA
else
DA
CLK=1;//八個bit都傳遞完成后變?yōu)楦唠娖?鎖存
}
}
/***********************************************************************
* MAX7219_Write()
*
* 描述: 向 MAX7219 寫命令
* Arguments : reg_number = register to write to
* dataout = da
* Returns : none
未完~
***************************************************************************/
void MAX7219_Write (Byte reg_number, Byte dataout)
{
LOAD=0;//也是鎖存上升沿之前的,發(fā)這兩個字節(jié)之前要變?yōu)榈碗娖?/p>
MAX7219_SendByte(reg_number);//發(fā)送寄存器地址
MAX7219_SendByte(dataout);//發(fā)送數據
LOAD=1;//變?yōu)楦唠娖?鎖存
}
/**************************************************************************
* MAX7219_DisplayChar()
*
* 描述: 使某一位顯示一個數字
* Arguments : digit = digit number (0-7)
* character = character to display (0-9, A-Z)
* Returns : none
**************************************************************************/
void MAX7219_DisplayChar(Byte digit, Byte character)
{
MAX7219_Write(digit, character);
}
/**************************************************************************
* MAX7219_Clear()
*
* 描述: 清除所有位的顯示
* Arguments : none
* Returns : none
***************************************************************************/
void MAX7219_Clear (void)
{
Byte i;
for (i=1; i<=2; i++)
MAX7219_Write(i, CLEAR);//把八個數碼管全都清零了,已經寫反了^_^
}
/**************************************************************************
* MAX7219_SetBrightness()
*
* 描述: 設置數碼管顯示亮度
* Arguments : brightness (0-15)
* Returns : none
***************************************************************************/
void MAX7219_SetBrightness (Byte brightness)
{
brightness &= 0x0f;
MAX7219_Write(REG_INTENSITY, brightness);
}
/**************************************************************************
* MAX7219_DisplayTestStart()
*
* 描述: 進入 test 模式
* Arguments : none
* Returns : none
***************************************************************************/
void MAX7219_DisplayTestStart (void)
{
MAX7219_Write(REG_DISPLAY_TEST, 1);
}
/**************************************************************************
* MAX7219_DisplayTestStop()
*
* 描述: 退出 test 模式
* Arguments : none
* Returns : none
***************************************************************************/
void MAX7219_DisplayTestStop (void)
{
MAX7219_Write(REG_DISPLAY_TEST, 0);
}
/**************************************************************************
* MAX7219_ShutdownStart()
*
* 描述: 進入 shutdown 模式
* Arguments : none
* Returns : none
***************************************************************************/
void MAX7219_ShutdownStart (void)
{
MAX7219_Write(REG_SHUTDOWN, 0);
}
/**************************************************************************
* MAX7219_ShutdownStop()
*
* 描述: 退出 shutdown 模式
* Arguments : none
* Returns : none
***************************************************************************/
void MAX7219_ShutdownStop (void)
{
MAX7219_Write(REG_SHUTDOWN, 1);
}
/**************************************************************************
* MAX7219_Init()
*
* Description: MAX7219初始化模塊; 應該先于其他MAX7219函數而被調用
* Arguments : none
* Returns : none
***************************************************************************/
void MAX7219_Init (void)
{
DA
CLK=1;
LOAD=1;
MAX7219_Write(REG_SCAN_LIMIT,1);//這里設置的是掃描兩個數碼管
MAX7219_Write(REG_DECODE, 0x00);
MAX7219_SetBrightness(INTENSITY_MAX);//設置最大亮度顯示
MAX7219_DisplayTestStart();
MAX7219_DisplayTestStop();
MAX7219_ShutdownStop();
MAX7219_Clear();
}
/**************************************************************************
* Delay_100us()
*
* 描述: 延時100us,主要用在消除開關抖動時
* Arguments : none
* Returns : none
***************************************************************************/
void Delay10ms(void)
{
unsigned char i,j;
for(i=20;i>0;i--)
for(j=248;j>0;j--);
}
/**************************************************************************
* GetHostStartKey()
*
* Description: 取得主持人開始按鍵的鍵值
* Arguments : none
* Returns : 1-->主持人按鍵; 0-->主持人未按鍵
***************************************************************************/
Bool GetHostStartKey (void)
{
if (HOST_START ==1)
return 0;
else
Delay10ms ();//如果發(fā)現主持人按鍵接通,要先延時100us,防止抖動
if (HOST_START==1)
return 0;
else
return 1;//延時時候還是接通,則判斷為該鍵確實按下
}
/**************************************************************************
* GetHostCancelKey()
*
* Description: 取得主持人取消按鍵的鍵值
* Arguments : none
* Returns : 1-->主持人按鍵; 0-->主持人未按鍵
***************************************************************************/
Bool GetHostCancelKey (void)
{
if (HOST_CANCEL ==1)
return 0;
else
Delay10ms ();//如果發(fā)現主持人按鍵接通,要先延時100us,防止抖動
if (HOST_CANCEL ==1)
return 0;
else
return 1;//延時時候還是接通,則判斷為該鍵確實按下
}
/**************************************************************************
* GetCounter
*
* Description: 取得預先設置的倒計時時間
* Arguments : none
* Returns : none
***************************************************************************/
void GetCounter(void)
{
beginNum=1;//在所有開關都沒有撥動的時候倒計時為1秒,比設置為0秒要好
intrCounter=20;//每一秒對應的中斷次數為20次
if (SWITCH1_3==1)
{
beginNum+=3;
}
if (SWITCH2_2==1)
{
beginNum+=2;
}
if (SWITCH3_2==1)
{
beginNum+=2;
}
if (SWITCH4_1==1)
{
beginNum+=1;
}//以上判斷語句為判斷撥碼開關狀態(tài)
intrCounter=20*beginNum;//計算總掃描次數
}
/**************************************************************************
* GetPressed
*
* Description: 從P1口連接搶答端的四位來判斷搶答情況
* Arguments : Byte KeyState-->P1 state
* Returns : 搶答端的號碼 ; 0-->沒人搶答
***************************************************************************/
Byte GetPressed(Byte KeyState)
{
Byte key;//記錄搶答端的號碼
KeyState&=0x0f;//取P1口的低四位
switch (KeyState)
{
case 0x0f: key=0;break;//全高,無人搶答
case 0x0e: key=1;break;//只有P1.1,key1搶答
case 0x0d: key=2;break;//只有P1.2,key2搶答
case 0x0b: key=3;break;//只有P1.3,key3搶答
case 0x07: key=4;break;//只有P1.4,key4搶答
}
/*
switch (KeyState)
{
case 0x00: key=0;break;//全高,無人搶答
case 0x01: key=1;break;//只有P1.0,key1搶答
case 0x02: key=2;break;//只有P1.1,key2搶答
case 0x04: key=3;break;//只有P1.2,key3搶答
case 0x08: key=4;break;//只有P1.3,key4搶答
}
*/
//上面是在用高電平來判斷搶答狀態(tài)時的程序,經證明不知道為何無效
return key;
}
/**************************************************************************
* IT0_Init
*
* Description: 初始化計時器T0的狀態(tài)
* Arguments : none
* Returns : none
***************************************************************************/
void IT0_Init(void)
{
TMOD=0x01;//設置T0在方式1下工作
TH0=0x3C;
TL0=0xAF;//這兩個寄存器存的是計數器的計數開始的值,計算發(fā)現這兩個值累加至溢出后正好是50ms
ET0=1;//使T0中斷可以溢出
EA=1;//開啟總中斷
TF0=0;//溢出位清零
TR0=1;//開啟T0
}
/**************************************************************************
* Timer0_Overflow() interrupt 1
*
* Description: 中斷溢出服務程序, 采用的是中斷方式1, 后面最好不加using選擇寄存器組以免與系統用在主程序的寄存器沖突
* Arguments : none
* Returns : none
***************************************************************************/
void Timer0_Overflow() interrupt 1
{
static Byte second=20;//用20次中斷來判斷1秒
TH0=0x3C;
TL0=0xAF;
second--;
intrCounter--;
/*
if (second==0)//每隔一秒的操作
{
MAX7219_DisplayChar(DIG_2,co
second=20;//重新賦值每秒計數器
if (intrCounter==0)
{
TR0=0;//關閉T0計數器
isStart=1;//計時結束,進入正常搶答
//SPEAKER_start();//開始搶答的聲音
}
//待顯示"0"以后就開始搶答
else
{
//SPEAKER_count();//倒計時聲音
}
}
*/
if (second==0)//每隔一秒的操作
{
//if (showNum!=1) SPEAKER_count();//倒計時聲音
//else SPEAKER_start();//開始搶答的聲音
MAX7219_DisplayChar(DIG_2,co
second=20;//重新賦值每秒計數器
}//待顯示"0"以后就開始搶答
if (intrCounter==0)
{
TR0=0;//關閉T0計數器
isStart=1;//計時結束,進入正常搶答
}
}
/**************************************************************************
* PressedHandle
*
* Description: 按鍵處理
* Arguments : Byte keyPressed-->按下的按鍵
* Returns : none
***************************************************************************/
void PressedHandle(Byte keyPressed)
{
MAX7219_Clear();//LED clear
MAX7219_DisplayChar(DIG_2,co
}
/**************************************************************************
* GetOrFoulHandle(Bool state)
*
* Description: 正常搶答或是犯規(guī)處理
* Arguments : Bool state-->是GET和FOUL兩個宏的取之之一
* Returns : none
***************************************************************************/
void GetOrFoulHandle(Bool state)
{
if (!state)
{
MAX7219_DisplayChar(DIG_1,LED_FOUL);//如果是犯規(guī)的話左邊的LED要顯示"F",foul
}
}
/**************************************************************************
* CancelHandle()
*
* Description: 處理主持人取消倒計時
* Arguments : none
* Returns : none
***************************************************************************/
void CancelHandle()
{
MAX7219_DisplayChar(DIG_1,LED_C);
MAX7219_DisplayChar(DIG_2,LED_L);//主持人取消倒計時之后,兩個數碼管顯示"CL"-->cancel
}
/**************************************************************************
* delayus()
*
* Description: 延時程序
* Arguments : t-->us
* Returns : time delayed
***************************************************************************/
void delayus(unsigned char t )
{
unsigned char j;
for(;t>0;t--)
for(j=19;j>0;j--);
}
/**************************************************************************
* SPEAKER_count/start/foul/get()
*
* Description: speaker發(fā)聲程序->計數/開始/犯規(guī)/搶答 四種聲音
* Arguments : none
* Returns : none
***************************************************************************/
void SPEAKER_count(void)
{
unsigned char i;
for (i=0;i<10;i++)
{
BEEP =1; //點亮
delayus(20);
BEEP =0; //熄滅
delayus(20);
}
}
void SPEAKER_start(void)
{
unsigned char i;
for(i=0;i<200;i++)
{
BEEP =1; //點亮
delayus(10);
BEEP =0; //熄滅
delayus(10);
}
}
void SPEAKER_foul(void)
{
unsigned char i;
for(i=0;i<250;i++)
{
BEEP =1; //點亮
delayus(15);
BEEP =0; //熄滅
delayus(17);
}
}
void SPEAKER_get(void)
{
unsigned char i;
for(i=0;i<250;i++)
{
BEEP =1; //點亮
delayus(10);
BEEP =0; //熄滅
delayus(10);
}
for(i=0;i<250;i++)
{
BEEP =1; //點亮
delayus(20);
BEEP =0; //熄滅
delayus(20);
}
}
/**************************************************************************
* 主程序
***************************************************************************/
void main()
{
Byte keyPressed,i;//選手按鍵號碼,沒有的話為0
Bool hostPressed;//用來記錄主持人按鍵取消,0為沒有動作,1為取消
number_temp=P1&0xf0;//P1口上次的狀態(tài),在調整倒計時時間的時候用到的
LS138_E1=1; //譯碼器初始化
MAX7219_Init();//數碼管初始化
GetCounter();//獲取開始時候設置的倒計時時間
MAX7219_DisplayChar(DIG_1,co
MAX7219_DisplayChar(DIG_2,READY);//調時間的時候右位的顯示
//while(1);
while(GetHostStartKey()==0)//當主持人沒有按鍵的時候進入循環(huán)
{
if (number_temp!=(P1&0xf0))//若調整了倒計時時間,則P1口狀態(tài)變了,就要重新設置和顯示
{
GetCounter();//獲取調整以后的倒計時時間
MAX7219_DisplayChar(DIG_1,co
number_temp=P1&0xf0;//記錄下來現在P1口的狀態(tài),以備后面的比較
}
} //當主持人按鍵以后就結束調整進入搶答倒計時
MAX7219_DisplayChar(DIG_1,READY);
while(GetHostCancelKey()==0);
MAX7219_DisplayChar(DIG_1,READY);//清空右邊一位數碼管
MAX7219_DisplayChar(DIG_2,co
for (i=100;i--;i>0)
Delay10ms();//防止后面出現連讀的情況..
//調整好倒計時時間后,按下start顯示"--",再按下cancel則顯示倒計時時間,此時可以開始倒計時了.
counterBack=intrCounter;
//這里原來寫的是while(1),寫上后就不行了,不知道為何..
while(1)//這里要用自己加的循環(huán)來把程序束縛在這里運行
{
showNum=beginNum;//設置要顯示的時間,當然時從倒計時時間開始
intrCounter=counterBack;//設置總中斷的次數
TR0=0;//禁用計時器0
isPressed=0;//記錄是否有人按鍵
isStart=0;//沒有開始搶答
while(GetHostStartKey()==0);
IT0_Init();//初始化計時器0, 啟用.
MAX7219_DisplayChar(DIG_1,CLEAR);//清空左邊一位數碼管
MAX7219_DisplayChar(DIG_2,co
while(!isPressed)//如果沒有記錄到有人按鍵就進入
{
keyPressed=GetPressed(P1);//查詢一下P1口的狀態(tài),即按鍵情況
hostPressed=GetHostCancelKey();
if (!keyPressed&&!hostPressed)//如果沒有人按鍵,就進入下次循環(huán)
continue;
else
{
TR0=0;//關閉定時器
isPressed=1;//記錄到有人按鍵,提供條件跳出循環(huán)
}
}
if (keyPressed!=0)
{
if (isStart)//如果已經開始搶答
{
PressedHandle(keyPressed);//處理按鍵,即顯示搶答選手號碼
GetOrFoulHandle(GET);//處理搶答
LS138_E1=0; //譯碼器準備工作
switch (keyPressed)
{
case 1: LS138_A=0;LS138_B=0;LS138_C=0;break; //1號成功燈亮
case 2: LS138_A=0;LS138_B=1;LS138_C=0;break; //2號成功燈亮
case 3: LS138_A=1;LS138_B=0;LS138_C=0;break; //3號成功燈亮
case 4: LS138_A=1;LS138_B=1;LS138_C=0;break; //4號成功燈亮
default : break;
}
//SPEAKER_get();//處理搶答
}
else//否則,沒有開始搶答
{
PressedHandle(keyPressed);//處理按鍵,即顯示搶答選手號碼
GetOrFoulHandle(FOUL);//處理犯規(guī),必須要放在后面,因為顯示數字的里面有一個clear
LS138_E1=0; //譯碼器準備工作
switch (keyPressed)
{
case 1: LS138_A=0;LS138_B=0;LS138_C=1;break; //1號犯規(guī)燈亮
case 2: LS138_A=0;LS138_B=1;LS138_C=1;break; //2號犯規(guī)燈亮
case 3: LS138_A=1;LS138_B=0;LS138_C=1;break; //3號犯規(guī)燈亮
case 4: LS138_A=1;LS138_B=1;LS138_C=1;break; //4號犯規(guī)燈亮
default : break;
}
//SPEAKER_foul();//犯規(guī)發(fā)聲
}
}
if (hostPressed==1)
{
CancelHandle();
for (i=100;i--;i>0)
Delay10ms();//防止后面出現連讀的情況..
}
while(GetHostCancelKey()==0);//如果主持人沒有按鍵再次開始,則停在次死循環(huán)處
LS138_E1=1; //關閉譯碼器
MAX7219_DisplayChar(DIG_1,READY);//清空右邊一位數碼管
MAX7219_DisplayChar(DIG_2,co
//到這里一次循環(huán)結束
}
}
評論