一个关于51单片机串口数据发送问题(c语言)

一个关于51单片机串口数据发送问题(c语言),第1张

你的问题是刚判断完帧头就处理数据了

应该判断完帧头后继续接受3个字节的数据再处理

#include <reg52h>

#include <intrinsh>

#define uchar unsigned char

#define uint  unsigned int

int zhibi_js[40], i0, com_dat;

unsigned char a,b,c,flag;

uchar Crcl,Crch;

const uint code CrcTable[832]=

{

0x0000,0x8005,0x800F,0x000A,0x801B,0x001E,0x0014,0x8011,

0x8033,0x0036,0x003C,0x8039,0x0028,0x802D,0x8027,0x0022,

0x8063,0x0066,0x006C,0x8069,0x0078,0x807D,0x8077,0x0072,

0x0050,0x8055,0x805F,0x005A,0x804B,0x004E,0x0044,0x8041,

0x80C3,0x00C6,0x00CC,0x80C9,0x00D8,0x80DD,0x80D7,0x00D2,

0x00F0,0x80F5,0x80FF,0x00FA,0x80EB,0x00EE,0x00E4,0x80E1,

0x00A0,0x80A5,0x80AF,0x00AA,0x80BB,0x00BE,0x00B4,0x80B1,

0x8093,0x0096,0x009C,0x8099,0x0088,0x808D,0x8087,0x0082,

0x8183,0x0186,0x018C,0x8189,0x0198,0x819D,0x8197,0x0192,

0x01B0,0x81B5,0x81BF,0x01BA,0x81AB,0x01AE,0x01A4,0x81A1,

0x01E0,0x81E5,0x81EF,0x01EA,0x81FB,0x01FE,0x01F4,0x81F1,

0x81D3,0x01D6,0x01DC,0x81D9,0x01C8,0x81CD,0x81C7,0x01C2,

0x0140,0x8145,0x814F,0x014A,0x815B,0x015E,0x0154,0x8151,

0x8173,0x0176,0x017C,0x8179,0x0168,0x816D,0x8167,0x0162,

0x8123,0x0126,0x012C,0x8129,0x0138,0x813D,0x8137,0x0132,

0x0110,0x8115,0x811F,0x011A,0x810B,0x010E,0x0104,0x8101,

0x8303,0x0306,0x030C,0x8309,0x0318,0x831D,0x8317,0x0312,

0x0330,0x8335,0x833F,0x033A,0x832B,0x032E,0x0324,0x8321,

0x0360,0x8365,0x836F,0x036A,0x837B,0x037E,0x0374,0x8371,

0x8353,0x0356,0x035C,0x8359,0x0348,0x834D,0x8347,0x0342,

0x03C0,0x83C5,0x83CF,0x03CA,0x83DB,0x03DE,0x03D4,0x83D1,

0x83F3,0x03F6,0x03FC,0x83F9,0x03E8,0x83ED,0x83E7,0x03E2,

0x83A3,0x03A6,0x03AC,0x83A9,0x03B8,0x83BD,0x83B7,0x03B2,

0x0390,0x8395,0x839F,0x039A,0x838B,0x038E,0x0384,0x8381,

0x0280,0x8285,0x828F,0x028A,0x829B,0x029E,0x0294,0x8291,

0x82B3,0x02B6,0x02BC,0x82B9,0x02A8,0x82AD,0x82A7,0x02A2,

0x82E3,0x02E6,0x02EC,0x82E9,0x02F8,0x82FD,0x82F7,0x02F2,

0x02D0,0x82D5,0x82DF,0x02DA,0x82CB,0x02CE,0x02C4,0x82C1,

0x8243,0x0246,0x024C,0x8249,0x0258,0x825D,0x8257,0x0252,

0x0270,0x8275,0x827F,0x027A,0x826B,0x026E,0x0264,0x8261,

0x0220,0x8225,0x822F,0x022A,0x823B,0x023E,0x0234,0x8231,

0x8213,0x0216,0x021C,0x8219,0x0208,0x820D,0x8207,0x0202

};

