C51单片机课设报告进制转换.docx

上传人:b****5 文档编号:4392663 上传时间:2022-12-01 格式:DOCX 页数:21 大小:802.05KB
下载 相关 举报
C51单片机课设报告进制转换.docx_第1页
第1页 / 共21页
C51单片机课设报告进制转换.docx_第2页
第2页 / 共21页
C51单片机课设报告进制转换.docx_第3页
第3页 / 共21页
C51单片机课设报告进制转换.docx_第4页
第4页 / 共21页
C51单片机课设报告进制转换.docx_第5页
第5页 / 共21页
点击查看更多>>
下载资源
资源描述

C51单片机课设报告进制转换.docx

《C51单片机课设报告进制转换.docx》由会员分享,可在线阅读,更多相关《C51单片机课设报告进制转换.docx(21页珍藏版)》请在冰豆网上搜索。

C51单片机课设报告进制转换.docx

C51单片机课设报告进制转换

单片机原理课程设计报告

题目:

51单片机实现进制转换

专业:

信息工程

班级:

信息101

学号:

1004020307

小组成员:

1004020103

指导教师:

吴叶兰

北京工商大学计算机与信息工程学院

题目:

51单片机实现进制转换

1、设计目的

1)熟悉51单片机的编程;

2)熟悉单片机开发的基本焊接;

3)熟悉单片机元件的使用方法;

4)熟悉C51的软件开发环境(编程软件Keil、烧录软件STC_ISP_v479

2、设计要求

1)按键输入数据,具有确定和清零功能;

2)1602液晶屏幕显示,具有显示输入输出和提示功能;

3)2,、8、10、16进制数可任意互相转化;

3、硬件电路设计(包括电路图及说明)

1)控制模块:

控制模块是由1块STC89C521个12MHz无源晶振、1个复位电路22uF电容、1个1k欧姆电阻、1个8位1k欧姆排阻组成。

STC89C52是STC公司生产的一种低功耗、高性能CMOS位微控制器,具有8K在系统可编程Flash存储器。

STC89C52使用经典的MCS-51内核,但做了很多的改进使得芯片具有传统51单片机不具备的功能。

在单芯片上,拥有灵巧的8位CPU和在系统可编程Flash,使

得STC89C52为众多嵌入式控制应用系统提供高灵活、超有效的解决方案。

具有以下标准功

能:

8k字节Flash,512字节RAM32位I/O口线,看门狗定时器,内置4KBEEPRQMIAX810复位电路,3个16位定时器/计数器,4个外部中断,一个7向量4级中断结构(兼容传统51的5向量2级中断结构),全双工串行口。

另外STC89C52可降至0Hz静态逻辑操作,支持2种软件可选择节电模式。

空闲模式下,CPU停止工作,允许RAM定时器/计数器、串口、中断继续工作。

掉电保护方式下,RAM内容被保存,振荡器被冻结,单片机一切工作停止,直到下一个中断或硬件复位为止。

最高运作频率35MHz6T/12T可选。

芯片实际选用的

STC89C5与protues模拟电路图中的AT89C52功能基本相同,区别在于烧录程序的方式。

2)显示模块:

显示模块由1块1602液晶显示屏、16根数据线组成。

1602液晶也叫1602字符型液晶,它是一种专门用来显示字母、数字、符号等的点阵型液晶模块。

它由若干个5X7或者5X11等点阵字符位组成,每个点阵字符位都可以显示一个字符,每位之间有一个点距的间隔,每行之间也有间隔,起到了字符间距和行间距的作用,正因为如此所以它不能很好地显示图形(用自定义CGRAM显示效果也不好)。

1602LCD是指显示的

内容为16X2,即可以显示两行,每行16个字符液晶模块(显示字符和数字)。

液晶显示屏际实际选用的是QC1602ATprotues模拟电路图中的LM016功能基本相同。

3)按键输入模块:

按键输入模块由1块4*4矩阵键盘、8根数据线组成、2个独立按键。

在键盘中按键数量较多时,为了减少I/O口的占用,通常将按键排列成矩阵形式,如图1所示。

在矩阵式键盘中,每条水平线和垂直线在交叉处不直接连通,而是通过一个按键加以连接。

这样,一个端口(如P1口)就可以构成4*4=16个按键,比之直接将端口线用于键盘多出了一倍,而且线数越多,区别越明显,比如再多加一条线就可以构成20键的键盘,而直

接用端口线则只能多出一键(9键)。

由此可见,在需要的键数比较多时,采用矩阵法来做键盘是合理的。

矩阵式结构的键盘显然比直接法要复杂一些,识别也要复杂一些,上图中,列线通过电阻接正电源,并将行线所接的单片机的I/O口作为输出端,而列线所接的I/O口则

作为输入。

这样,当按键没有按下时,所有的输入端都是高电平,代表无键按下。

