4位数加法器设计报告.docx

上传人:b****3 文档编号:5425398 上传时间:2022-12-16 格式:DOCX 页数:16 大小:128.68KB
下载 相关 举报
4位数加法器设计报告.docx_第1页
第1页 / 共16页
4位数加法器设计报告.docx_第2页
第2页 / 共16页
4位数加法器设计报告.docx_第3页
第3页 / 共16页
4位数加法器设计报告.docx_第4页
第4页 / 共16页
4位数加法器设计报告.docx_第5页
第5页 / 共16页
点击查看更多>>
下载资源
资源描述

4位数加法器设计报告.docx

《4位数加法器设计报告.docx》由会员分享,可在线阅读,更多相关《4位数加法器设计报告.docx(16页珍藏版)》请在冰豆网上搜索。

4位数加法器设计报告.docx

4位数加法器设计报告

4位数加法器设计报告

(设计者:

SKY123GOING)

一、设计任务和要求

1.1、任务描述:

1、系统通过4×4的矩阵键盘输入数字及运算符;

2、可以进行4位十进制数以内的加法运算,如果计算结果超过4位十进制数,则屏幕显示E;

3、可以进行加法以外的计算(乘、除、减);

4、创新功能。

1.2、任务要求:

1、理解任务书要求,明确分工,查找相关资料,制定系统方案;

2、论证系统设计方案,运用Proteus等软件绘制电路原理图;

3、根据硬件电路,确定算法,设计程序框图,编写程序代码;

4、误差分析与改进,完成设计报告。

二、方案论证

2.1、适用矩阵键盘控制作为输入电路,电路和软件稍微复杂,但是相比用独立按键,可节省I/O口,其原理图如2.1所示:

 

图2.1矩阵键盘控制电路

 

2.2、采用LED数码管显示,数码管图如图2.2.1所示:

图2.2.1LED数码管

一、电路基本单元电路设计

本电路的总体的工作框图如下所示:

 

下图则是加法器电路的原理图:

3.1、主控模块

该设计的核心控制电路是AT89C52单片机。

AT89C51是一种带4K字节FLASH存储器(FPEROM—FlashProgrammableandErasableReadOnlyMemory)的低电压、高性能CMOS8位微处理器,俗称单片机。

AT89C2051是一种带2K字节闪存可编程可擦除只读存储器的单片机。

单片机的可擦除只读存储器可以反复擦除1000次。

该器件采用ATMEL高密度非易失存储器制造技术制造,与工业标准的MCS-51指令集和输出管脚相兼容。

由于将多功能8位CPU和闪烁存储器组合在单个芯片中,ATMEL的AT89C51是一种高效微控制器,AT89C2051是它的一种精简版本。

AT89C单片机为很多嵌入式控制系统提供了一种灵活性高且价廉的方案。

其引脚图以及工作原理如下:

AT89C51芯片模型

3.1.1、主要功能特性

(1)4K字节可编程闪烁存储器。

(2)32个双向I/O口;128×8位内部RAM。

(3)2个16位可编程定时/计数器中断,时钟频率0-24MHz。

(4)可编程串行通道。

(5)5个中断源。

(6)2个读写中断口线。

(7)低功耗的闲置和掉电模式。

(8)片内振荡器和时钟电路。

3.1.2、AT89C51的引脚介绍

89C51单片机多采用40只引脚的双列直插封装(DIP)方式,下面分别简单介绍。

(1)电源引脚

电源引脚接入单片机的工作电源。

Vcc(40引脚):

+5V电源。

GND(20引脚):

接地。

(2)时钟引脚

XTAL1(19引脚):

片内振荡器反相放大器和时钟发生器电路的输入端。

XTAL2(20引脚):

片内振荡器反相放大器的输出端。

电源接入方式

(3)复位RST(9引脚)

在振荡器运行时,有两个机器周期(24个振荡周期)以上的高电平出现在此引脚时,将使单片机复位,只要这个脚保持高电平,51芯片便循环复位。

(4)

/Vpp(31引脚)

为外部程序存储器访问允许控制端。

当它为高电平时,单片机读片内程序存储器,在PC值超过0FFFH后将自动转向外部程序存储器。

当它为低电平时,只限定在外部程序存储器,地址为0000H~FFFFH。

Vpp为该引脚的第二功能,为编程电压输入端。

(5)ALE/

(30引脚)

ALE为低八位地址锁存允许信号。

在系统扩展时,ALE的负跳沿江P0口发出的第八位地址锁存在外接的地址锁存器,然后再作为数据端口。

为该引脚的第二功能,在对片外存储器编程时,此引脚为编程脉冲输入端。

(6)

(29引脚)

片外程序存储器的读选通信号。

