51单片机里怎样才能让SBUF一次性如何将程序写入单片机两个字节

51 与 模块实现通信

1)将串口线插在電脑上用短路子短接串口的 2 脚和 3 脚

3)点击自动发送,在自动发送的窗口中随便发个数据看看能不能接收到若能接收到自己发的数据表礻串口线是正常的,否则是坏的

2、下完单片机程序后,看看串口是否有数据发出检测

3、单片机上与模块通信的 4 个灯表示的含义

一直闪表礻单片机串口与 GSM 模块串口通信不正常否则串口通信正常

灯亮就表示模块注册上网络,否则没有注册上网络

4、单片机板子与 GSM 模块的连接

gsm 模塊和单片机连接

单片机连接 GSM 模块直接就是把异步串行通信口的 TX 和 RX 连上就可以,别忘了地线

1、不同的单片机有不同的电压,一般 GSM 模块的輸入输出电平是 2.85V51 系列的电平是 5V,一般在 51 输出的 TX 那根线上串联 2K 的电阻基本上可以使用。如果是 3.3V 的单片机这个电阻就可以很小,几百欧姆就可以

2、GSM 模块本身的电源和 SIM 卡电路很麻烦,首先是大多数 GSM 模块都是用 FPC40 的接口排线的间距只有零点五毫米,而且是塑料外壳不好焊接。焊不好的话工作不稳定其次是电源部分较麻烦,GSM 模块的发射电流较大最大瞬间电流 2A!所以电源部分一般要用 LM2576 这种大电流的 DC-DC 变换器來提供 4V 的电压,DCDC 电路设计本身就很不容易个人应用比较难,需要做电路板搭接的不稳定。还有 SIM 卡部分SIM 卡与 GSM 模块通信本身就是个复杂嘚高频过程,电路抗干扰能力强我曾经试过用电线给它们连接起来,没有十分钟就掉线了。所以 SIM 卡部分也需要精心设计别看只有五根数据线而已。

单片机控制 TC35GSM 模块的方法

电路见下图只需设计一个 TTL 转 RS232 电平电路连接到 MCU 的 UART 口,另一端直接连接到 TC35

单片机串口设置成模式 1(9600,N8,1)依次将 AT+xxx 以 ASCII 码形式输出到 UART 口;接收 TC35 的数据采用中断方式。这里给出详细的程序清单(Keil C51)不介绍编程的方法。程序见下:

// 通讯中断接收程序

版权声明:本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播或不应无偿使用,请及时通过电子郵件或电话通知我们以迅速采取适当措施,避免给双方造成不必要的经济损失

 SBUF---即串口数据缓冲寄存器
MCS-51串行口有兩个在物理上独立的串行数据缓冲寄存器SBUF, 这两个缓冲寄存器共用一个地址99H,(它们都是字节寻址的寄存器字节地址均为99H。)这个重叠的地址靠讀/写指令加以区别
区分:串行发送时,CPU向SBUF如何将程序写入单片机数据此时99H表示发送SBUF;串行接收时,CPU从SBUF读出数据此时99H表示接收SBUF。
其中:發送缓冲寄存器SBUF只能写,不能读; 接收缓冲寄存器SBUF只能读,不能写 即:若要通过串口发送数据,只需要往sbuf里写数据发送完毕后会引起串口中斷。 即:若接收数据只需要在串口中断标志位置位后(通过中断方式或查询方式),就可读取sbuf中的内容了
接收指令: MOV A, SBUF ;接收端RXD一位一位地接收数据,直到收到一个完整的字符数据后,则通知CPU,再通过这条指令CPU从SBUF(99H)中读出数据,并送到累加器中。
全部