行线输出是低电平,一旦有键按下,则输入线就会被拉低,这样,通过读入输入线的状态就可得知是否有键按下了。

实际电路中使用的是集成的1块4*4矩阵键盘,跟protues模拟电路图中的

16个独立按键相比体积更小,实现的功能相同。

4)全局设计:

通过STC89C52的P3.0-P3.7口读入4*4矩阵键盘的按键输入数据,STC89C52的P0.0-P0.7先接入8位1k欧姆排阻作上拉电阻,再接入1602液晶屏幕的数据口D01-D07,输出数据到1602液晶屏幕,液晶屏幕显示读到的数据。

P2.1-P2.3口分别接入1602的RSRWE控制读写。

P2.4-P2.5口分别接入1独立按键最为“确定”键和“清零”键。

STC89C52的RST接VCCt源,即上点就复位。

XTAL1,XTAL2接12MHz无源晶振。

4、软件设计(包括流程图及程序)

1)程序流程图:

2)程序代码:

#include//

#include

sbitRS=

:

P2A1;//

定义端口

sbitRW:

=卩2八2;

sbitEN=

:

卩2八3;

sbitQL=

P2A4;

//外接清零端口

sbitQD=

=P2A5;

//外接确定端口

sbitwela=P2A7;//

锁存器控制端定义

unsignedintqlbz=O;unsignedintqdbz=O;

#defineRS_CLRRS=O

#defineRS_SETRS=1

#defineRW_CLRRW=0

#defineRW_SETRW=1

#defineEN_CLREN=0

#defineEN_SETEN=1

#defineDataPortP0

#defineKeyPortP3

unsignedcharcode

asc_code[]={'0','1',2,3,'4','5','6',7',8,'9','A','B',C,'D','E','F'};〃

转换成液晶显示的字符

/*

uS延时函数

*/

voidDelayUs2x(unsignedchart)

{while(--t);

}

/*

mS延时函数

*/

voidDelayMs(unsignedchart)

{

while(t--)

{

//大致延时1mS

DelayUs2x(245);

DelayUs2x(245);

}

}

/*

判忙函数

*/

bitLCD_Check_Busy(void)

{

DataPort=OxFF;

RS_CLR;

RW_SET;

EN_CLR;

_nop_();

EN_SET;

return(bit)(DataPort&0x80);

}

/*

写入命令函数

*/

voidLCD_Write_Com(unsignedcharcom){

while(LCD_Check_Busy());//忙则等待

RS_CLR;

RW_CLR;

EN_SET;

DataPort=com;

_nop_();

EN_CLR;

}

/*

写入数据函数

----*/

voidLCD_Write_Data(unsignedcharData)

{

while(LCD_Check_Busy());//

忙则等待

RS_SET;

RW_CLR;

EN_SET;

DataPort=Data;

_nop_();

EN_CLR;

}

/*

清屏函数

----*/

voidLCD_Clear(void)

{

LCD_Write_Com(0x01);

DelayMs(5);

}

/*

写入字符串函数

----*/

voidLCD_Write_String(unsignedcharx,unsignedchary,unsignedchar*s)

{

if(y==0)

LCD_Write_Com(0x80+x);

}

else

{

LCD_Write_Com(0xC0+x);

}

while(*s)

{

LCD_Write_Data(*s);

s++;

}

}

/*

写入字符函数

*/

voidLCD_Write_Char(unsignedcharx,unsignedchary,unsignedcharData){

if(y==0)

{

LCD_Write_Com(0x80+x);

}

else

{

LCD_Write_Com(0xC0+x);

}

LCD_Write_Data(Data);

}

/*

初始化函数

*/

voidLCD」nit(void)

{

LCD_Write_Com(0x38);/*显示模式设置*/

DelayMs(5);

LCD_Write_Com(0x38);

DelayMs(5);

LCD_Write_Com(0x38);

DelayMs(5);

LCD_Write_Com(0x38);

LCD_Write_Com(0x08);

/*

显示关闭*/

LCD_Write_Com(0x01);

/*

显示清屏*/

LCD_Write_Com(0x06);

/*

显示光标移动设置*/

DelayMs(5);

LCD_Write_Com(0x0C);

/*

显示开及光标设置*/

}

/*

按键扫描函数0,返回小键盘扫描键值

*/

unsignedcharKeyScan(void)//

键盘扫描函数,使用行列反转扫描法

 

行列值中间变量行线输出全为0读入列线值

unsignedcharcord_h,cord_l;〃

KeyPort=0x0f;//

cord_h=KeyPort&0x0f;//

if(cord_h!

=0x0f)//

先检测有无按键按下

{

去抖

读入列线值

输出当前列线值

读入行线值

DelayMs(100);//if((KeyPort&0x0f)!

=0x0f)