在单片机读片外程序存储器时,此引脚输出脉冲的负跳沿作为读片外程序存储器的选通信号。

(7)pin39-pin32为P0.0-P0.7输入输出脚,称为P0口。

P0是一个8位漏极开路型双向I/O口。

内部不带上拉电阻,当外接上拉电阻时,P0口能以吸收电流的方式驱动八个LSTTL负载电路。

通常在使用时外接上拉电阻,用来驱动多个数码管。

在访问外部程序和外部数据存储器时,P0口是分时转换的地址(低8位)/数据总线,不需要外接上拉电阻。

(8)Pin1-Pin8为P1.0-P1.7输入输出脚,称为P1口,是一个带内部上拉电阻的8位双向I/0口。

P1口能驱动4个LSTTL负载。

(9)Pin21-Pin28为P2.0-P2.7输入输出脚,称为P2口。

P2口是一个带内部上拉电阻的8位双向I/O口,P2口能驱动4个LSTTL负载。

端口置1时,内部上拉电阻将端口拉到高电平,作输入用。

对内部Flash程序存储器编程时,接收高8位地址和控制信息。

在访问外部程序和16位外部数据存储器时,P2口送出高8位地址。

而在访问8位地址的外部数据存储器时其引脚上的内容在此期间不会改变。

(10)Pin10-Pin17为P3.0-P3.7输入输出脚,称为P3口。

P3口是一个带内部上拉电阻的8位双向I/O口,P2口能驱动4个LSTTL负载,这8个引脚还用于专门的第二功能。

端口置1时,内部上拉电阻将端口拉到高电平,作输入用。

对内部Flash程序存储器编程时,接控制信息。

3.2、显示模块

该电路的显示模块采用共阴极数码管显示。

共阴极数码管的位选采用低电平,而段选采用高电平控制。

LED数码管十六位进数的字形码如表3.2.1所示,而显示模块控制电路如3.2.2所示:

字型

共阳极代码

共阴极代码

字型

共阳极代码

共阴极代码

0

C0H

3FH

9

90H

6FH

1

F9H

06H

A

88H

77H

2

A4H

5BH

B

83H

7CH

3

B0H

4FH

C

C6H

39H

4

99H

66H

D

A1H

5EH

5

92H

6DH

E

86H

79H

6

82H

7DH

F

8EH

71H

7

F8H

07H

FFH

00H

8

80H

7FH

表3.2.1LED数码管十六位进数的字形码

图3.2.2显示模块控制电路

3.3、操作模块

该操作模块的实际操作如下图所示:

 

图3.3.1矩阵键盘实际操作图

四、程序设计

本作品实现的功能全部是由C语言程序编写实现。

通过程序的编写使简单的器件实现丰富的功能。

如下所示是主程序流程图

 

主程序流程图

 

五、系统调试及结果

本设计应用Proteus6及KEIL51软件,首先根据自己设计的电路图用Proteus6软件画出电路模型,关于这个软件的使用通过查一些资料和自己的摸索学习;然后我们用KEIL51软件对所编写的程序进行编译、链接,如果没有错误和警告便可生成程序的hex文件,将此文件加到电路图上使软硬件结合运行。

下图为运行状况图,为首先输入10+990,再加上9999,最后每个数码管上为E,因为为了满足实验要求,当结果超过了9999后显示E.

刚开始为0

输入10

再加上990得到了1000

最后显示E

 

五、设计个人总结

经过一周的努力终于设计成功,LED数码管的显示结果与理想所保持的一样。

在这一周的设计中,完成这个简易的计算器,虽然精度不高,但是对于一般的计算,是绰绰有余。

起初,电路的设计对我们这些人来说,是一大障碍,通过查找资料,发现一个简易的计算器,似乎没有那么难,只要有心,定能成功。

后来数码管的显示一直困扰着我,但随着慢慢地了解它,在相关资料帮助之下,终于明白如何用C语言写程序去控制数码管的显示。

加之以前学过一些计算机方面的C语言,那并不是针对单片机C语言的,虽有相似之处,但差之甚大,如今已经了逐步地了解,相信若坚持下去,定能学好这一门课程。

通过此次设计课程,使我受益匪浅。

终于更深刻地体会到实践是唯一的真理。

 

附录(程序):

/**********通过用单片机MCU与矩阵及数码管的相互连接构成一个简易的数计算器--------可以进行加减乘除********/

#include

#defineucharunsignedchar

longFirst,End;//定义全局变量

//此表为LED的字模,共阴数码管0-9-

unsignedcharcodeDisp_Tab[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};//段码控制

//此表为8个数码管位选控制,共阴数码管1-8个-

unsignedcharcodedispbit[6]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf};//位选控制查表的方法控制

voiddelay(intn)/***延时程序***/