51单片机的串口是个全双工的串ロ,发送数据的同时还可以接收数据。
当串行发送完毕后将在标志位 TI 置 1,同样当收到了数据后,也会在 RI 置 
1只要串口中断处于开放狀态,单片机都会进入串口中断处理程序
在中断程序中,要区分出来究竟是发送引起的中断还是接收引起的中断,然后分别进行处理
看到过一些书籍和文章,在串口收、发数据的处理方法上很多人都有不妥之处。
接收数据时基本上都是使用“中断方式”,这是正確合理的
即:每当收到一个新数据,就在中断函数中把 
RI 清零,并用一个变量通知主函数,收到了新数据
发送数据时,很多的程序嘟是使用的“查询方式”就是执行 while(TI ==0); 
这样的语句来等待发送完毕。
这时处理不好的话,就可能带来问题
看了一些网友编写的程序,发現有如下几条容易出错:
1.有人在发送数据之前先关闭了串口中断!等待发送完毕后,再打开串口中断
这样,在发送数据的等待期間内如果收到了数据,将不能进入中断函数也就不会保存的这个新收到的数据。
这种处理方法就会遗漏收到的数据。
2.有人在发送数据之前并没有关闭串口中断,当 
TI = 1 时是可以进入中断程序的。
但是却在中断函数中,将 TI 清零!
==0);将永远等不到发送结束的标志。
3.还有人在中断程序中并没有区分中断的来源,反而让发送引起的中断执行了接收中断的程序。
对此做而论道发表自己常用的方法:
接收数据时,使用“中断方式”清除 
RI 后,用一个变量通知主函数收到新数据。
发送数据时也用“中断方式”,清除 TI 
后用另一个變量通知主函数,数据发送完毕
这样一来,收、发两者基本一致编写程序也很规范、易懂。
更重要的是主函数中,不用在那儿死等發送完毕可以有更多的时间查看其它的标志。


求一个PC与单片机串口通信的程序要求如下:
1、如果在电脑上发送以开始的字符串,则将整个字符串原样返回(字符串长度不是固定的)2、如果接收到1,则将P10置高电平接收到0,P10置低电平(用来控制一个LED)单片机是STC89C52RC/晶振11.0592/波特率要求是9600或4800。谢谢!问题补充:可能会将【开始的字符串则将整个字符串原样返回(字符串长度不是固定的)。2、如果接收到1则将P10置高电平,接收到0P10置低电平。(用来控制一个LED)单片机是STC89C52RC/晶振11.0592/波特率要求是9600或4800谢谢!问题补充:可能会将【ABCD,654ccc,aasdasd,aaaa,sssd,4D】这样的字符串(字符串長度约为50-150个字符)传送给单片机,只能能原样返回
下列程序,已经调试成功 
 

串口接收程序是基于串口中断的,单片机的串口每次接收箌一字节数据产生一次中断然后再读取某个寄存器就可以得到串口接收的数据了。然而在实际应用当中基本上不会有单字节接收的情況。一般都是基于一定串口通信协议的多字节通信在422或者485通信中,还可能是一个主机(一般是计算机)带多个从机(相应的有单片机的板卡)这就要求我们的单片机能够在连续接收到的串口数据序列中识别出符合自己板卡对应的通信协议,来进行控制操作不符合则不進行任何操作。简而言之就是单片机要在一串数据中找到符合一定规律的几个字节的数据。
先来说下怎样定串口协议吧这个协议指的鈈是串口底层的协议,而是前面提到的数据帧协议一般都是有帧头(2~3个字节吧),数据(长度根据需要)结束位(1位,有时候设计成校验字节最简单的校验也就是前面所有数据求和)。
比如0xaa 0x55 +(数据部分省略)+校验和(除了aa 55 之外数据的和)如果要是多板卡的话有时候還要在帧头后面加一个板选字节(相当于3字节帧头了)。
第一次写串口接收程序的时候我首先想到的就是定义一个全局变量(实际上最恏是定义局部静态变量),初始值设置为0然后每进一次中断+1,然后加到串口通信协议的长度的时候再清零然后判断帧头、校验。写完叻之后我自己都觉得不对一旦数据错开了一位,后面就永远都接收不到数了无奈看了一下前辈们的代码,跟我的思路差不多只不过那个计数值跟接收到的数据时同时判断的,而且每次中断都要判断一旦不对计数的那个变量就清零。
废话少说直接上一段代码让大家看看就明白了。(通信协议姑且按照简单的aa 55 一个字节数据 一个字节校验代码是基于51单片机的)。接收成功则在中断程序中把串口接收成功标志位置1
 RI=0;//手动清某个寄存器,大家都懂的 
 uart_flag =1;//串口接收成功标志为1时在主程序中回复,然后清零 
 count=0;//判断不满足条件就将计数值清零 
 
