51单片机基本知识串口方式0,为什么一直收到FF

简单快乐,追求卓越
VB和51单片机串口通信讲解(只针对VB部分)
标记:该篇文章全部搬自如下网址:http://www.crystalradio.cn/thread--1.html,谢谢啦
里面关于中文接收的部分,大家可以好好学习下,题主也在研究中...................
Commport;设置或返回串口号。
SettingS:以字符串的形式设置或返回串口通信参数。Portopen:设置或返回串口状态。InputMode:设置或返回接收数据的类型。Inputlen:设置或返回一次从接收缓冲区中读取字节数。InBufferSize:设置或返回接收缓冲区的大小,缺省值为l 024字节。InBufferCount:设置或返回接收缓冲区中等待计算机接收的字符数。Input:从接收缓冲区中读取数据并清空该缓冲区,该属性设计时无效,运行时只读。OutBufferSize:设置或返回发送缓冲区的大小,缺省值为512字节。OutBufferCount:设置或返回发送缓冲区中等待计算机发送的字符数。Output:向发送缓冲区发送数据,该属性设计时无效,运行时只读。设置好这些属性和方法,程序很容易就可以编出来了,其中要注意的是串口的波特串设置,OnComm事件的程序编写。
程序的设计是这样的,计算机向单片机发送一个’S’,表示通信开始。然后紧接着又发送9个字符,其中最后一个字符是前面9个字符(包括’S’)的校验和。单片机正确接收到10个字符后,把10个字符从新又送回来。//---------------------------------------初始化串口设计-----------------------------
Private Sub Form_Load()Comm1.Setting="9600,n,8,1,"
’设置波特率和发送字符格式Comm1.CommPort=1
’设置通讯串口Comm1.InputLen=0
’设置或返回一次从接收缓冲区中读取字节数,0表示一次读取所有数据Comm1.InBuffersize=512Comm1.InBufferCount=0Comm1.OutBufferCount=0Comm1.Rthreshold=1Comm1.PortOpen=TrueEnd Sub
’打开串口//--------------------------------------给单片机发送’S’,开始通信-----------------------------
Private Sub Command1_C1ick()Timer1.Enabled=TrueEnd SubPrivate Sub Command2_C1ick()Varbuffet=“S”
Comm1.Ouput=varbuffeTimer2.Enabled=TrueEnd SubPrivate Sub Form_Unload(Cancel As Integer)Comm1.PortOpen=FalseEnd Sub//---------------------------------------向单片机发送数据-----------------------------
Private Sub Timer2_ Timer()Outputsignal=Str(Text2.text)
’向单片机发送数据Temp(1)=Cbyte(outputsignal)Varbuffer=tempComml.Output=varbufferTimer2.Enabled=FalseEnd Sub//---------------------------------------接收单片机发送的数据,并显示-----------------------------
Private Sub Comm1_OnComm()Select Case Comm1.CommEvent
’设置oncomm事件,读取片机内存的值Case comEvReceiveInputsignal=comm1.InputText1.Text=Asc(Inputsignal)
’单片机内存的值用textbox显示出Case ElseEnd selectEnd Sub关于MSCOMM控件的一些说明
VB5.0/6.的MSComm通信控件提供了一系列标准通信命令的接口,它允许建立串口连接,可以连接到其他通信设备(如Modem).还可以发送命令、进行数据交换以及监视和响应在通信过程中可能发生的各种错误和事件,从而可以用它创建全双工 、事件驱动的、高效实用的通信程序。但在实际通信软件设计过程中,MSComm控件并非像想像中那样完美和容易控制.特别是在中文Wln95/98下通信时更会出现问题。下面就从基础开始介绍,然后逐步讨沦MSComm控件在编程中出现的问题以及编程技巧。一、用MSComm控件通信1.串口通信基础知识
一般悦来,计算机都有一个或多个串行端口,它们依次为com1、Com2、…,这些串口还提供了外部设备与pC进行数据传输和皿信的通道。这些串口在CPU和外设之间充当解释器的角色。当字符数据从CPU发送给外设时,这些字符数据将被转换成串行比特流数据;当接收数据时,比特流数据被转换为字符数据传递给CPU,再进一步说,在操作系统方面,Windows用通信驱动程序(COMM.DRV)调用API函数发送和接收数据,当用通信控件或声明调用API函数时,它门由COMM. DRV解释并传递给设备驱动程序,作为一个vB程序员,要编写通信程序.只需知道通信控件提供给Windows通信AP1函数的接口即可.换句话说,只需设定和监视通信控件的属性和事件即可。2.使用Mscomm控件在开始使用MSComm控件之前。需要先了解其属性、事件或错误属性
描述CommPort
设置或返回通信端口号Settings
以字符串的形式设置或返回波特率、奇偶校验、数据位和停止位PortOpen
设置或返回通信端口的状态。也可以打开和关闭端口Input
返回和删除接收缓冲区中的字符Output
将字符串写入发送缓冲区CommEvent属性为通信事件或错误返回下列值之一。在该控件的对象库中也可以找到这些常量。常量
描述ComEventBreak
收到了断开信号ComEventCTSTO
Clear To Send Timeout。在发送字符时,在系统指定的事1件内,CTS(Clear To Send)线是低电平ComEventDSRTO
Data Set Ready Timeout。在发送字符时,在系统指定的事件内,DSR(Data Set Ready)线是低电平ComEventFrame
数据帧错误。硬件检测到一个数据帧错误ComEventOverrun 1006
端口溢出。硬件中的字符尚未读,下一个字符又到达,并且丢失ComEventCDTO
Carrier Detect Time。在发送字符时,在系统指定的事件内,CD(Carrier Detect)线是低电平。CD
也称为RLSD(Receive Line Singal Detect,接收线信号检测)ComEventRxOver 1008
接收缓冲区溢出。在接收缓冲区中没有空间ComEventRxParity 1009
奇偶校验错。硬件检测到奇偶校验错误7ComEventTxFull 1010
发送缓冲区满。在对发送字符排队时,发送缓冲区满ComEventDCB
检取端口DCB(Device Control Blick)时发生了没有预料到的错误通信事件包含了下面的设置:常量
描述ComEvSend
发送缓冲区中的字符数比Sthreshold值低ComEvReceive
接收到了Rthreshold个字符。持续产生该事件,直到使用了Input属性删除了接收缓冲区中的数据ComEvCTS
CTS(Clear To Send)线改变ComEvDSR
DSR(Data Set Ready)线改变。当DSR从1到0改变时,该事件发生ComEvCD
CD(Carrier Detect)线改变ComEvRing6检测到响铃信号。一些URAT(Universal AsynchronousReciver-
-Transmitters,通用异步收发器)不支持该事件ComEvEOF
收到了EOF字符(ASCII字符26)Error消息(MSComm控件)下表列出了MSComm控件可捕获的错误消息:常量
描述ComInvalidPropertyValue
无效的属性值ComSetNotSupported
属性只读ComGetNotSupported
属性只读ComPortOpen
端口打开时该存在无效
超时设置必须比0值大ComPortInvalid
无效的端口号
属性只在运行时有效
属性在运行时是只读的ComPortAleadyOpen
端口已经打开
设备标识符无效或不支持
不支持设备的波特率
指定的字节大小无效
缺省参数错误
硬件不可用(被其他设备锁住)
函数不能分配队列ComNoOpen
设备没有打开
设备已经打开
不能使用通信通知ComSetCommStateFailed
不能设置通信状态
不能设置通信事件屏蔽ComPortNotOpen
该存在只在端口打开是有效
设备忙ComReadError
通信设备读错误ComDCBError
检取端口设备控制块时出现内部错误搞清楚以上基本属性后,就可以开始编写通信许程序了。在VB5.0/6.0中新建一个工程文件。添加Microsoft Comm Control 5.0组件,在简体Form1中加入Command命令按钮并取名为CmdTest,MSComm控件取名为MSComm1,加入如下程序代码。Private Sub cmdTestClick ( )
'打开串口MSComml.CommPort =2
'设定Com2If MSComml.PortOpen = False ThenMSComm1.Settings = "9600,n,8,1"
'9600波特率,无校验,8位数据位,1位停止位MSComm1.PortOpen = True
'打开串口End ifMSComm1.OutBufferCount = 0
'清空发送缓冲区MSComm1.InBufferCount = 0
'滑空接收缓冲区'发送字符数据时注意必须用回车符(vbcr)结束MSComm1.Output="This is a qood book ! " &vbCr'泼打电话号码或发送AT命令MSComm1.Output = "ATDT
, & vbCr'发送字符数组数据时注意ByteArray必须事先定义赋值Dim ByteArray as byte( )'定义动态数组ReDim ByteArray(1)'重定义数组大小ByteArray ( 0 ) =0ByteArray ( 1 ) = 1MSComm1.Output = ByteArrayEnd Subprivate Sub MScommEvent( )Select Case MSComm1.CommEventCase comEvReceiveDim Buffer As VariantMSComm1.InputLen = 0'接收二进制数据MSComm1.InputMode= ComInputModeBinaryBuffer=MSComm1.Input'接收字符数据MSComm1.InputMode=comInputModeTextBuffer = MSComml.InputCase elseEnd SelectEnd sub( 程序1)
-----------------------------------------------------------------------------------------------------------------------
二、中文Win 95/98下的通信问题与解决方法1.接收的数据少于发送的数据
如果通过MSComm控件一次性传送较多的二进制数据,那么,很可能收到的数据不足。例如在设置为24oobps传输率的情况下,一次性可以传输2048个字符数据那么在大多数情况下。一次只能收到1200个字符左右,这址出为新版的MSComm32.OCX中存在一个影响传输二进制数据的臭虫(bug).注意这不是特性。
32位Windows API函数(以下简称API)使用了几个用COMMTIMEOUTS结构表示的限时变量,WriteTotalTimeOutConstant 即是其中的一个,它被Windows内部设定为5000(即5秒),这个常量决定了在通信驱动程序停止传输之前花费在发送缓冲区中数据的时间的长短,5秒钟意味着通信速度为1200bps情况下仅能发送600个字符,24oobps情况下仅能发送1200个左右的字符。事实上,在一个缓冲区内一次性发送更多的数据是非常可能的。这个bug同样也能引发问题,甚至在高速串口门通信情况下,即使系统在使用流控制,无论丛软件流(Xon/XofI)还是硬件流(CTS/RTS)。假如数据在发送缓冲区中时,流控制停止了传输,如果停止时间超过5秒钟.则数据就会丢失。在某些环境下,5秒钟可能相当短.不过也不必担心, VB 5.0/6.0版本的MSComm控件有一个新增的重要的属性称为CommID, CommID指的是当串口被打开时,被API所调用的串口句柄或称标志,这也意味着能利用API接口函数去修改这个常量。每次串口关闭后,Windows会自动将之恢复为5000,所以,每次打开串口后需要重斩设定以下API声明,其代码见下程序。Type COMMTIMEOUTSReadIntervalTimeout As LongReadTotalTimeoutMultiplier As LongReadTotalTimeoutConstant As LongWriteTotalTimeoutMultiplier As LongWriteTotalTimeoutConstant As LongEnd TypeDeclare Function SetCommTimeouts Lib "Kernel32"(BYVal hFile As Long, lpComm TimeoutsAs COMMTIMEOUTS) As LongDeclare Function GetCommTimeouts Lib "Kernel32"(ByVal hFile As Long, lpCommTimeouts As COMMTIMEOUTS) As LongDim timeouts As COMMEOUTSDim Ret As LongIf Comm1.PortOpen = False ThenComm1.PortOpen = TrueEnd ifRet=GetCommTimeouts ( Comm1.CommID , timeouts )'Set some default timeoutstimeOuts.ReadIntervalTimeout = 1timeouts.ReadTotalTimeoutMultiplier =1timeouts.ReadTotalTimeoutConstant =1timeouts.WriteTotalTimeoutMultiplier =1timeouts.WriteTotalTimeoutConstant=( Comm1.OutBufferSize\Val(Comm1.Settings))*Ret=SetCommTimeouts( Comm1.CommID , timeouts )( 程序2)2.如何发送大于128的字符数据
在通信程序中,以单字符方式逐个发送数据时,每一个数据范围 0-255(即十六进制的00-FF)。在单字符版本的英文Win95或DOS版的BASIC程序中,只需要将相应的数据转换成相应的字符发送到通信端口即可。但在中文Win95/98下却行不通,假设在中文Win95/98下运行以下程序:Dim iFor i=0 to 255MSComm1.Output=chr(i)Next i
希望在接收端得到预期的0-255之间的数据,结果却是:前129个数据接收正确,为0-128,后面127个数据为126个0和一个255,造成这种给果的原因在于中文Windows使用的是双字节字符集(DBCS)系统。DBCS系统使用0-128之间的数字表示ASCII字符,大于128的数字仅作为前导字符,它只是显示是一个非拉丁语系的字符,而并不代表实际意义。上述程序在调用CHR()函数时用到了DBCS字符集,冈此产生了此类错误。那么,如何发送人于128的数据呢?答案是使用字符数组,将以上程序改为:Dim cc(255) As ByteFor i = 0 To 255cc(i) = iNext iMSComm1.Output = ccDoDoEventsLoop Until MSComm1.OutBufferCount = 0'接收过程 MSComm1_OnComm()Select Case MSComm1.CommEventCase comEvReceiveDim Buffer As Variant, b1,iMSComm1.InputMode=comInputModeBineryMSComm1.InputLen = 0Buffer = MSComm1.InputFor i=LBound (Buffer) To UBound (Buffer )Debug.Print Buffer ( i ) ;Next iCase . . . . .
3.如何发送0字符(00H,NULL)在VisuaI C++中使用串口控件发送0字符有些麻烦,但在VB5.0/6.0中只要注意以下两点即可:(1)设置MSComm控件的属性 NullDiscard=False;。(2)使用二进制接收,即用 MSComm1.InputMode=ComInputModeBinary便可以解决问题;
4.如何发送递中文字符串(DBcS字符)VB5.0/6.0的各种参考书上均指明MSComm通信控件不能发送或接收双字节字符集系统DBCS)的二进制数据,这对于我国及亚洲一些使用DBCS字符集的国家不能不说是一大人遗憾。但是我在实践中发现,用MSComm控件也可以发送中文字符,具体方法有以下两种:(1)直接发送
直接发送即把中文字符等同于英文字符。如:MSComm1.Intput= " 这是一行中文数据!" ,但这种方法发送的中文数据不能太长,发送缓冲区和接收缓冲区的大小需设定为中文字符的两倍以上,而且发送与接收系统所处的操作系统版本最好要一致,否则会出现接收或发送缓冲区溢出之类的错误。这种方法时用于一般要求不太高的场合。(2)间接发送
在发送端将汉字或字符转换为机器内码或区位码数据数组,然后将咏转换后的数据发送到串口,在接收端接收到数据后,按照相反的顺序得到的数据转换为相应的汉字或字符,在转换过程中.要用到位运算,如取得汉字的内码后需要将高字节和低字节分开,而VB5.0/6.0中并没有提供此类函数,以下是求整数高、低字节的函数。Public Function HiByte(a As Integer )Dim bb= a And &HFF00b = b / 256If b&0 Then b = b + 256HiByte = bEnd FunctionPublic Function LowByte(a As Integ`er)Dim bb = a And &HFFLowByte = bEnd Function5.如何用单机进行通信测试
通常在写好了通信程序后需要两台PC或一台Pc、一台单片机.将通信口连接后进行测试,但很多时侯因条件限制仅有单台PC机,测试项目很简单,那么能否测试呢?当然可以,而且方法也很简单。对于九针的串口,找一个废弃的串口鼠标,剥外鼠标线,将连接2、3针的线对接即可;对于25针的串口,找一枚曲别针(最好有塑料外套的)将它扯直,剥削去两头的塑料后在两头各弯一个圆圈,中间对忻后直接套接在串口的2、3针上即可。如果但心不够安全,则可以将5针按地。'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''关于mscomm的用法,提高篇......[mgwmj]&MSCOMM控件是个好东西,如果您能够充分了解他,他会为您衷心的效劳。
大致看了一下下午有关讨论MSCOMM的话题,觉得有必要说说我的心得,我一般只做硬件,没有系统的学过软件,只是业余时间学学用用,多少掌握了一点,也在此拿出来玩玩,不知有错没有,我可是以为我已经做的很好了^_^这是一个VB通用串口事件驱动接收程序。一次性接收一个数据包,数据包可以为任意字节,保证不会丢失一个数据!Private Sub MSComm_OnComm()
Dim S() As Byte
Dim SS(1024) As Byte
Static N As Long
Static T As Variant
If (MSComm.CommEvent = comEvReceive) Then
S = MSComm.Input
'只要有数据就收进来,哪怕只是一个
If (Timer - T & 0.01) Then
'间隔10MS以上就认为是一个新的包
'text1用于搜集和显示接收(HEX格式)
For i = 0 To UBound(S)
'一个数据包可能产生若干个oncomm事件
Text1.Text = Text1.Text & Right("0" & Hex(S(i)) & "H", 3) + " "
SS(N+i)=S(i)
'接收数据包缓存于SS()
N=N+UBound(S)
End IfEnd Sub转自AMOBBS
单片机与RS-232的串口通信+VB程序
VB6使用API实现串口通信
VB.NET串口通信例子--我的回忆录
串口调试工具_VB_自己制作的
VB中串口通讯的实现
没有更多推荐了,查看: 1720|回复: 4
我用单片机读GT21H16S2Y字库芯片,但是读出来数组全是0xff,为什么啊?
主题帖子精华
新手入门, 积分 32, 距离下一级还需 -12 积分
在线时间0 小时
求一个正确的程序
这是我写的程序
[C] 纯文本查看 复制代码#include&stc12c5a60s2.h&
#define uchar unsigned char
#define uint unsigned int
xdata uchar a_dot[32],a_dot_1[256];
void save();
void led();
/*-------------------------------------------
引脚连接宏定义
------------------------------------------------*/
sbit GT_CS = P1^0;
sbit GT_SO = P3^5;
sbit GT_SI = P1^2;
sbit GT_CLK = P1^3;
/*点阵定义*/
sbit LA = P0^0; //138译码输入
sbit LB = P0^1;
sbit LC = P0^2;
sbit LD = P0^3; //138译码输入
sbit R1_595 = P0^4; //红数据输入1
sbit R2_595 = P0^5; //红数据输入2
sbit EN = P2^1; //输出使能
sbit LT = P2^2; //锁存
sbit SK = P2^3; //时钟
void delay(uchar i)
uchar x,y;
for(x=i;x&0;x--)
for(y=110;y&0;y--);
/*--------------------------------------------------
:ReadByte
:读取任意地址数据
输入参数:
输出参数:
---------------------------------------------------
unsigned char ReadByte(unsigned long Address)
unsigned char i = 0,j = 0,temp = 0x0B;
GT_CS = 0;
//发送指令
for(i=0; i&8; i++)
GT_CLK = 0;
GT_SI = temp&0x80;
temp &&= 1;
GT_CLK = 1;
//发送地址
for(j=1; j&4; j++)
temp = ( (unsigned char*)&Address )[j] ;
for(i=0; i&8; i++)
GT_CLK = 0;
GT_SI =temp & 0x80;
temp &&= 1;
GT_CLK = 1;
//发送空字节
for(i=0; i&8; i++)
GT_CLK = 0;
GT_SI = 0;
GT_CLK = 1;
//接收数据
for(j=0;j&32;j++)
for(i=0; i&8; i++)
GT_CLK = 0;
temp &&= 1;
if(GT_SO)temp |= 0x01;
else temp &= 0xFE;
GT_CLK = 1;
GT_CS = 1;
/*GB2312 功能: 把Unicode内码转换为GB2312内码
参数: srcCode:输入Unicode内码串。
destCode:转换后得到的GB2312内码串。*/
//uint UnicodeToGB2312(uint code1)
//unsigned char i = 0,j = 0,temp = 0x0b;
//unsigned long ptr,result=0;
//uchar GB2312Code[2];
//unsigned long UToGb2312Buff =0x67d70;
////uchar UCSadd1,UCSadd2,UCSadd3;
//if(code1&0xa0) result=1;
//else if(code1&=0xf7) h=code1-160;
//else if(code1&0x2c7) result=1;
//else if(code1&=0x2c9) h=code1-160-463;
//else if(code1&0x2010) result=1;
//else if(code1&=0x2312) h=code1-160-463-7494;
//else if(code1&0x2460) result=1;
//else if(code1&=0x2642) h=code1-160-463-;
//else if(code1&0x3000) result=1;
//else if(code1&=0x3017) h=code1-160-463-3;
//else if(code1&0x3220) result=1;
//else if(code1&=0x3229) h=code1-160-463-3-520;
//else if(code1&0x4e00) result=1;
//else if(code1&=0x9b54) h=code1-160-463-3-520-7126;
//else if(code1&0x9c7c) result=1;
//else if(code1&=0x9ce2) h=code1-160-463-3-520-;
//else if(code1&0x9e1f) result=1;
//else if(code1&=0x9fa0) h=code1-160-463-3-520-;
//else if(code1&0xe76c) result=1;
//else if(code1&=0xe774) h=code1-160-463-3-520--18379;
//else if(code1&0xff00) result=1;
//else if(code1&=0xff5f) h=code1-160-463-3-520--;
//else if(code1&0xffe0) result=1;
//else if(code1&=0xffe5) h=code1-160-463-3-520---128;
//else result=1;
//if(result==0)
// ptr = UToGb2312Buff + (h&&1); //UCS2转GB2312在芯片的首地址
////UCSadd1=ptr/65536;
////UCSadd2=ptr/256;
////UCSadd3=ptr%256;
//GT_CS = 0;
//发送指令
for(i=0; i&8; i++)
GT_CLK = 0;
GT_SI = temp&0x80;
temp &&= 1;
GT_CLK = 1;
for(j=1; j&4; j++)
temp = ( (unsigned char*)&ptr )[j] ;
for(i=0; i&8; i++)
GT_CLK = 0;
GT_SI =temp & 0x80;
temp &&= 1;
GT_CLK = 1;
//发送空字节
for(i=0; i&8; i++)
GT_CLK = 0;
GT_SI = 0;
GT_CLK = 1;
for(j=0;j&2;j++)
for(i=0;i&8;i++)
temp=temp&&1;
GT_CLK = 0;
GT_CLK = 1;
if(GT_SO) temp++;
GB2312Code[j]=
GBcode=GB2312Code[0]*256+GB2312Code[1]; //高地位组合
return (GBcode);//返回GB2312汉字编码
void hangout(uchar hang)
LA=hang&0x01;
LB=hang&0x02;
LC=hang&0x04;
LD=hang&0x08;
void Send_595(uchar dat,uchar dat1)//双参数,时钟、单锁存共用,双数据输出
for(i=0; i&8; i++)
SK = 0;//拉低同步时钟
dat &&= 1;//数据移位
R1_595=~CY;//数据位输出
dat1&&=1;//数据1移位
R2_595=~CY;//数据1位输出
SK = 1;//恢复同步时钟
void led(uint GB_code)
uchar MSB,LSB,ADH;
unsigned long Address,BaseA
MSB= GB_code/256;
LSB= GB_code%256;
BaseAdd=0;
if(MSB == 0xA9 && LSB &=0xA1)
Address =(282 + (LSB - 0xA1))*32+ BaseA
else if(MSB &=0xA1 && MSB &= 0xA3 && LSB &=0xA1)
Address =( (MSB - 0xA1) * 94 + (LSB - 0xA1))*32+ BaseA
else if(MSB &=0xB0 && MSB &= 0xF7 && LSB &=0xA1)
Address = ((MSB - 0xB0) * 94 + (LSB - 0xA1)+ 846)*32+ BaseA
// ADH=Address/0x10000;
ReadByte(Address);
void main()
// uchar temp,temp1;
P3M1=0X10;
P3M0=0X00;
// UnicodeToGBa);
led(0xB0A2);
for(j=0;j&16;j++)
Send_595(a_dot[2*j],0);Send_595(a_dot[2*j+1],0);
Send_595(0,0);Send_595(0,0);
Send_595(0,0);Send_595(0,0);
Send_595(0,0);Send_595(0,0);
LT=1;//锁存
hangout(j);//行选
EN=0;//开显示0
delay(1);//显示延时。
EN=1;//关显示 1
主题帖子精华
金钱130415
在线时间1181 小时
主题帖子精华
高级会员, 积分 814, 距离下一级还需 186 积分
在线时间252 小时
把程序做几步调试,先做一个字符表用查表的方式送595显示,正确后再读字库数据并显示。
主题帖子精华
新手入门, 积分 20, 距离下一级还需 0 积分
在线时间0 小时
我也遇到同样的问题,有的字通过内码可以读出数据,但感觉数据错误,有的字的不能读出,读出全为0xFF
主题帖子精华
初级会员, 积分 169, 距离下一级还需 31 积分
在线时间28 小时
多看看程序,多半是程序有问题。
必看:"原子哥”极力推荐 /1
资料包括正点原子开发板全套视频,源码,手册等。原子粉丝请关注公众号。
Powered by幸福就在呼吸间
51单片机 11.0592MHz 产生115200 波特率
51单片机 11.0592MHz 产生115200 波特率
连接:http://homezzy.blog.hexun.com/_d.html
具体定义如下:
SM0 SM1 SM2 REN TB8 RB8 TI RI
SM0、SM1为串行口工作模式设置位,这样两位可以对应进行四种模式的设置。
看表8-2串行口工作模式设置。
同步移位寄存器
fosc/32或fosc/64
表中的fosc代表振荡器的频率,也就是晶振的频率。
UART为(Universal Asynchronous Receiver)的英文缩写。
今天解决了一个小问题(查书后才得到确切结论。。。)用51单片机+11.0592的晶振,如何产生115200的波特率?
本来感觉这个小意思,直接初始化定时器1,程序如下:
void init_com( void )
SCON = 0x50 ; //串口工作方式1,8位UART,波特率可变
TMOD |= 0x20 ; //定时器1,工作方式2,自动再装入8位定时器
PCON |= 0x80 ; //SMOD=1; 波特率加倍
TH1 = 0 //波特率:9600 晶振=11.0592MHz
IE |= 0x90 ; //使能串口中断
TR1 = 1 ; // 定时器1开始
结果发现这样只能得到9600的波特率。。。
当SMOD=1时,K=2,波特率加倍,公式为:
波特率=K××12×(256-TH1)
所以,TH1=0xfa=256-(2××波特率) 其中波特率为9600
这时,及时令TH1=0xff,所得波特率最大只能为57600,也就是说,这样无法得到115200的波特率。。。
这样就只有采用其他方法了:
1、换晶振,用22.1184M晶振,在TH1=0xff时,刚好可以产生115200波特率。
2、采用6个时钟周期的单片机(换单片机啊。。。)
3、增强型51单片机有定时器2!(幸好偶用的是增强型。。。)
就用第三种方法啦!这时的波特率公式如下:
波特率= / { 32×[65536-(RCAP2H,RCAP2L)] }
其中的RCAP2H,RCAP2L为自动重装值,由上式得:
RCAP2H,RCAP2L=00 / (32×波特率)
这样得波特率为115200时,RCAP2H,RCAP2L=0xff,0xfd,初始化程序如下:
void init_com( void )
SCON=0x50; //串口工作方式1,8位UART,波特率可变
//波特率:115200 晶振=11.0592MHz
RCAP2H=0xFF;
RCAP2L=0xFD; //16位自动再装入值
/*****************/
EXEN2=0; //波特率发生器工作方式
/*****************/
TR2=1 ; //定时器2开始
这样就实现用51单片机+11.0592的晶振,产生115200的波特率了,当然老鸟看来很基本,但好长时间不用了,再复习一下~~
用51单片机+11.0592的晶振,如何产生115200的波特率
关于波特率与字节传输速率计算
关于89C52单片机11.0592M晶振产生115200波特率的方法
串口通信中定时器1与波特率的关系
时钟频率、波特率、波特因子关系详解
没有更多推荐了,

我要回帖

更多关于 单片机串口发送数据 的文章

 

随机推荐