{

cord_h=KeyPort&0x0f;//KeyPort=cord_h|0xf0;//cord」=KeyPort&0xf0;//

while((KeyPort&0xf0)!

=0xf0);〃等待松开并输出

return(cord_h+cord_l);〃键盘最后组合码值

}

}return(0xff);//返回该值

}

/*

按键扫描函数1,返回确定键扫描键值

*/

unsignedcharKeyScanQD(void)//键盘扫描函数

{

if(QD!

=1)//

先检测有无按键按下

{

DelayMs(100);

//去抖

if(QD!

=1)

{

while(QD==0);//

等待松开并输出

}

qdbz=1;

}return(0);//

返回该值

}

/*

按键扫描函数2,返回清零键扫描键值

*/

unsignedcharKeyScanQL(void)//键盘扫描函数

{

if(QL!

=1)//

先检测有无按键按下

{

DelayMs(100);

//去抖

if(QL!

=1)

{

while(QL==0);//

等待松开并输出

}

qlbz=1;

}return(0);//

返回该值

}

/*

按键值处理函数,返回扫键值

*/

unsignedcharKeyPro(void)

{

switch(KeyScan())

{

case0x7e:

return1;break;//0按下相应的键显示相对应的码值

case0x7d:

return2;break;//1

case0x7b:

return3;break;//2

case0x77:

return4;break;//3

case0xbe:

return5;break;//4

case0xbd:

return6;break;//5

case0xbb:

return7;break;//6

case0xb7:

return8;break;//7

case0xde:

return9;break;//8

case0xdd:

return0;break;//9

case0xdb:

return10;break;//a

case0xd7:

return11;break;//b

case0xee:

return12;break;//c

case0xed:

return13;break;//d

case0xeb:

return14;break;//e

case0xe7:

return15;break;//f

default:

return0xff;break;

}

}

/*

主函数

*/

voidmain(void)

