|
50黑幣
#include<reg52.h>
#include<stdlib.h> //包含隨機函數rand()的定義文件
#include<intrins.h> //包含_nop_函數定義的頭文件
#define OP_READ 0xa1 //器件地址以及讀取操作,0xa1即為1010 0001B
#define OP_WRITE 0xa0 //器件地址以及寫入操作,0xa1即為1010 0000B
#define uchar unsigned char
#define uint unsigned int
sbit K5=P3^2;
sbit RS=P2^0; //寄存器選擇位,將RS位定義為P2.0引腳
sbit RW=P2^1; //讀寫選擇位,將RW位定義為P2.1引腳
sbit E =P2^2; //使能信號位,將E位定義為P2.2引腳
sbit BF=P0^7; //忙碌標志位,將BF位定義為P0.7引腳
sbit SCL=P3^4; //將串行時鐘總線SCL位定義在為P3.4引腳
sbit SDA=P3^5;
uint data T0_cnt,D_cnt;
uchar data cnt_3,cnt_distance,cnt_cost,begin;
uchar data lose_power=1;
uchar data state_val,flag=0;
uchar data key_val, key_val_old; //定義新舊鍵值
uchar code cost_val[3]={1,2,1}; //收費標準 cost_val[0]白天 cost_val[1] 黑夜cost_val[2] 等待
uchar code digit[]={"012456789"};//定義字符數組顯示數字
uchar code string1[]={"price:"};//定義字符數組顯示提示信息
uchar code string2[]={"/km"};
uchar code string3[]={"total:"};
uint h,d,t;
void delay(uint i)//1ms延時基準函數
{
uchar j;
while(i--)
{
for(j=0;j<115;j++);
}
}
void delaynms(uint n)
{
delay(n);
}
void init_variant()//初始化一些變量的內容
{
cnt_3=0; //3分鐘的倒計時
D_cnt=0; //脈沖的個數
cnt_distance=0;//距離的計數
cnt_cost=0;//保存總價格
}
uchar scan_key()
{
uchar i,k;
i=P1;
if(i==0xff)
{
k=255;
} //無建按下
else //有建按下
{
delay(10); //延時去抖動
if(i!=P1)
{
k=255;
}
else
{
switch (i)
{
case 0xef:k=0;
break;//P1.4按下,啟動鍵
case 0xdf:k=1;
break;//P1.5按下,清除鍵
case 0xbf:k=2;
break;//P1.6按下,切換鍵
}
}
}
return k;
}
/*************************************
函數功能:開始數據傳送
**************************************/
void start()
//開始位
{
SDA=1; //SDA初始化為高電平"1"
SCL=1; //開始數據傳送時,要求SCL為高電平"1"
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
SDA=0; //SDA下降沿被認為是開始信號
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
SCL=0; //SCL為低電平時,SDA上數據才允許變化(即允許以后的數據傳遞)
}
/*************************************
函數功能:結束數據傳送
**************************************/
void stop()
//停止位
{
SDA=0; //SDA初始化為低電平"0"
SCL=1; //結束數據傳送時,要求SCL為高電平"1"
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
SDA=1; //SDA上升沿沿被認為是結束信號
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
SDA=0;
SCL=0;
}
/*************************************
函數功能:檢測應答位
**************************************/
bit Ask() //檢測應答
{
bit ack_bit; //儲存應答位
SDA=1; //發送設備(主機)應在時線鐘脈沖的高電平期間(SCL=1)
//釋放SDA線,以讓SDA線轉由接收設備(AT24Cxx)控制
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
SCL=1; //根據上述規定,SCL應為高電平
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
ack_bit=SDA;//接受設備(AT24Cxx)向SDA送低電平,表示已經接收到一個字節
//若送高電平,表示沒有接收到,傳送異常 結束發送
SCL=0; //SCL為低電平時,SDA上數據才允許變化(即允許以后的數據傳遞)
return ack_bit;//返回AT24Cxx應答位
}
/*************************************
函數功能:從AT24Cxx讀取數據
出口參數:x
**************************************/
unsigned char ReadData()//從AT24Cxx移入數據到MCU
{
unsigned char i;
unsigned char x; //儲存從AT24Cxx中讀出的數據
for(i=0;i<8;i++)
{
SCL=1; //SCL置為高電平
x<<=1; //將x中的各二進位向左移一位
x|=(unsigned char)SDA;//將SDA上的數據通過按位“或”運算存入x中
SCL=0; //在SCL的下降沿讀出數據
}
return(x); //將讀取的數據返回
}
/*************************************
函數功能:向AT24Cxx的當前地址寫入數據
入口參數:y(存儲待寫入的數據)
**************************************/
//在調用此數據寫入函數前需首先調用開始函數start(),所以SCL=0
void WriteCurrent(unsigned char y)
{
unsigned char i;
for(i=0;i<8;i++) //循環移入8個位
{
SDA=(bit)(y&0x80); //通過按位”與“運算將最高位數據傳送到S
//因為傳送時高位在前,低位在后
_nop_(); //等待一個機器周期
SCL=1; //在SCL的上升沿將數據寫入AT24Cxx
_nop_(); //等待一個機器周期
_nop_(); //等待一個機器周期
SCL=0; //將SCL重新置為低電平,
//以在SCL線形成傳送數據所需的8個脈沖
y<<=1; //將y中的各二進位向左移一位
}
}
/*************************************
函數功能:向AT24Cxx的當前地址寫入數據
入口參數:add(存儲指定的地址);dat(存儲待寫入的數據)
**************************************/
void Write(unsigned char add,unsigned char dat)
//在指定的地址addr處寫入數據WriteCurrent
{
start(); //開始數據傳遞
WriteCurrent(OP_WRITE);//選擇操作的AT24Cxx芯片,并告知要對其寫入數據
Ask();
WriteCurrent(add); //寫入指定地址
Ask();
WriteCurrent(dat); //向當前地址(上面指定的地址)寫入數據
Ask();
stop(); //停止數據傳遞
delaynms(4); //1個字節的寫入周期為1ms,最好延時1ms以上
}
/*************************************
函數功能:從AT24Cxx的當前地址讀取數據
出口參數:x(存儲讀出的數據)
**************************************/
unsigned char ReadCurrent()
{
unsigned char x;
start(); //開始數據傳遞
WriteCurrent(OP_READ);//選擇要操作的AT24Cxx芯片,并告知要讀其數據
Ask();
x=ReadData(); //將讀取的數據存入x
stop(); //停止數據傳遞
return x; //返回讀取的數據
}
/*************************************
函數功能:從AT24Cxx中的指定地址讀取數據
入口參數:set_addr
出口參數:x
**************************************/
unsigned char ReadSet(unsigned char set_addr)
//在指定地址讀取
{
start(); //開始數據傳遞
WriteCurrent(OP_WRITE);//選擇要操作的AT24Cxx芯片,并告知要對其寫入數據
Ask();
WriteCurrent(set_addr);//寫入指定地址
Ask();
return (ReadCurrent()) ; //從指定地址讀取數據并返回
}
/***********************************************
函數功能:判斷液晶模塊的忙碌狀態
返回值:result。result=1,忙碌:result=0,不忙
************************************************/
uchar BusyTest(void)
{
bit result;
RS=0; //根據規定,RS為低電平,RW為高電平時,可以讀狀態
RW=1;
E=1; //E=1,才允許讀寫
_nop_();
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬件反應時間
result=BF; //將忙碌標志電平賦給result
E=0; //將E恢復低電平
return result;
}
/**********************************************
函數功能:將模式設置指令或顯示地址寫入液晶模塊
入口參數:dictate
***********************************************/
void WriteInstruction(uchar dictate)
{
while(BusyTest()==1); //如果忙就等待
RS=0; //根據規定,RS和R/W同時為低電平時,可以寫入指令
RW=0;
E=0; //E置低電平(根據8-6,寫指令時,E為高脈沖
//就是讓E從0到1發生跳變,所以應先置"0'
_nop_();
_nop_(); //空操作兩個機器周期,給硬件反應時間
P0=dictate; //將數據送入P0口,即寫入指令或地址
_nop_();
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬件反應時間
E=1; //E置高電平
_nop_();
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬件反應時間
E=0; //當E由高電平跳變為低電平時,液晶模塊開始執行命令
}
/**********************************************
函數功能:指定字符顯示的實際地址
入口參數:x
***********************************************/
void WriteAddress(uchar x)
{
WriteInstruction(x|0x80); //顯示位置的確定方法規定為”80H+地址碼”
}
/**********************************************
函數功能:將數據(字符的標準ASII碼)寫入液晶模塊
入口參數:y(為字符常量)
***********************************************/
void WriteData(uchar y)
{
while(BusyTest()==1);
RS=1; //RS為高電平,RW為低電平時,可以寫入數據
RW=0;
E=0; //E置低電平(根據8-6,寫指令時,E為高脈沖
//就是讓E從0到1發生跳變,所以應先置"0'
P0=y; //將數據送入P0口,即將數據寫入液晶模塊
_nop_();
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬件反應時間
E=1; //E置高電平
_nop_();
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬件反應時間
E=0; //當E由高電平跳變為低電平時,液晶模塊開始執行命令
}
/**********************************************
函數功能:對LCD的顯示模式進行初始化設置
***********************************************/
void LcdInitiate(void)
{
delay(15); //延時15ms,首次寫指令時應給LCD一段較長的反應時間
WriteInstruction(0x38); //顯示模式設置:16x2顯示,5x7點陣,8位數據接口
delay(5); //延時5ms,給硬件一點反應時間
WriteInstruction(0x38);
delay(5);
WriteInstruction(0x38); //連續三次,確保初始化成功
delay(5);
WriteInstruction(0x0C); //顯示模式設置:顯示開,無光標,光標不閃爍
delay(5);
WriteInstruction(0x06); //顯示模式設置:光標右移,字符不移
delay(5);
WriteInstruction(0x01); //清屏指令,將以前的顯示內容清除
delay(5);
}
void lcd_show()
{
uchar i;
WriteAddress(0x00);
i=0; //從字符數組的第1個元素開始顯示
while(string1[i]!='\0') //只要沒有顯示到字符串的結束標志'0',就繼續
{
WriteData(string1[i]); //將第i個字符數組元素寫入LCD
i++; //指向下一個數組元素
}
h=cost_val[state_val]/10;
d=cost_val[state_val]%10;
WriteAddress(0x06);
WriteData(digit[h]); //將十位數字的字符常量寫入LCD price
WriteData(digit[d]); //將個位數字的字符常量寫入LCD
WriteAddress(0x08);
WriteData(0x5c); //幣符號
WriteAddress(0x09);
i=0; //從字符數組的第1個元素開始顯示
while(string2[i]!='\0') //只要沒有顯示到字符串的結束標志'0',就繼續
{
WriteData(string2[i]); //將第i個字符數組元素寫入LCD
i++; //指向下一個數組元素
}
WriteAddress(0x40);
i=0; //從字符數組的第1個元素開始顯示
while(string3[i]!='\0') //只要沒有顯示到字符串的結束標志'0',就繼續
{
WriteData(string3[i]); //將第i個字符數組元素寫入LCD
i++; //指向下一個數組元素
}
h=cnt_distance/10;
d=cnt_distance%10;
WriteAddress(0x46); //寫顯示地址,將十位數字顯示在第2行第8列
WriteData(digit[h]); //將十位數字的字符常量寫入LCD km
WriteData(digit[d]); //將個位數字的字符常量寫入LCD
WriteAddress(0x48);
i=1;
while(string2[i]!='\0') //只要沒有顯示到字符串的結束標志'0',就繼續
{
WriteData(string2[i]); //將第i個字符數組元素寫入LCD
i++; //指向下一個數組元素
}
h=cnt_cost/10; //取整運算,求得十位數字
d=cnt_cost%10; // 取余運算,求得個位數字
WriteAddress(0x4C);//寫顯示地址,將十位數字顯示在第2行第8列
WriteData(digit[h]); //將十位數字的字符常量寫入LCD 金額
WriteData(digit[d]); //將個位數字的字符常量寫入LCD
WriteAddress(0x4E);
WriteData(0x5C);
}
void Time0(void) interrupt 1 using 0 //"interrupt"聲明函數為中斷服務函數
//其后的1為定時器T0的中斷編號;0表示使用第0組工作寄存器
{
T0_cnt++;
if(T0_cnt>40) //如果計數>3999,計時1S
{
T0_cnt=0;
if(cnt_3<180)
{
cnt_3++;
}
else //超過180S,途中等待計價
{
cnt_3=0;
cnt_cost=cnt_cost+cost_val[2];
}
}
TH0=0x4C; //定時器T0的高8位重新賦值
TL0=0x00; //定時器T0的高8位重新賦值
}
void init1() interrupt 0 //T1中斷
{
if(K5==0)
{
delay(10);
if(TR0==1&&flag==1) //每來一個脈沖,中斷一次
{
cnt_3=0; //30S的計時清零
if(D_cnt<543)
{
D_cnt++;
}
else //計數543次,每次1.88米,表示一公里
{
D_cnt=0;
begin++;
if(begin>2)
{
cnt_distance=cnt_distance+1;
cnt_cost=cnt_cost+cost_val[state_val];
}
}
}
while(!K5);
}
}
void main()
{
LcdInitiate();
EX0=1; //允許使用外中斷
IT0=1; //選擇負跳變來觸發外中斷
EA=1; //開關總中斷
ET0=1; //定時器T0允許中斷
TMOD=0X01; //使用定時器T0的模式1
TH0=0x4C; //定時器T0高8位賦初值
TL0=0x00; //定時器T0高8位賦初值
TR0=0; //啟動定時器 TR0=1時啟動計數 TR0=0時停止計數
SDA=1; //SDA=1,SCL=1;使主從設備處于空閑狀態
SCL=1;
cnt_cost=ReadSet(2); //讀取保存的數據賦值于sec
cnt_distance=ReadSet(3);
flag=ReadSet(4);
while(1)
{
key_val=scan_key(); //255
if(key_val!=key_val_old)
{
key_val_old=key_val;
if(key_val!=255)
{
switch(key_val)
{
case 0: //啟動鍵
TR0=1;
begin=0;
//cnt_distance=0;
if(flag==0)
{
cnt_cost=(state_val==0)?6:7;
flag=1; //啟動計時,TR0=1為啟動了的標志
}
else
{
if(lose_power==0)
flag=0;
}
break;
case 1: //清除鍵
init_variant(); //清除變量
lose_power=0;
TR0=0; //關閉定時器
break;
case 2: //白天/黑夜的切換
if(state_val==0)
{
state_val=1;
}
else
{
state_val=0;
}
break;
}
}
}
WriteSet(4,flag);
WriteSet(2,cnt_cost);
WriteSet(3,cnt_distance);
lcd_show();
}
}
|
|