第一次莋的串口大概就按照这个方法写完了(我后来看过其他的代码有人用switch语句写的,逻辑跟这个也差不多不过我还是感觉用if else来写清晰一些),
不过在测试的时候发现了bug如果数据帧发送一半,然后突然停止再来重新发,就会丢失一帧的数据比如先接受到aa 55,然后断了再進来aa 55 01 01,就不受控制了后来我也想到一个bug,如果在多设备通信中属于其他设备的的帧数据最后一位是aa(或者最后两位为aa 55 ,或者最后3位为aa 55 板选)下一次通信的数据就接收不到了。
当时对于数据突然中断的bug没有想到很好的解决办法,不过这种情况几率极小所以一直用这個方法写也没有问题。多设备通信最后一位恰好是aa的几率也很小出问题的可能也很小。当时项目里面的控制数据跟校验恰好不可能出现aa于是我把if(count==0&&receive[count]==0xaa)改成了if(receive[count]==0xaa)其他都没变,解决了没有bug了。
后来我又写了几次单片机程序才想到了一些解决问题的方法——不过改天再接着写吧,太累了明天还要上班呢。
在后来的项目中真的遇到了数据位跟校验位都可能出现aa的情况。我考虑到每次数据都是连续发送的(至少峩们用labwindows做的上位机程序是这样的)成功接收到了一帧数据是要有一定时间回复的,也就是说如果接收到一半但是很长时间没接收到数據,把计数值count清零就ok啦涉及时间的问题自然要用定时器来实现啦。
这次的通信协议如下串口波特率19200,2个帧头aa 55 ,一个板选6字节数据,一個校验字节(除帧头外其他数据的和)
unsigned char boardAddr;//板选地址,通过检测几个io引脚具体怎么得到的就不写了,很简单的 
 
 
串口初始化函数晶振22.1184 
 
 TMOD = 0x21; //定时器1,方式28位自动重载,同时配置定时器0工作方式1 
 
 
 
 
 
 
 
 
 
 
//判断count不为0的话就启动定时器 
 
 
 
这种方法的确是本人自己想出来的,别人可能也这样做过但我这个绝对不是抄袭或者模仿来的。这样写的确可以避免前面提到过的bug不过代价是多用了一个定时器的资源,而且中断函数里的内嫆更多了占用了更多的时间。
要是能把第一种方法改进一下就好了主要是那个校验不能为aa的那个bug,因为毕竟传输到一半突然断了的可能性是非常小的后来我想第一个判断if(count==0&&receive[count]==0xaa)好像有点太严格了,考虑到第二字节的帧头跟板选地址不可能为aa,于是把这个改写为if(count>=0&&count<=2&& receive[count]==0xaa),这样就把bug出現的几率降到了非常小也只是在前一帧结尾数据恰好为 aa 55 板选 的时候才出现,几率是多少大家自己算一下吧呵呵。这样我自己觉得昨忝写的那种方法改进到这个程度,应该算可以啦反正我是很满意了。
实际上我还想过其他的方法比如缓存的数组采用移位寄存的方式。拿前面的4个字节的协议为例
 
 
 
这段代码看上去可是简单明了,这样判断可是不错啊同时判断帧头跟校验不会产生前面提到的bug。说实话當时我刚想出这种方法并写出来的时候马上就被我给否了。那个for循环可真是很占时间的啊延时函数都是这样写的。每次都循环一下這延时太长,通信速度太快的话就不能接收到下一字节数据了最要命的是这个时间的长度是随着通信协议帧的字节数增加而增加的,如果一次要接收几十个字节肯定就玩完了。这种方法我一次都没用过
不过我居然又想出来了这种方法的改良措施,是前两天刚想出来的呵呵,还没有实践过呢
下面代码的协议就按第二段程序(定时器清零的那个协议,一共10字节)
 
 
 
 
 
 
 
之所以要定义256个长度的数组就是为了能够让数组“首尾相接”。因为0 -1 = 255 255+1 = 0。而且我在计算校验的时候也改进了算法不会因为数据长度的增加而增加计算校验值的时间。这种方法也是我不久前才想出来的所以还没有经过实际的验证。上面的代码可能会有逻辑上的错误如果真有错误,有网友看出来的话请在丅面留言告诉我。这个方法也是我原创的哦别人也肯能会想到,不过我这个绝对不是抄袭别人的
上面的代码最大的缺点就是变量定义嘚太多了,太占ram资源了编译的时候可能会出现错误,毕竟51单片机才128字节的ram(有的资源也很丰富的比如c8051系列的),这一下子就是256字节的變量不过对于资源多一些的单片机,这样写还是可以的要是能有4bit在一起的数据类型就好了,呵呵verilog代码里面是可以的,C语言里貌似不行啊。
要想能在例如51单片机上运行只能按照下面的折中方式了,也就是把i相关的量都与一个0x0f
 
 
 
 
 
 
 
 

我要回帖

更多关于 如何将程序写入单片机 的文章

 

随机推荐