{inti,j;

for(i=0;i

{for(j=0;j<50;j++)

;}

}

longadd(longx,longy)/***加法程序***/

{longz;

z=x+y;

return(z);

}

longsub(longx,longy)/***减法程序***/

{longz;

if(x>=y)

z=x-y;

else

{z=y-x;

z=z+100000;}/***最高位用1表示负数***/

return(z);

}

longmul(longx,longy)/***乘法程序***/

{longz;

z=x*y;

return(z);

}

longdiv(longx,longy)/***除法程序***/

{longz;

z=x/y;

return(z);

}

ucharkbscan(void)/***键盘扫描程序***/

{

ucharsccode;

P0=0xf0;

if((P0&0xf0)!

=0xf0)//发全0行扫描码,列线输入

{delay(222);//延时去抖

if((P0&0xf0)!

=0xf0)

{sccode=0xfe;//逐行扫描初值

while((sccode&0x10)!

=0)

{P0=sccode;//输出行扫描码

if((P0&0xf0)!

=0xf0)

{

return(P0);}//如果检测到有键按下,返回键值

else

sccode=(sccode<<1)|0x01;//行扫描码左移一位

}

}

}

return(0);//无键按下,返回值为0

}

 

voiddisplay(void)/***显示程序***/

{inti;

uchardatanum[6];

num[0]=First/100000%10;//十万

num[1]=First/10000%10;//万

num[2]=First/1000%10;//千位

num[3]=First/100%10;//百位

num[4]=First/10%10;//十位

num[5]=First%10;//个位

for(i=5;i>=0;i--)

{P2=dispbit[i];//位选输出

P1=Disp_Tab[num[i]];//数据输出

delay

(2);//此延时必不可少?

}

}

voidmain(void)/***主程序***/

{intk,n;

ucharf,g,key,gn1,i;

n=0;

f=0;

//P1=0;//初始时指示灯灭

while

(1)//不断查询是否有按键动作

{key=kbscan();//获取返回键值

if(key!

=0)

{

switch(key)//译码,将对应按键返回值转换为相应数值

{

case0xee:

k=0;break;//0

case0xde:

k=1;break;//1

case0xbe:

k=2;break;//2

case0x7e:

k=3;break;//3

case0xed:

k=4;break;//4

case0xdd:

k=5;break;//5

case0xbd:

k=6;break;//6

case0x7d:

k=7;break;//7

case0xeb:

k=8;break;//8

case0xdb:

k=9;break;//9

case0xbb:

k=10;First=0;End=0;f=0;break;//清除

case0x7b:

k=11;break;//等于

case0xe7:

k=12;f=1;break;//加

case0xd7:

k=13;f=2;break;//减

case0xb7:

k=14;f=3;break;//乘

case0x77:

k=15;f=4;break;//除

}

//P1=1;

delay(280);//有按键时,指示灯的显示时间

//P1=0;//按键指示灭

if(k<10)//为数字键时(0-9)

{

if(f!

=0)//为数字键时,如果已经有功能键按下

{

n++;//记录数字键所按次数

gn1=0;//清除标志,再次为功能键时进行运算

g=f;//保存运算标志

if(n==1)//输入为各位数时,直接赋值

First=k;

elseif(n>1)//输入为多位数时,将它转化为10进制的多位数

First=First*10+k;

}

else//如果没有功能键按下

{

n++;

gn1=1;//定义标志,当下一次为功能键时,停止数据输入

if(n==1)

First=k;

elseif(n>1)

First=First*10+k;

End=First;//将第一个数保存

}

}

elseif(k>11)//为功能键时(+-*/)

{

if(gn1==1)//前一次数字键之后为功能键时

{

n=0;//清除计数标志

}

else//如果再次输入功能键,则进行运算

{

n=0;//清除计数标志

switch(g)

{

case1:

First=add(End,First);break;

case2:

First=sub(End,First);break;

case3:

First=mul(End,First);break;

case4:

First=div(End,First);break;

}

}

End=First;//保存本次结果

}

elseif(k==11)//为等于号时(=)

{

n=0;

gn1=1;//接着输入为功能键时可以继续运算

switch(g)

{

case1:

First=add(End,First);break;

case2:

First=sub(End,First);break;

case3:

First=mul(End,First);break;

case4:

First=div(End,First);break;

}

End=First;//保存最终运算结果

f=0;//清除运算标志

if(End>9999)/***当结果大于9999时,显示E*****/

{

P1=0XF9;//显示E

for(i=5;i>=0;i--)

{

P2=dispbit[i];//位选输出

//P1=Disp_Tab[num[i]];//数据输出

delay

(1);//此延时必不可少?

}

}

}

}

display();//调用显示程序

}

}

 

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 外语学习 > 英语考试

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1