基于單片機的出租車計價系統設計
1 設計目的,要求和設計方法
1.1設計目的
(1)學習基本理論在實踐中綜合運用的初步經驗,掌握電路設計的基本方法、設計步驟,培養綜合設計與調適能力,進一步提高綜合運用所學知識和設計能力的的目的。
(2)掌握C語言程序設計方法。
(3)培養實踐技能,提高分析和解決實際問題的能力,并具備一定程度的設計能力。
1.2基本設計要求
本出租車自動計費,上電后顯示最初的起步價,里程計費單價,等待時間計費單價,通過按鍵可以調節起步價,里程計費單價,等待時間計費單價。同時具有運行,等待,暫停等狀態,可以顯示暫停的時間。出租車停止后能夠顯示行駛的總費用。
1.3性能指標
1、費用的計算是按行駛里程收費。設起步價為7.00元。
2、當里程<3km時,按起價計算費用。
3、當里程>3km時,每公里按1.7元計費。
4、等待累計時間>2min時,按1.5元/min計費。
1.4顯示功能
1、顯示行駛里程:用兩位數字顯示,顯示方式為“XX”,單位為km。計程范圍0-99km,精確到1km。
2、顯示等候時間:用兩位數字顯示,顯示方式為“XX”,單位為min。計時范圍0-59min,精確到1min。
3、顯示總費用:用四位數字顯示,顯示方式為“XXX.X”,單位為元。計價范圍0-999.9元,精確到0.1元。
1.3 設計方法
本設計采用AT89C51單片機為主控器,并用開關或者頻率信號發生器模擬車速,利用AT89C51的定時器工作在方式1下定時實現對出租車的計時及計價設計,輸出采用LCD液晶顯示屏顯示。
2 設計方案及原理
2.1 設計方案
本設計,利用單片機豐富的端口和控制的靈活性,采用AT89C51單片機為主控器,并用開關或頻率信號發生器模擬車速,通過石英振蕩器,LCD液晶顯示和發光二級管以及單片機內部程序構成完整的出租車計價系統,基本可以實現了按運行狀態計價的功能,價格調整功能。
2.2 設計原理
出租車計價是根據車所行駛的路程以及乘客乘車的里程綜合決定的。出租車行駛總路程可以通過車輪的周長乘車輪旋轉圈數得到。即可計算得到車輪旋轉幾周出租車能行駛一公里的路程。通過計數接收到的脈沖個數,計算出當前所行駛的路程。同時,通過LCD液晶顯示器顯示當前的行駛里程、等待時間和需支付的車費。出租車計價器用于記錄里程、起步公里數與價格的關系。模擬出租車計價器能根據總里程數、起步公里數的情況作出相應報價等。
3 出租車計價器系統的硬件設計
3.1 出租車硬件框圖
圖2-1系統的硬件框圖
3.2 出租車計價器硬件連接圖
AT89C51的最小系統:時鐘電路是單片機內部有一個高增益反相放大器,其輸入端為芯片引腳XTAL1,其輸出端為引腳XTAL2。通過這兩個引腳在芯片外并接石英晶體振蕩器和兩只電容(電容和一般取30pF)。這樣就構成一個穩定的自激振蕩器。復位操作是按鍵復位,按鍵復位具有上電復位功能外,若要復位,只要按圖中的RESET鍵,電源VCC經電阻R1、R2分壓,在RESET端產生一個復位高電平。AT89C51的最小系統硬件圖如圖3.2(a)所示。
圖3.2(a) AT89C51的最小系統硬件圖
液晶顯示和LED燈指示電路:顯示電路對于現實電路我們采用液晶顯示,液晶又分字符型和點陣型,我們使用的液晶是字符型液晶,并且帶字符庫的,不需要查找代碼,英文字符就可以。運行狀態指示電路采用發光二級管對運行方式進行指示,可清楚看到計價器的運行狀態。液晶顯示和LED燈指示電路如圖3.2(b)所示。
圖3.2(b) 液晶顯示和LED燈指示電路
按鍵電路:按鍵電路是通過按鍵去調節初始選擇、調節,模式選擇,開始運行,終止等功能。霍爾傳感器里程計算是通過安裝在車輪旁的霍爾傳感器檢測到的信號,送到單片機,經處理計算,送給顯示單元的。而由于本次實驗室的局限不能利用霍爾傳感器,所以我們利用按鍵來代替霍爾傳感器,通過按一次鍵代表汽車行駛了1公里,當在行駛過程中通過按鍵來選擇運行狀態。按鍵電路如圖3.2(c)所示。
圖3.2(c) 按鍵電路
4 出租車計價器系統的軟件設計
4.1系統流程圖
出租車計價器系統流程圖如附錄1所示。
4.2源程序代碼
根據設計的要求,在本系統中主要有主程序和延時程序,主程序中包括中斷服務程序和鍵盤顯示子程序、計算子程序等。現在應用更廣泛的是單片機C語言,因其簡單明了,故此次課程設計采用單片機C語言編程。程序源代碼如附錄2所示。
4.3程序調試域運行結果
在Keil C51 uvision4中對系統建立項目,選定合適的單片機AT89C51,創建一個計價器的新文件,并將上述的源程序進行編輯和選項操作進行編譯,以生成計價器的HEX文件。
5 系統仿真及實際調試
5.1運行圖示
仿真開始后,首先按動初始選擇按鈕, 則可以在運行單價,等待單價,起步價之間切換,每按一次切換一個量,選定后可以按動初始調節按鈕進行調節,調節完成后,按下開始運行按鈕則開始運行,選擇運行模式后開始計費,不同的運行模式會有不同的指示燈亮起,在運行過程中液晶屏會隨時顯示走過的路程和等待的時間,以及總的費用。仿真圖如附錄3所示。
5.2仿真問題分析
問題需要改進,例如本設計經過多次的調試與改正,最終達到了設計方案中的結果。在仿真試驗中各種設計要求都能夠得到驗證。但由于設計的原因,本系統仍然總在一些自動化程度不夠,運行模式還需要手動選擇。
6 總結
出租車計費器系統的設計已經全部完成,能利用AT89C51單片機對出租車啟動,停止,暫停等運行狀態進行選擇和顯示,能將單價和費用適時通過LCD顯示出來。本款出租車計價器包括單價輸出、單價調整、顯示當前的總費用等功能。雖然達到了基本的設計目的,但仍然存在一些問題,策劃的不夠全面,對單片機不夠熟悉,系統編寫不夠簡潔完善,使得計價器計費不夠細致,不夠人性化。
經過這次課程設計讓我學會了很多的東西。經過自己努力,基本上完成了設計要求的內容,在系統可行性分析、原理圖設計等方面都作了一些實際工作,同時也遇到了一些問題,存在一些不足。學會了怎么去查找些資料,把所找到的東西與自己的問題所結合起來并給予解決。這次設計使我學會把以前學到的理論知識應用于實踐,使我認識到理論知識與實踐之間有一定的差距,只有通過不斷的努力學習和實踐才能很好的把理論知識應用到實踐當中,也只有通過不斷的實踐才能加深對理論知識的理解。
參考文獻
[1] 王思明,張金敏,張鑫等.單片機原理及應用系統設計.北京:科學出版社,2012
[2] 張金敏,董海棠,高博等.單片機原理與應用系統設計.成都:西南交通大學出版社, 2010
[3] 李華,王思明,張金敏.單片機原理及應用.蘭州:蘭州大學出版社,2001
附錄1
附錄2
#include<reg52.h>
#define ucharunsigned char
#define uintunsigned int
/*定義1602讀寫使能端口*/
sbit rw=P2^1;
sbit rs=P2^0;
sbit en=P2^2;
sbit b=P0^7;
sbit gl=P1^0;
void lcd_init(); //lcd設置函數
void wr_com(ucharcommand); //命令函數
void wr_data(uchardata0); //數據函數
void lcd_clear(); //清屏函數
void lcd_set(); //屏幕設置函數
void busy(); //測忙函數
void display(); //啟動前顯示函數
voidprintstring(uchar *s);
void display1(); //啟動后顯示函數
void key1(); //啟動前按鍵
sbit K=P3^0;
sbit K1=P3^1;
sbit K2=P3^5; //開始計價
sbit K3=P3^3; //模式選擇(行程/等待)
sbit K4=P3^4; //復位
sbit g=P2^7;
sbit aa=P2^3; //運行
sbit bb=P2^4; //等待
sbit cc=P2^5; //暫停
//sbitled_run=P3^0;
//sbit led_await=P3^1;
//sbitled_stop=P3^2;
bit f_start;
bit jump_in;
bit jump_out;
void key(); //啟動后按鍵
/*定義液晶顯示地址數組*/
uchar codetable[]={0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39,0x70};
uchardispbuf[]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};
uchar dispbuf1[]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};
uchar v,v1;
uchar i,m=0;
void init();
void chuli();
void chuli1();
void delay(uchart); //延時函數
void clear(); //終止函數
void set(uchar x);
void set1(uchary);
uchar command;
ucharcount,count1;
ucharsecond,minite,second1,minite1;
unsigned inttt,tt1;
unsigned charvalue1=70,value2=17,value3=15; //value1=起步價,value2=超出每公里價格,value3=等待價格
unsigned intmoney;
void main()
{
// Init_Timer1(); //定時器中斷函數
EX0=1;//開外部中斷0
IT0=1;//采用跳變沿觸發方式
init();
lcd_init();
while(1)
{
key1();
chuli();
display();
if(jump_in==1)
{
jump_in=0;
lcd_init();
while(1)
{
key();
chuli1();
display1();
if(jump_out==1)
{
lcd_init();
jump_out=0;
clear();
break;
}
}
}
}
}
void clear(){
TR0=0;
TR1=0;
money=0;
second=0;
second1=0;
minite=0;
minite1=0;
value1=70;
value2=17;
value3=15;
// led_run=1;
//led_await=1;
// led_stop=1;
v=0;
v1=0;
for(i=0;i<15;i++)
{
dispbuf=0;
dispbuf1=0;
}
m=0;
}
void init() //中斷定時
{
TMOD=0x11;
TH0=(65536-50000)/256;
TL0=(65536-50000)%256;
TH1=(65536-50000)/256;
TL1=(65536-50000)%256;
ET0=1;
ET1=1;
EA=1;
TR0=0;
TR1=0;
}
void t0_(void)interrupt 1 using 0 //中斷1模式0 用來計時 秒、分
{
count++;
if(count==20)
{
count=0;
second++;
if(second==60)
{
second=0;
minite++;
if(minite==99)
{
minite=0;
}
}
}
TH0=(65536-50000)/256;
TL0=(65536-50000)%256;
dispbuf1[0]=m/10;
dispbuf1[1]=m%10;
dispbuf1[2]=second/10;
dispbuf1[3]=second%10;
}
void t1_(void)interrupt 3 using 3 //中斷3模式3用來計時 秒、分
{
count1++;
if(count1==20)
{
count1=0;
second1++;
if(second1==60)
{
second1=0;
minite1++;
if(minite1==99)
{
minite1=0;
}
}
}
TH1=(65536-50000)/256;
TL1=(65536-50000)%256;
dispbuf1[4]=minite1/10;
dispbuf1[5]=minite1%10;
dispbuf1[6]=second1/10;
dispbuf1[7]=second1%10;
}
void lcd_init()
{
wr_com(0x3c);
wr_com(0x06);
wr_com(0x0c);
wr_com(0x01);
}
void wr_com(ucharcommand) //命令
{
busy();
rs=0;
rw=0;
P0=command;
en=1;
en=0;
}
void wr_data(uchardata0) //數據
{
busy();
rs=1;
rw=0;
P0=data0;
en=1;
en=0;
}
void busy() //測忙函數
{
while(1)
{
en=0;
rs=0;
rw=1;
P0=0xff;
en=1;
if(b!=1)break;
}
en=0;
}
void chuli() //分離出百位十位個位
{
dispbuf[0]=value1/100%10;
dispbuf[1]=value1/10%10;
dispbuf[2]=value1%10;
dispbuf[3]=value2/100%10;
dispbuf[4]=value2/10%10;
dispbuf[5]=value2%10;
dispbuf[6]=value3/100%10;
dispbuf[7]=value3/10%10;
dispbuf[8]=value3%10;
}
void display() //初始顯示函數
{
set(0);
printstring("step :");
wr_data(table[dispbuf[0]]);
wr_data(table[dispbuf[1]]);
printstring(".");
wr_data(table[dispbuf[2]]);
set(16);
printstring("mileage :");
wr_data(table[dispbuf[3]]);
wr_data(table[dispbuf[4]]);
printstring(".");
wr_data(table[dispbuf[5]]);
set1(0);
printstring("await :");
wr_data(table[dispbuf[6]]);
wr_data(table[dispbuf[7]]);
printstring(".");
wr_data(table[dispbuf[8]]);
set1(16);
printstring("run : ");
wr_data(table[dispbuf1[0]]);
wr_data(table[dispbuf1[1]]);
printstring(":");
wr_data(table[dispbuf1[2]]);
wr_data(table[dispbuf1[3]]);
}
void chuli1()
{
if(f_start==1)
{
tt=minite*60+second;
tt1=minite1;
if(m<=3)
{money=value1+value3*tt1;}
if(m>3)
{money=value1+value2*(m-3)+value3*tt1;}
}
else if(f_start==0)
{
money=0;
}
dispbuf1[8]=money/100%10;
dispbuf1[9]=money/10%10;
dispbuf1[10]=money%10;
}
void display1() //運行后顯示函數
{
set(0);
printstring("run: ");
//wr_data(table[m]);
wr_data(table[dispbuf1[0]]);
wr_data(table[dispbuf1[1]]);
//printstring(":");
//wr_data(table[dispbuf1[2]]);
//wr_data(table[dispbuf1[3]]);
set(14);
printstring("await:");
wr_data(table[dispbuf1[4]]);
wr_data(table[dispbuf1[5]]);
printstring(":");
wr_data(table[dispbuf1[6]]);
wr_data(table[dispbuf1[7]]);
set1(0);
printstring("sum is: ");
wr_data(table[dispbuf1[8]]);
wr_data(table[dispbuf1[9]]);
printstring(".");
wr_data(table[dispbuf1[10]]);
}
voidprintstring(uchar *s) //輸出顯示函數
{
while(*s)
wr_data(*s++);
}
void key() //啟動按鍵
{
if(K3==0)
{
delay(50); //延時消抖動
if(K3==0)
{
while(K3==0);
v1++;
if(v1==4) v1=1;
}
}
switch(v1)
{
case1:f_start=1;TR0=1;TR1=0;/*led_run=0;led_await=1;led_stop=1*/;aa=0;bb=1;cc=1;break; //運行
case2:TR0=0;TR1=1;/*led_run=1;led_await=0;led_stop=1*/;aa=1;bb=0;cc=1;break; //等待
case3:TR0=0;TR1=0;/*led_run=1;led_await=1;led_stop=0*/;aa=1;bb=1;cc=0;break; //暫停
}
if(K4==0) //跳出運行
{
delay(50);
if(K4==0)
{
while(K4==0);
jump_out=1;
}
}
}
void delay(uchart)
{
while(--t);
}
void set(uchar x)
{
command=0x80+x;
wr_com(command);
}
void set1(uchar y)
{
command=0xc0+y;
wr_com(command);
}
void key1() //初始調節按鍵
{
if(K==0)
{
delay(100);
if(K==0)
{
while(K==0);
v++;
if(v==4) v=0;
}
}
switch(v)
{
case 0:break;
case1:if(K1==0){while(K1==0);value1=value1+10;}break;
case 2:if(K1==0){while(K1==0);value2=value2+5;}break;
case 3:if(K1==0){while(K1==0);value3=value3+5;}break; }
if(K2==0)
{
delay(50);
if(K2==0)
{
while(K2==0);
jump_in=1;
}
}
}
void mode( )interrupt 0 //外部中斷0控制選位
{
m++;
}
|