/

校验查表算法

/

void UpdateCrc(const uchar num)

{

  uint table_addr;

  table_addr=(num ^ Crch);

  Crch=(CrcTable[table_addr] >> 8) ^ Crcl;

  Crcl=(CrcTable[table_addr] & 0x00FF);

}

/

  高\低位置高电平

/

void ResetCrc(void)

{

Crcl=0xFF;

Crch=0xFF;

}   

/

       UART初始化

      波特率:9600

/

void UART_init(void)

{

     SCON = 0x50;        // 10位uart,允许串行接受

 TMOD = 0x21;        // 定时器1工作在方式2(自动重装)

 TH1 = 0xfd;

 TL1 = 0xfd;

 IE  = 0x82;

 TR1 = 1;

 ES=1;

}

/

      UART 发送一字节

/

void UART_send_byte(unsigned char dat)

{

SBUF = dat;

while (!TI);

TI = 0;

}

/

      数组第二位判断

/

unsigned char  panduan ()

        if(zhibi_js[1]==0x00)  return 1;    //第二位是0x00为真                

//判断a值为1

if(zhibi_js[1]==0x80)  return 1; //第二位是0x80为真

else  return 0;                 //否则为假

}

/

7F开头数组接受判断

/

void wr_test(void)

{

   if(flag==2)          //收到7F开头数组  //■■■■■■■■此处有改动           

      {                   

    a=panduan(); //数组第二位判断

        flag=0;         //有效数组标志清零

      }

}

/

        主函数

  对符合要求数组进行校验并串口输出校验码

/

void main(void)  

{  

    UART_init();    //串口初始化    

  while(1)

  {

   wr_test(); //判断是否有符合数组

if(a==1) 

{

 a=0;

 b=(zhibi_js[2]+3);   //由接收数组的第三个元素确定b的值

 ResetCrc();   //置高电平

 for(i0=1;i0<b;i0++) UpdateCrc(zhibi_js[i0]); //对接收数组进行计算

             zhibi_js[i0++]=Crcl;  //得出低8位校验码

             zhibi_js[i0++]=Crch;  //得出高8位校验码

 UART_send_byte(Crcl);  //串口输出低8位校验码

 UART_send_byte(Crch);  //串口输出低8位校验码

}       

     

  }    

}   

 

/

         串口接收

/

void Com_Int(void) interrupt 4

{

    if(RI == 1)                             

        {

            if(flag=0 && SBUF == 0x7F) com_dat=0,flag=1; //接收7F开头的数组//■■■■■■■■此处有改动

            

            zhibi_js[com_dat++] = SBUF;  //串口接收数组

     RI = 0;

     if(flag==1 && com_dat>=4)//■■■■■■■■此处有改动

     { //■■■■■■■■此处有改动

       flag=2; //■■■■■■■■此处有改动

     }            //■■■■■■■■此处有改动     

    

        }         

}

我写的串口通信代码比较长,给几个关键函数你吧,可实现PC与单片机的双工通信

/串口通讯 单片机晶振:110592MHz

T1工作于方式2 波特率=2^smodfosc/32/12/(256-X)/

#include<commonh>

void do_uart(void); //串口接收执字符时的 *** 作

/串口发送字符串 /

void sendstr(uchar p)

{

while(p!='\0')

{ SBUF=p; //待发送的数据写入缓冲区

while(!TI); //等待发送完成

TI=0; //清零发送标志位

p++; //指针加1

cnt_s++; //发送计数

}

}

/串口接收中断函数/

void int_rec(void) interrupt 4 using 2

{

if(RI) //查询接收标志位(有数据发送过来时置为1)

{

RI = 0; //接收标志位清零

// sendchar(SBUF);//让从电脑上传到单片机的数据,传回的电脑显示

do_uart();//对接受到得字符 *** 作

}

}

/串口初始化/

void com_init(uint baud)

