52单片机如何对PC串口发送来的数据进行存储,存储在哪?RAM还是ROM中?
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了52单片机如何对PC串口发送来的数据进行存储,存储在哪?RAM还是ROM中?相关的知识,希望对你有一定的参考价值。
ROM不是存储程序的么,怎么存储接受数据?数据大小为1160字节,跪求精通串口通信大神解救,百思不得其解啊!如果连续发送三次数据,大小分别是1160,1360,1800字节要怎么分别存储在单片机中?怎么定义存储地点?新人接触串口通信。。。能有具体C程序解释更棒了!
参考技术A STC内部有256字节的ram,还有1024bytes的扩展RAM,请查看数据手册,但这显然不能满足你的要求,所以建议你使用外部扩展Ram,选择6264(8K)的RAM就可以了,怎么接请参看C51外部RAM扩展相关文档,另外不建议使用FlashRom存取,因为首先有写入次数限制,在写入时需要擦出整个扇区,不知道这样说你清楚不?定义你的接收缓冲区:
#define MAX_BUF_LEN 1800
xdata char cRxdBuf[MAX_BUF_LEN];//data 指底128bytes,idata指高128bytes,xdata指外部RAM
另外,接外部RAM时记得禁止扩展的1024Bytes,否则会产生冲突。追问
感谢你,资料查了很多,大概懂你的意思,我还是想用片内EEPROM来存放这三组数据,而且片内EEPROM有4K已经满足大小的要求了
追答哦,你一定要这样作那也可以哈,不过超过擦出次数你的东西就挂掉了,另外STC手册上有例程,可以直接拿来用。
参考技术B STC单片机内部有部分空间是EEPROM,可以存储数据。串口收到的数据存在EEPROM中是可以的,但写数据前,需先擦除。
串口波特率是多少,写入数据有时间要求?追问
嗯,是STC的片,波特率是9600,写入数据的是时间没要求。有4K的EEPROM,但是C没学到家,不知道怎么样写这段数据存储程序,能具体点求教么?新人不懂的太多,时间又紧
追答/*向单片机内部EEPROM空间读出一字节数据*/
uchar read_51flash_controlword(uint eeprom_addr)
uchar temp;
IAP_CONTR=0X83;
IAP_CMD =read_cmd;//字节读命令,,
IAP_ADDRL=eeprom_addr&0xff;
IAP_ADDRH=eeprom_addr>>8;
IAP_TRIG=0X5a;
IAP_TRIG=0Xa5;//
//delay_nus(2);
Nop();
temp=IAP_DATA;
IAP_CONTR=0X00;
IAP_CMD =0X00;
IAP_TRIG=0X00;
IAP_ADDRH=0Xff;
IAP_ADDRL=0Xff;
return temp ;
void write_51eeprom_1byte(uint eeprom_addr,uchar w_data)//写数据到指定地址
IAP_CONTR=0X83;
IAP_CMD =program_cmd;//字节编程命令,,
IAP_ADDRL=eeprom_addr&0xff;
IAP_ADDRH=eeprom_addr>>8;
IAP_DATA =w_data;
IAP_TRIG=0X5a;
IAP_TRIG=0Xa5;//
Nop();
IAP_CONTR=0X00;
IAP_CMD =0X00;
IAP_TRIG=0X00;
IAP_ADDRH=0Xff;
IAP_ADDRL=0Xff;
/*擦除单片机内部EEPROM空间一个数据的扇区*/
void earse_one_sector_in51_ctr(uint eeprom_page_addr)
EA=0;
IAP_CONTR=0X83;
IAP_CMD =erase_cmd;//字节擦除命令,, IAP_CMD
IAP_ADDRH=eeprom_page_addr>>8;
IAP_ADDRL=eeprom_page_addr;
IAP_TRIG=0X5a;
IAP_TRIG=0Xa5;//
//delay_nus(2);
Nop();
Nop();
IAP_CONTR=0X00;
IAP_CMD =0X00;
IAP_TRIG=0X00;
IAP_ADDRH=0Xff;
IAP_ADDRL=0Xff;
EA=1;
追问
十分感谢,一定给你满意答案,但是有还有两个点没看懂,1.这段程序的存储地址是从哪个地址开始的?2.若果要接受串口的数据,需要把我Receive()这个函数写到那个语句。谢谢了!
追答地址,需要根据具体的芯片来决定,不同的芯片地址空间是不同的。
这三个是子程序,写数据时,收完数据后就可以调用这几个函数。
例如:
Receive(); //接收数据完成
earse_one_sector_in51_ctr(); //先擦除,要给定擦除地址
write_51eeprom_1byte();//写数据,给定地址,和要写的数据
第6句:IAP_CONTR=0X83;中的83是什么意思?是擦除地址么?
追答不是的,这是根据芯片资料里说的来的。
包括5a,a5,都是根据芯片资料来的。不同芯片,这些数据可能不同。
地址是16位的,不是8位的。
数据保存在RAM中,一般52单片机有256字节的RAM,像STC好多单片机都有1k的RAM扩展,也有EEPROM扩展。 参考技术D 其实每次只能发一个字节,储存在串口特定ram里,如果不够大需要外加存储芯片追问
嗯,RAM不是只有128个字节么?没办吧储存数据啊
追答存数据速度也不快呀,还不如找个spi的ram
第5个回答 2013-11-27 数据太多了吧,简易把数据精简追问显示图片,没办法再精简了
51单片机串口通讯
51单片机串口是不是全双工的,我记得书上写的是,但是用中断的话,又想发送和接收同时进行,程序怎么写?按理说在中断里根据是RI=1还是TI=1是可以判断接收引起的中断还是发送引起的中断,但是发送和接收同时进行是不是会出现RI和TI同时为1?对此很困惑,总觉得不是全双工,哪位帮我解释一下,谢谢!
51单片机串口通信来源:维库 作者:
关键字:51单片机 串口通信
这节我们主要讲单片机上串口的工作原理和如何通过程序来对串口进行设置,以及根据所给出的实例实现与PC 机通信。
一、原理简介
51 单片机内部有一个全双工串行接口。什么叫全双工串口呢?一般来说,只能接受或只能发送的称为单工串行;既可接收又可发送,但不能同时进行的称为半双工;能同时接收和发送的串行口称为全双工串行口。串行通信是指数据一位一位地按顺序传送的通信方式,其突出优点是只需一根传输线,可大大降低硬件成本,适合远距离通信。其缺点是传输速度较低。
与之前一样,首先我们来了解单片机串口相关的寄存器。
SBUF 寄存器:它是两个在物理上独立的接收、发送缓冲器,可同时发送、接收数据,可通过指令对SBUF 的读写来区别是对接收缓冲器的操作还是对发送缓冲器的操作。从而控制外部两条独立的收发信号线RXD(P3.0)、TXD(P3.1),同时发送、接收数据,实现全双工。
串行口控制寄存器SCON(见表1) 。
表1 SCON寄存器
表中各位(从左至右为从高位到低位)含义如下。
SM0 和SM1 :串行口工作方式控制位,其定义如表2 所示。
表2 串行口工作方式控制位
其中,fOSC 为单片机的时钟频率;波特率指串行口每秒钟发送(或接收)的位数。
SM2 :多机通信控制位。 该仅用于方式2 和方式3 的多机通信。其中发送机SM2 = 1(需要程序控制设置)。接收机的串行口工作于方式2 或3,SM2=1 时,只有当接收到第9 位数据(RB8)为1 时,才把接收到的前8 位数据送入SBUF,且置位RI 发出中断申请引发串行接收中断,否则会将接受到的数据放弃。当SM2=0 时,就不管第位数据是0 还是1,都将数据送入SBUF,并置位RI 发出中断申请。工作于方式0 时,SM2 必须为0。
REN :串行接收允许位:REN =0 时,禁止接收;REN =1 时,允许接收。
TB8 :在方式2、3 中,TB8 是发送机要发送的第9 位数据。在多机通信中它代表传输的地址或数据,TB8=0 为数据,TB8=1 时为地址。
RB8 :在方式2、3 中,RB8 是接收机接收到的第9 位数据,该数据正好来自发送机的TB8,从而识别接收到的数据特征。
TI :串行口发送中断请求标志。当CPU 发送完一串行数据后,此时SBUF 寄存器为空,硬件使TI 置1,请求中断。CPU 响应中断后,由软件对TI 清零。
RI :串行口接收中断请求标志。当串行口接收完一帧串行数据时,此时SBUF 寄存器为满,硬件使RI 置1,请求中断。CPU 响应中断后,用软件对RI 清零。
电源控制寄存器PCON(见表3) 。
表3 PCON寄存器
表中各位(从左至右为从高位到低位)含义如下。
SMOD :波特率加倍位。SMOD=1,当串行口工作于方式1、2、3 时,波特率加倍。SMOD=0,波特率不变。
GF1、GF0 :通用标志位。
PD(PCON.1) :掉电方式位。当PD=1 时,进入掉电方式。
IDL(PCON.0) :待机方式位。当IDL=1 时,进入待机方式。
另外与串行口相关的寄存器有前面文章叙述的定时器相关寄存器和中断寄存器。定时器寄存器用来设定波特率。中断允许寄存器IE 中的ES 位也用来作为串行I/O 中断允许位。当ES = 1,允许 串行I/O 中断;当ES = 0,禁止串行I/O 中断。中断优先级寄存器IP的PS 位则用作串行I/O 中断优先级控制位。当PS=1,设定为高优先级;当PS =0,设定为低优先级。
波特率计算:在了解了串行口相关的寄存器之后,我们可得出其通信波特率的一些结论:
① 方式0 和方式2 的波特率是固定的。
在方式0 中, 波特率为时钟频率的1/12, 即fOSC/12,固定不变。
在方式2 中,波特率取决于PCON 中的SMOD 值,即波特率为:
当SMOD=0 时,波特率为fosc/64 ;当SMOD=1 时,波特率为fosc/32。
② 方式1 和方式3 的波特率可变,由定时器1 的溢出率决定。
当定时器T1 用作波特率发生器时,通常选用定时初值自动重装的工作方式2( 注意:不要把定时器的工作方式与串行口的工作方式搞混淆了)。其计数结构为8 位,假定计数初值为Count,单片机的机器周期为T,则定时时间为(256 ?Count)×T 。从而在1s内发生溢出的次数(即溢出率)可由公式(1)所示:
从而波特率的计算公式由公式(2)所示:
在实际应用时,通常是先确定波特率,后根据波特率求T1 定时初值,因此式(2)又可写为:
51单片机串口通讯
二、电路详解
下面就对图1 所示电路进行详细说明。
图1 串行通信实验电路图
最小系统部分(时钟电路、复位电路等)第一讲已经讲过,在此不再叙述。我们重点来了解下与计算机通信的RS-232 接口电路。可以看到,在电路图中,有TXD 和RXD 两个接收和发送指示状态灯,此外用了一个叫MAX3232 的芯片,那它是用来实现什么的呢?首先我们要知道计算机上的串口是具有RS-232 标准的串行接口,而RS-232 的标准中定义了其电气特性:高电平“1”信号电压的范围为-15V~-3V,低电平“0”
信号电压的范围为+3V~+15V。可能有些读者会问,它为什么要以这样的电气特性呢?这是因为高低电平用相反的电压表示,至少有6V 的压差,非常好的提高了数据传输的可靠性。由于单片机的管脚电平为TTL,单片机与RS-232 标准的串行口进行通信时,首先要解决的便是电平转换的问题。一般来说,可以选择一些专业的集成电路芯片,如图中的MAX3232。MAX3232 芯片内部集成了电压倍增电路,单电源供电即可完成电平转换,而且工作电压宽,3V~5.5V 间均能正常工作。其典型应用如图中所示,其外围所接的电容对传输速率有影响,在试验套件中采用的是0.1μF。
值得一提的是MAX3232 芯片拥有两对电平转换线路,图中只用了一路,因此浪费了另一路,在一些场合可以将两路并联以获得较强的驱动抗干扰能力。此外,我们有必要了解图中与计算机相连的DB-9 型RS-232的引脚结构(见图2)。
图2 DB-9连接器接口图
其各管脚定义如下(见表4)。
表4 DB-9型接口管脚定义
三、程序设计
本讲设计实例程序如下:
#include "AT89X52.h" (1)
void Init_Com(void) ( 2)
TMOD = 0x20; ( 3)
PCON = 0x00; ( 4)
SCON = 0x50; ( 5)
TH1 = 0xE8; ( 6)
TL1 = 0xE8; ( 7)
TR1 = 1; ( 8)
void main(void) ( 9)
unsigned char dat; ( 10)
Init_Com(); ( 11)
while(1) ( 12)
程序详细说明:
(1)头文件包含。
(2)声明串口初始化程序。
(3)设置定时器1 工作在模式2,自动装载初值(详见第二讲)。
(4)SMOD 位清0,波特率不加倍。
(5)串行口工作在方式1,并允许接收。
(6)定时器1 高8 位赋初值。波特率为1200b/s(7)定时器1 低8 位赋初值。
(8)启动定时器。
(9)主函数。
(10)定义一个字符型变量。
(11)初始化串口。
(12)死循环。
(13)如果接收到数据。
(14)将接收到的数据赋给之前定义的变量。
(15)将接收到的值输出到P0 口。
(16)对接收标志位清0,准备再次接收。
(17)将接收到的数据又发送出去。
(18)查询是否发送完毕。
(19)对发送标志位清0。
四、调试要点与实验现象
接好硬件,通过冷启动方式将程序所生成的。hex文件下载到单片机运行后,打开串口调试助手软件,设置好波特率1200,复位单片机,然后在通过串口调试助手往单片机发送数据(见图3),可以观察到在接收窗口有发送的数据显示,此外电路板上的串行通信指示灯也会闪烁,P0 口所接到LED 灯会闪烁所接收到的数据。
图3 串口软件调试界面
另外串口调试助手软件使用时应注意的是,如果单片机开发板采用串口下载而且和串口调试助手是使用同一串口,则在打开串口软件的同时不能给单片机下载程序,如需要下载,请首先点击“关闭串口”,做发送实验的时候,注意如果选中16 进制发送的就是数字或者字母的16 进制数值,比如发送“0”,实际接收的就应该是0x00,如果不选中,默认发送的是ASCII 码值,此时发送“0”,实际接收的就应该是0x30,这点可以通过观察板子P0 口上的对应的LED 指示出来。
五、总结
本讲介绍了单片机串口通信的原理并给出了实例,通过该讲,读者可以了解和掌握51 单片机串口通信的原理与应用流程,利用串口通信,单片机可以与计算机相连,也可以单片机互联或者多个单片机相互通信组网等,在实际的工程应用中非常广泛。从学习的角度来说,熟练的利用串口将单片机系统中的相关信息显示在计算机上可以很直观方便的进行调试和开发。 参考技术A 请看程序:的确是全双工的,发送和接收可以同时进行,区别只是程序不一样
#include<reg52.h>
#include "intrins.h"
typedef unsigned char BYTE; //使用typedef为现有类型创建别名,定义易于记忆的类型名
typedef unsigned int WORD;
#define FOSC 18432000L
#define BAUD 9600
#define NONE_PARITY 0
#define ODD_PARITY 1
#define EVEN_PARITY 2
#define MARK_PARITY 3
#define SPACE_PARITY 4
#define PARITYBIT EVEN_PARITY
sbit bit9=P2^2;
bit busy;
void SendData(BYTE dat);
void SendString(char *s);
void main()
#if(PARITYBIT==NONE_PARITY)
SCON=0x50;
#elif(PARITYBIT==ODD_PARITY)||(PARITYBIT==EVEN_PARITY)||(PARITYBIT==MARK_PARITY)
SCON=0x50;
#elif(PARITYBIT==SPACE_PARITY)
SCON=0x50;
#endif
TMOD=0x20;
TH1=TL1=-(FOSC/12/32/BAUD);
TR1=1;
ES=1;
EA=1;
SendString("STC89-90xx\r\nUart Test!\r\n");
while(1);
void Uart_Isr()interrupt 4 using 1
if(RI)
RI=0;
P0=SBUF;
bit9=RB8;
if(TI)
TI=0;
busy=0;
void SendData(BYTE dat)
while(busy);
ACC=dat;
if(P)
#if(PARITYBIT==ODD_PARITY)
TB8=0;
#elif(PARITYBIT==EVEN_PARITY)
TB8=1;
#endif
else
#if(PARITYBIT==ODD_PARITY)
TB8=1;
#elif(PARITYBIT==EVEN_PARITY)
TB8=0;
#endif
busy=1;
SBUF=ACC;
void SendString(char *s)
while(*s)
SendData(*s++);
以上是关于52单片机如何对PC串口发送来的数据进行存储,存储在哪?RAM还是ROM中?的主要内容,如果未能解决你的问题,请参考以下文章