机电信息工程学院
单片机系统课程设计报告
完成日期:20##年5月31日
目 录
一、设计任务和性能指标... 2
1.1设计任务... 2
1.2性能指标... 2
二.设计方案... 2
三.系统硬件设计... 3
3.1单片机最小系统.. 3
3.2键盘接口电路... 3
3.3数码管显示电路... 4
3.4错误报警电路... 5
四、系统软件设计... 6
4.1键盘扫描子程序设计... 6
4.2移位子程序及结果计算子程序设计... 10
4.3显示子程序设计... 12
4.4主程序设计... 13
五、调试及性能分析... 13
5.1调试步骤... 13
5.2性能分析... 14
六、心得体会... 14
参考文献... 14
附录1 系统硬件电路图... 15
附录2 程序清单... 16
利用单片机及外围接口电路(键盘接口和显示接口电路)设计制作一个计算器,用LED显示计算数值及结果。
要求用Protel 画出系统的电路原理图(要求以最少组件,实现系统设计所要求的功能),印刷电路板(要求布局合理,线路清晰),绘出程序流程图,并给出程序清单(要求思路清晰,尽量简洁,主程序和子程序分开,使程序有较强的可读性)。
1. 加法:四位加法,计算结果若超过四位则显示计算错误
2. 减法:四位减法,计算结果若小于零则显示计算错误
3. 乘法:个位数乘法
4. 除法:整数除法
5. 有清零功能,计算错误报警
按照系统设计的功能的要求,初步确定设计系统由主控模块、显示模块、错误报警模块、键扫描接口电路共四个模块组成,电路系统构成框图如图1.1所示。主控芯片使用51系列AT89C52单片机,采用高性能的静态80C51设计,由先进工艺制造,并带有非易失性Flash程序存储器。它是一种高性能、低功耗的8位COMS微处理芯片,市场应用最多。
键盘电路采用4*4矩阵键盘电路。
显示模块采用4枚共阳极数码管和74ls273锁存芯片构成等器件构成。
错误报警电路采用5V蜂鸣器。
单片机最小系统就是支持主芯片正常工作的最小电路部分,包括主控芯片、复位电路和晶振电路。
主控芯片选取STC89C52RC芯片,因其具有良好的性能及稳定性,价格便宜应用方便。
晶振选取11.0592MHz,晶振旁电容选取30pF。
采用按键复位电路,电阻分别选取100Ω和10K,电容选取10μF。
以下为单片机最小系统硬件电路图。
单片机最小系统硬件电路
计算器所需按键有:
数字键:’1’,’2’,’3’,’4’,’5’,’6’,’7’,’8’,’9’,’0’
功能键:’+’, ’-‘ , ’*’, ’/ ’ , ’ = ’, ’ C( 清零)’
共计16个按键,采用4*4矩阵键盘,键盘的行和列之间都有公共端相连,四行和四列的8个公共端分别接P1.0~P1.7,这样扫描P1口就可以完成对矩阵键盘的扫描,通过对16个按键进行编码,从而得到键盘的口地址,对比P1口德扫描结果和各按键的地址,我们就可以得到是哪个键按下,从而完成键盘的功能。
以下为键盘接口电路的硬件电路图
键盘接口电路
采用4位数码管对计算数据和结果的显示,这里选取共阳数码管,利用NPN三极管对数码管进行驱动,为了节省I/O资源,采取动态显示的方法来显示计算数据及结果。
利用74273锁存器来实现数码管的动态显示,P0口输出显示值,P2.4为段选口,控制273锁存器的时钟引脚,从而得到对数码管输入数据的控制。
P2.0~P2.3用来作为位选端,控制哪几位数码管进行显示。
以下为数码显示电路的硬件电路图
数码显示电路硬件电路图
错误报警电路就是在计算结果出现错误时或输入数据出现错误时,发出声音警报,提示使用者错误出现。
这里就采用5V蜂鸣器作为报警设备,利用PNP三极管对蜂鸣器进行驱动,有P2.5对其进行控制,这样在出现错误的同时用P2.5输出低,就可以使蜂鸣器工作,完成报警任务。
以下为报警电路硬件电路图
报警电路硬件电路图
系统整体硬件电路图见附录一
要进行数据的计算就必须先进行数据的输入,也就必须确定按键输入的数值是什么,这就需要对键盘进行扫描,从而确定究竟是哪个键按下。
对于键盘的扫描,既可以用行扫描也可以用列扫描,这里采用行扫描的方法来完成对键盘的扫描。
行扫描就是逐行扫描键盘,看是哪一行有键按下,再通过返回的键码来确定究竟是哪个按键按下。如对第一行扫描就令P1.0为低,P1口其余为高,这样若第一行有键按下,则P1口的值就会由0xfe变为其他值,再由这个值来确定是哪个键按下。
以下为键盘扫描子程序的程序清单。
uchar keyscan()
{
int i;
P1=0xfe;
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
delay(5);
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
temp=P1;
switch(temp)
{
case 0xee:{rdat++;num=1;left(rdat,num);}
break;
case 0xde:{rdat++;num=2;left(rdat,num);}
break;
case 0xbe:{rdat++;num=3;left(rdat,num);}
break;
case 0x7e:{rdat++;num=4;left(rdat,num);}
break;
}
while(temp!=0xf0)
{
temp=P1;
temp=temp&0xf0;
}
}
}
P1=0xfd;
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
delay(5);
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
temp=P1;
switch(temp)
{
case 0xed:{rdat++;num=5;left(rdat,num);}
break;
case 0xdd:{rdat++;num=6;left(rdat,num);}
break;
case 0xbd:{rdat++;num=7;left(rdat,num);}
break;
case 0x7d:{rdat++;num=8;left(rdat,num);}
break;
}
while(temp!=0xf0)
{
temp=P1;
temp=temp&0xf0;
}
}
}
P1=0xfb;
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
delay(5);
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
temp=P1;
switch(temp)
{
case 0xeb:{rdat++;num=9;left(rdat,num);}
break;
case 0xdb:{rdat++;num=10;left(rdat,num);}
break;
case 0xbb:{equ();}
break;
case 0x7b:{rdat=0;add=0;subb=0;mul=0;div=0;
result=0;
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
}
while(temp!=0xf0)
{
temp=P1;
temp=temp&0xf0;
}
}
}
P1=0xf7;
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
delay(5);
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
temp=P1;
switch(temp)
{
case 0xe7:{rdat=0;add=1;subb=0; mul=0;div=0;
for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
result=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
case 0xd7:{rdat=0;add=0;subb=1; mul=0;div=0;
for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
result=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
case 0xb7:{rdat=0;add=0;subb=0; mul=1;div=0;
for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
result=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
case 0x77:{rdat=0;add=0;subb=0; mul=0;div=1;
for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
result=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
}
while(temp!=0xf0)
{
temp=P1;
temp=temp&0xf0;
}
}
}
return num;
}
输入数据要存储在一四位数组内,而我们键入的值是数据的高位,后键入的值是低位,这样我们就需要在输入低位数值时将高位数值从数组的低位移向数组的高位,这就是编写移位子程序的目的。
对于结果计算子程序,包含加、减、乘、除四种运算。以加法运算为例,各种运算各有其标志位来代表计算类型,当加法标志位add=1是,就将输入的两个数据按照加法进行计算。
首先将数组内的数按照对应的位关系,将其转化为一个十进制数,这样我们就得到了加速和被加数这样俩个十进制数,从而我们就可以简单的将两个数进行相加,结果就是我们所求的数值。但这个数值不能直接显示到数码管上,我们还要对其进行处理,使其变为对应进位的四个数存入数组内,以便显示。既通过对结果数值分别除以1000、100、10和对10取余,得到我们想要的四个数,送显示子程序显示。其余减、乘、除的计算方法与加法的计算方法一样,这里不再累述。
以下为移位子程序和结果计算子程序的程序清单。
void left(uchar rx,uchar date)
{
switch(rx)
{
case 1:dat[0]=date;break;
case 2:dat[1]=dat[0],dat[0]=date;break;
case 3:dat[2]=dat[1],dat[1]=dat[0],dat[0]=date;break;
case 4:dat[3]=dat[2],dat[2]=dat[1],dat[1]=dat[0],dat[0]=date;break;
}
}
void equ()
{
int i,j,k;
long int s;
if(add==1){for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
s=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
result=result+s;add=0;}
if(subb==1){for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
s=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
result=result-s;subb=0;}
if(mul==1){for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
s=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
result=result*s;mul=0;
}
if(div==1){for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
s=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
result=result/s;div=0;
}
If(result>9999){dat[0]=11;dat[3]=dat[2]=dat[1]=0;}
if(result<=9999)
{
dat[0]=result%10;
dat[1]=(result/10)%10;
dat[2]=(result/100)%10;
dat[3]=(result/1000)%10;
}
for(j=3;j>0;j--)
{ if(dat[j]>0)
{
for(k=j-1;k>=0;k--)
{
if(dat[k]==0){dat[k]=10;}
}
}
}
if(dat[0]==0){dat[0]=10;}
}
从始至终无论是输入的计算数据,还是计算后的结果值。都存储在同一数组dat[ ]中,这样我们只要在显示时一直调用dat[ ]中的值,就能正确的显示数据。
以下为显示子程序的程序清单。
void display()
{
uchar aa;
keyscan();
P2=0x07;
aa=dat[0];
P0=table[aa];
P2=0x27;
delay(3);
P2=0x0b;
aa=dat[1];
P0=table[aa];
P2=0x2b;
delay(3);
P2=0x0d;
aa=dat[2];
P0=table[aa];
P2=0x2d;
delay(3);
P2=0x0e;
aa=dat[3];
P0=table[aa];
P2=0x2e;
delay(3);
}
主程序既把以上各子程序串连成一个整体,使整个程序循环运行。而在以上程序中也已经加入了个程序之间的连接点,首先进入程序后就立即进入显示子程序,而显示子程序内又调用键盘扫描子程序,若有键按下,则会跳转到移位子程序和结果计算子程序进行相应的处理。通过计算或移位后,数组内的值发生改变,显示的值也会同时发生改变。之后再进行键盘扫描,如此反复运行,就构成了程序的整体。
以下为主程序清单。
void main()
{
num=0;
while(1)
{
display();
}
}
整体程序清单见附录二。
在焊接好器件后,先不要将芯片插在芯片座上,要先验证先板上电源是否好用,有无短路等。接上USB电源,用万用表测量个芯片座对应电源和地之间的电压值,观察电压值是否正常。一切正常后方可将芯片插入芯片座,以继续测试其他功能。
将芯片插上后,对各个模块进行调试,按键是否工作正常,数码管是否显示正常等。编写相关部分的测试程序对其进行测试。
各部分硬件检测无误后,下载程序进行整体调试,一切正常后,结束调试过程。
在具体调试时首先遇到的问题是程序无法下载进入单片机,通过将电路板接线与原理电路图接线的对比发现,串口芯片与单片机连接的输入,输出接反,重新用铜线连接后,依然无法下载程序。后找到原因是由于下载串口与设计封装不符,用相对应的下载线可以下载。
成功下载程序后,发现数码管显示不正确,查看后发现有先没有连接,可能是制板时漏印,连接后显示正常。
对于计算器的性能,主要的衡量指标就在于计算的精度,本次制作的计算器性能情况如下:
加法运算:四位加法运算,和值不超过9999,若超过上限,则显示错误提示E,蜂鸣器报警提示。
减法运算:四位减法运算,若结果为负,对其取绝对值。
乘法运算:积不超过9999的乘法运算,若超出上限,显示错误提示E,蜂鸣器报警提示。
除法运算:整数除法,既计算结果为整数,若除数为零,则显示错误提示E,蜂鸣器报警提示。
通过对实际性能的分析,可以得到本次设计满足设计的要求。
通过本次课程设计我真正的自己完成了对给定要求系统的硬件设计、电路设计、电路板设计、软件设计以及对成品的调试过程。从整个过程中学习到了很多方面的知识,了解到以往学习中自己知识在某方面的不足之处,是对以往学习科目的一种贯穿和承接,从而能更好的认识和学习,也对将来从事工作大有裨益。
从本次课设中我也看到了自身的很多不足之处,对知识的掌握不够扎实,有一知半解的现象。有时做事不够稳定,过于毛躁,不能平心静气的去分析所遇到的问题和错误。这在以后的工作和生活中是不可取的,通过对自身问题的认识与改正相信再遇到同样问题时会更好的解决。以后的设计实验也会更好的完成。
[1] 徐维祥、刘旭敏. 单片微型机原理及应用. 大连:大连理工大学出版社,1996
[2] 李光飞、楼然苗、胡佳文、谢象佐. 单片机课程设计与实例指导. 北京: 北京航空航天大学出版社,2004
[3] 余永权. 89系列FLASH单片机原理及应用. 北京:电子工业出版社,2002
[4] 李群芳,黄建. 单片机微型计算机与接口技术. 北京:电子工业出版社,2001
[5] 楼然苗、李光飞. 51系列单片机设计实例. 北京:北京航空航天大学出版社,2003
#include<reg52.h>
#define uint unsigned int
#define uchar unsigned char
sbit dula=P2^6;
sbit wela=P2^7;
sbit key1=P3^4;
uchar code table[]={
0xff,0xf9,0xa4,0xb0,
0x99,0x92,0x82,0xf8,
0x80,0x90,0xc0,0x86};
uchar dat[]={10,0,0,0,0};
uchar s[],a[];
uchar num,temp,num1,rdat,add,subb,mul,div;
unsigned long int result;
void left(uchar rx,uchar date);
uchar keyscan();
void equ();
void delay(uint z)
{
uint x,y;
for(x=z;x>0;x--)
for(y=110;y>0;y--);
}
void display();
void main()
{
num=0;
while(1)
{
display();
}
}
void display()
{
uchar aa;
keyscan();
P2=0x07;
aa=dat[0];
P0=table[aa];
P2=0x27;
delay(3);
P2=0x0b;
aa=dat[1];
P0=table[aa];
P2=0x2b;
delay(3);
P2=0x0d;
aa=dat[2];
P0=table[aa];
P2=0x2d;
delay(3);
P2=0x0e;
aa=dat[3];
P0=table[aa];
P2=0x2e;
delay(3);
}
uchar keyscan()
{
int i;
P1=0xfe;
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
delay(5);
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
temp=P1;
switch(temp)
{
case 0xee:{rdat++;num=1;left(rdat,num);}
break;
case 0xde:{rdat++;num=2;left(rdat,num);}
break;
case 0xbe:{rdat++;num=3;left(rdat,num);}
break;
case 0x7e:{rdat++;num=4;left(rdat,num);}
break;
}
while(temp!=0xf0)
{
temp=P1;
temp=temp&0xf0;
}
}
}
P1=0xfd;
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
delay(5);
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
temp=P1;
switch(temp)
{
case 0xed:{rdat++;num=5;left(rdat,num);}
break;
case 0xdd:{rdat++;num=6;left(rdat,num);}
break;
case 0xbd:{rdat++;num=7;left(rdat,num);}
break;
case 0x7d:{rdat++;num=8;left(rdat,num);}
break;
}
while(temp!=0xf0)
{
temp=P1;
temp=temp&0xf0;
}
}
}
P1=0xfb;
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
delay(5);
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
temp=P1;
switch(temp)
{
case 0xeb:{rdat++;num=9;left(rdat,num);}
break;
case 0xdb:{rdat++;num=10;left(rdat,num);}
break;
case 0xbb:{equ();}
break;
case 0x7b:{rdat=0;add=0;subb=0;mul=0;div=0;
result=0;
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
}
while(temp!=0xf0)
{
temp=P1;
temp=temp&0xf0;
}
}
}
P1=0xf7;
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
delay(5);
temp=P1;
temp=temp&0xf0;
while(temp!=0xf0)
{
temp=P1;
switch(temp)
{
case 0xe7:{rdat=0;add=1;subb=0; mul=0;div=0;
for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
result=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
case 0xd7:{rdat=0;add=0;subb=1; mul=0;div=0;
for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
result=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
case 0xb7:{rdat=0;add=0;subb=0; mul=1;div=0;
for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
result=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
case 0x77:{rdat=0;add=0;subb=0; mul=0;div=1;
for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
result=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
dat[0]=10;dat[1]=dat[2]=dat[3]=0;
}
break;
}
while(temp!=0xf0)
{
temp=P1;
temp=temp&0xf0;
}
}
}
return num;
}
void left(uchar rx,uchar date)
{
switch(rx)
{
case 1:dat[0]=date;break;
case 2:dat[1]=dat[0],dat[0]=date;break;
case 3:dat[2]=dat[1],dat[1]=dat[0],dat[0]=date;break;
case 4:dat[3]=dat[2],dat[2]=dat[1],dat[1]=dat[0],dat[0]=date;break;
}
}
void equ()
{
int i,j,k;
long int s;
if(add==1){for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
s=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
result=result+s;add=0;}
if(subb==1){for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
s=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
if(s>result) {result=s-result;}
else
result=result-s;subb=0;}
if(mul==1){for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
s=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
result=result*s;mul=0;
}
if(div==1){for(i=0;i<5;i++){
if(dat[i]==10) {dat[i]=0;}
}
s=dat[0]+10*dat[1]+100*dat[2]+1000*dat[3];
if(s==0) result=10000;
else
result=result/s;div=0;
}
if(result>9999){dat[0]=11;dat[3]=dat[2]=dat[1]=0;}
if(result<=9999)
{
dat[0]=result%10;
dat[1]=(result/10)%10;
dat[2]=(result/100)%10;
dat[3]=(result/1000)%10;
}
for(j=3;j>0;j--)
{ if(dat[j]>0)
{
for(k=j-1;k>=0;k--)
{
if(dat[k]==0){dat[k]=10;}
}
}
}
if(dat[0]==0){dat[0]=10;}
}
中南大学本科生课程设计(实践)任务书、设计报告(大学计算机基础)计算机实践过程与体会题目学生姓名指导教师学院专业班级学生学号祁彦翔…
学生成绩管理系统课程设计报告课程设计报告格式一功能描述要求学生对其所完成的课程设计给出各个功能模块详细的描述包括文字说明和图形说明…
北京科技大学计算机应用实践报告机房名称时间1300至1625学院自动化学院专业班级姓名学号指导教师成绩20xx年8月说明一计算机应…
目录第一章实训任务概述111实训目的112实训任务1第二章课程设计结果221指令的执行流程222位扩展和字扩展523设计计算机运算…
计算机综合课程设计报告小组编号06课题名称趣味贪吃蛇小组成员20xx年10月计算机综合课程设计报告目录目录11项目背景112系统目…
中南大学本科生课程设计(实践)任务书、设计报告(大学计算机基础)计算机实践过程与体会题目学生姓名指导教师学院专业班级学生学号祁彦翔…
武汉理工大学专业课程设计2课程设计说明书目录1基本功能描述12设计思路13软件设计431设计步骤44结论与心得体会127附设计中的…
MFC计算器课程设计报告计算机应用3班黄锦湫罗洁饶益指导老师蒋鹏20xx315一题目利用MFC框架编写简易计算器要求使用MFC框架…
MFC表达式计算器课程设计报告班级5班学生姓名及学号丁健华020xx507李晓奇020xx513吴继超020xx516完成日期20…
计算机网络课程设计报告一.课程设计的题目、目的及要求.......................................…