{

unsignedinti,j,k,kk,Jz,num,n,t1,t2,a,b,x,y;

unsignedcharjinzhishu1[2],jinzhishu2[2];

unsignedchardushu[16];

unsignedchardushu2[16];

LCD」nit();

wela=1;P0=0Xff;wela=0;〃关闭数码管

LCD_Write_Com(0x0F);〃光标开,光标闪烁开

while⑴

{

LCD_Clear();

/*下为输入部分

A1:

jinzhishu1[1]='';

jinzhishu1[2]='';

LCD_Write_String(O,O,"EnterINsystem");

i=0;

while(qdbz==O)

{

Jz=KeyPro();

if(Jz!

=Oxff)

{

LCD_Write_Char(i,1,asc_code[Jz]);〃

jinzhishu1[i]=asc_code[Jz];

i++;

}

KeyScanQD();

KeyScanQL();

if(qlbz==1)

{

qlbz=0;LCD_Clear();i=0;j=0;

gotoA1;

}

if(i==2)qdbz=1;

}

qdbz=0;

LCD_Write_String(0,1,"");

依次显示输入字符

a=10;

if(jinzhishu1[0]=='1'&&jinzhishu1[1]=='0')

elseif(jinzhishu1[0]==2&&jinzhishu1[1]=='')

elseif(jinzhishu1[0]==8&&jinzhishu1[1]=='')

elseif(jinzhishu1[0]=='1'&&jinzhishu1[1]=='6')

else

{

LCD_Write_String(0,0,"WrongINsystem");

while(qdbz==0)

{

KeyScanQD();

}

qdbz=0;

gotoA1;

}

A2:

for(i=0;i<16;i++)

{

dushu2[i]='';

dushu[i]='';

}

LCD_Write_String(0,0,"EnterINnumber");

i=0;

while(qdbz==0)

{

num=KeyPro();

if(num!

=0xff)

{

LCD_Write_Char(i,1,asc_code[num]);〃

dushu2[i]=num;

i++;

}

KeyScanQD();

a=2;

a=8;

a=16;

依次显示输入字符

KeyScanQL();

if(qlbz==1)

{

qlbz=0;LCD_Clear();i=0;j=0;

gotoA2;

}

if(i==15)qdbz=1;

}

n=i;

qdbz=0;

LCD_Write_String(0,1,"");

A3:

jinzhishu2[1]='';

jinzhishu2[2]='';

LCD_Write_String(0,0,"EnterOUTsystem");

i=0;

while(qdbz==0)

{

Jz=KeyPro();

if(Jz!

=0xff)

依次显示输入字符

{

LCD_Write_Char(i,1,asc_code[Jz]);〃jinzhishu2[i]=asc_code[Jz];

i++;

}

KeyScanQD();

KeyScanQL();

if(qlbz==1)

{

qlbz=0;LCD_Clear();i=0;j=0;

gotoA3;

if(i==2)qdbz=1;

qdbz=O;

LCD_Write_String(0,1,"");

if(jinzhishu2[0]=='1'&&jinzhishu2[1]=='O')b=10;elseif(jinzhishu2[0]=='2'&&jinzhishu2[1]=='')b=2;elseif(jinzhishu2[0]==8&&jinzhishu2[1]=='')b=8;elseif(jinzhishu2[0]=='1'&&jinzhishu2[1]=='6')b=16;else

{

LCD_Write_String(0,0,"WrongOUTsystem");while(qdbz==0)

{

KeyScanQD();

}

qdbz=0;

gotoA3;

}

/*下为运算部分

n=n-1;

t2=dushu2[0];

for(j=0;j

{

t1=t2*a;

t2=t1+dushu2[j+1];

}

x=t2;//输入的10进制数先转为10进制

for(j=0;x>=b;j++)

{

y=x%b;

x=x/b;

dushu[j]=asc_code[y];

}

y=x;

dushu[j]=asc_code[y];k=j;

kk=j;

for(i=0;i<=kk;i++,k--)

{

dushu2[i]=dushu[k];

}

for(i=0;i<=kk;i++)

{

dushu[i]=dushu2[i];

}

/*下为输出部分

LCD_Write_String(0,0,"TheOUTnumber");

LCD_Write_String(0,1,"");

LCD_Write_String(0,1,dushu);

while(qdbz==0)

{

KeyScanQD();

KeyScanQL();

}

qdbz=0;

5、调试过程(包括出现的问题及解决的方法,以及要改进的地方,体会)

1)硬件调试:

焊接最小系统板完成之后就开始测试是否有焊接错误,首先是检验单片机的底座针脚是否联通,通过LED丁测试证明了所有的底座针脚焊接无误。

接着开始焊接排阻,电容,晶振等器件。

最后引出两个针脚插座作为VCC和GND勺插口。

接下来焊接独立按键和VCCGND莫块,完成后通过LED丁测试按键效果,证明焊接无误。

焊接工作完成后开始连线,连接控制模块和显示模块,连接控制模块和按键模块,连接各模块VCCGND口至UVCC和GND勺专用插口。

烧录入液晶显示范例程序,按键显示范例程序,测试硬件部分正常。

2)软件调试:

先编写子函数程序:

延时函数、判忙函数、写入命令函数、写入数据函数、清屏函数、写入字符函数、写入字符穿函数、LCD初始化函数、4*4矩阵按键扫描函数、确定/清零键扫描函数、矩阵按键值处理函数。

最后开始编写进制转换的主函数程序。

通过Keil编译软件编

译,用STC_ISP_v479烧录程序到单片机,利用学习开发板的集成按键、显示屏模块测试代码。

最先遇到的问题是读入按键值的问题,由于需要循环扫描的速度很快,实时监听键盘,多导致不能在同一个循环同时保持LCD的单个字符输出。

最后为了让LCD的逐一输出正常,在实时扫描的循环体内内嵌入一个判断语句,把LCD输出语句放入判断内,这样既保证了键

盘的实时监听,也保证了LCD显示屏的单个字符逐一正常输出。

接下来遇到的问题是键盘接收到的字符不能直接用于显示,由于读到的是有按键值处理函数的返回值,而显示的值却是要求ASCII型字符。

所以最后决定在开头编写转码函数表,把返回的1-16char型数字转成ASCII型的“0-9”“A-F”。

如此就使得LCD能够显示所希望得到的字符。

但再往下进行进制转换的具体实施发现,转成的ASCII码无法进行加减乘除运算,而原先的返回值是char型,也不确定是否能进行运算,同时也开始考虑int型变量是否需要使用。

最后经查阅资料发现,无论是char还是int型,都能直接被读取为机器语言进行运算,唯一区别是char型是1个字节,int型是2个字节的变量。

于是通过字符型数组储存读到的

按键返回值,并进行运算

通过之前积累的编程经验,采用的转进制方法是通过把所有进制数都转成十进制数作为中间数,然后在把中间数转为需要的进制数。

实施的初步阶段没有什么大问题,但是遇到大于255的10进制数的时候,发现运算出现了溢出,再次查阅资料发现了char型变量最多能

够表示0-255的10进制数,于是将此部分运算的中间变量设为int型,这样能够容纳的范围就变成了0-65535.

最后是输出运算的结果让LCD这其中也费了一番功夫,把底层运算的结果转成能够在LCD上显示的ASCII型又用到了转码过程。

到此软件调试基本告一段落,期间遇到的细节上的小问题数不胜数,最后在经过无数次修改程序烧录调试后得到了圆满解决。

 

最终软件调试成功时的烧录统计次数:

127

&实验结果(将运行结果拍照)

输入2进制

输入2进制数100011110

输出8进制

 

输出8进制数436

最终实物连线图

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

当前位置:首页 > 高中教育 > 数学

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

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