{

SCON = 0x50; //串口工作方式为1,串行允许接受

TMOD = 0x21; //定时器1工作在方式2 定时器0工作在方式1

//PCON = 0x80; //SMOD = 1; 波特率加倍

TH1=256-fosc/32/12/baud;

TL1=TH1;

ES = 1; //开串口中断

TR1 = 1; //允许定时器1工作

EA = 1; //开总中断

}

void sendchar(uchar uart_dat) //串口发送字符函数

{

SBUF = uart_dat; //待发送的数据写入缓冲区

while(!TI); //等待发送完成

TI = 0; //清零发送标志位

cnt_s++; //发送计数

}

///////////////////////////////////////////////////////////

// 设计题目 : 基于51单片机的串口通信

// 功能 :

// 说明 : 晶振用12M,

///////////////////////////////////////////////////////////

#include <reg51h> // 12M 晶振

#define uchar unsigned char

#define uint unsigned int

///////////////////////////////////////////////////////////

// Name : 串口发送函数

// In :

// Out :

// Function :

// Explain :

///////////////////////////////////////////////////////////

void send(uchar x)

{

SBUF=x; // 将数据放到缓冲区,发送

while(!TI); // 等待发送完成

TI=0; // 清标志位

}

///////////////////////////////////////////////////////////

// Name : 初始化函数

// In :

// Out :

// Function :

// Explain :

///////////////////////////////////////////////////////////

void init(void)

{

TMOD=0x20; // T1工作在方式2,自动重载

TH1=0xe6;

TL1=0xe6; // 设置波特率为1200

SM0=0;

SM1=1; // 串口方式1

TR1=1; // 打开定时器1

}

///////////////////////////////////////////////////////////

// Name : 主函数

// In :

// Out :

// Function :

// Explain :

///////////////////////////////////////////////////////////

void main(void)

{

uchar i;

init(); // 初始化

while(1)

{

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

{ send(i); }

}

}

可以。

首先你可以用程序判断你待发送的字符的校验和y

设你在校验位要发送的位为x

设校验方式为z

因为 x=y xor z

所以 z=x xor y

即根据校验和y 与 你的要求x 决定用奇校验还是偶校验。

要是没有猜错,你可能是想用校验位来多一个发送位,

达到特殊控制功能。不要忘记,上述方法只能在一个字符

发送完成后才能改变校验方式。

uchar idata SystemBuf[10]; //用于接收

uchar Rx=0; //接收数据条数

uchar code AT[]="YES,IT IS";

void UART_init (void)

{

TMOD=0x20; //用定时器1

PCON=0x00; //波特率不加倍

SCON=0x50; //串行方式1

TH1=0xFD; //9600

TL1=0xFD; //

EA=1; //

ES=1; //

TR1=1;

}

void sendchar(uchar ch) //串口送一个字节

{

SBUF=ch;

while(TI==0);

TI=0;

}

void sendstring(uchar p) //送字符串

{

while(p)

{

sendchar(p);

p++;

}

}

///////

void receive(void) interrupt 4 using 1 //中断

{

if(RI)

{

if(Rx<10) //这儿最多收10个字节

{

SystemBuf[Rx]=SBUF;

Rx++;

}

RI=0;

}

}

///

void main()

{

uchar i;

UART_init();

while(1)

{

if(SystemBuf=='S')

{

sendstring(AT);

for(i=0;i<10;i++) //接收清0

{

SystemBuf[i]=0;

}

Rx=0;

}

}

}

//////////如有小错误,自己调下,手打的,大小写可能不正常,这是标准程序,直接用,

以上就是关于一个关于51单片机串口数据发送问题(c语言)全部的内容,包括:一个关于51单片机串口数据发送问题(c语言)、单片机串口232通信c语言、求一个用C语言编写出单片机甲通过串口向PC机乙发送数据0~9的程序(晶振频率为12MHZ,波特率为1200b/s)等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/zz/9426936.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-04-28
下一篇2023-04-28

发表评论

登录后才能评论

评论列表(0条)

    保存