51单片机多机通信电路图及C语言程序

发布者:真瓷堂最新更新时间:2021-02-07 来源: eefocus关键字:51单片机  多机通信  C语言程序 手机看文章 扫描二维码
随时随地手机看文章

多机通信电路图

此处,U1作为主机,U2为从机1,U3为从机2。

多机通信C语言程序

(1)主机程序

#include

#include

#define _SUCC_ 0x0f//数据传送成功

#define _ERR_ 0xf0//数据传送失败

unsigned charTable[9]={0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39};

unsigned char Buff[20]; //数据缓冲区

unsigned char temp=0xff;

sbit KEY1=P1^6;

sbit KEY2=P1^7;

//unsigned char addr;

//延时1ms函数

void delay_1ms(unsigned int t)

{

unsigned int x,y;

for(x=t;x>0;x--)

for(y=110;y>0;y--);

}

//缓冲区初始化

void Buff_init()

{

unsigned chari; //将Table里的数据放到缓冲区里

for(i=0;i

{

Buff= Table;

delay_1ms(100);

}

}

//串口初始化函数

void serial_init()

{

TMOD=0x20; //定时器1工作于方式2

TH1=0xfd;

TL1=0xfd; //波特率为9600

PCON=0;

SCON=0xd0; //串口工作于方式3

TR1=1; //开启定时器

TI=0;

RI=0;

}

//发送数据函数

void SEND_data(unsigned char *Buff)

{

unsigned char i;

unsigned char lenth;

unsigned char check;

lenth=strlen(Buff); //计算数据长度

check=lenth;

TI=0; //发送数据长度

TB8=0; //发送数据帧

SBUF=lenth;

while(!TI);

TI=0;

for(i=0;i

{

check=check^Buff;

TB8=0;

SBUF=Buff;

while(!TI);

TI=0;

}

TB8=0; //发送校验字节

SBUF=check;

while(!TI);

TI=0;

}

//向指定从机地址发送数据

void ADDR_data(unsigned addr)

{

while(temp!=addr) //主机等待从机返回其地址作为应答信号

{

TI=0; //发送从机地址

TB8=1; //发送地址帧

SBUF=addr;

while(!TI);

TI=0;

RI=0;

while(!RI);

temp=SBUF;

RI=0;

}

temp=_ERR_; //主机等待从机数据接收成功信号

while(temp!=_SUCC_)

{

SEND_data(Buff);

RI=0;

while(!RI);

temp=SBUF;

RI=0;

}

}

void main()

{

Buff_init();

serial_init();

while(1)

{

if(KEY1==0)

{

delay_1ms(5);

if(KEY1==0)

{

while(!KEY1);

ADDR_data(0x01);

}

}

if(KEY2==0)

{

delay_1ms(5);

if(KEY2==0)

{

while(!KEY2);

ADDR_data(0x02);

}

}

}

}

(2)从机1程序

#include

#include

#defineaddr 0x01//从机1的地址

#define _SUCC_ 0x0f//数据传送成功

#define _ERR_ 0xf0//数据传送失败

unsigned char aa=0xff;//主机与从机之间通信标志

unsigned char Buff[20];//数据缓冲区

//串口初始化函数 www.dgzj.com

void serial_init()

{

TMOD=0x20; //定时器1工作于方式2

TH1=0xfd;

TL1=0xfd; //波特率为9600

PCON=0;

SCON=0xd0; //串口工作于方式3

TR1=1; //开启定时器

TI=0;

RI=0;

}

//接收数据函数

unsigned char RECE_data(unsigned char *Buff)

{

unsigned char i,temp;

unsigned char lenth;

unsigned char check;

RI=0; //接收数据长度

while(!RI);

if(RB8==1) //若接收到地址帧,则返回0xfe

return 0xfe;

lenth=SBUF;

RI=0;

check=lenth;

for(i=0;i

{

while(!RI);

if(RB8==1) //若接收到地址帧,则返回0xfe

return0xfe;

Buff=SBUF;

check=check^(Buff);

RI=0;

}

while(!RI); //接收校验字节

if(RB8==1) //若接收到地址帧,则返回0xfe

return 0xfe;

temp=SBUF;

RI=0;

check=temp^check; //将从主机接收到的校验码与自己计算的校验码比对

if(check!=0) //校验码不一致,表明数据接收错误,向主机发送错误信号,函数返回0xff

{

TI=0;

TB8=0;

SBUF=_ERR_;

while(!TI);

TI=0;

return 0xff;

}

TI=0; //校验码一致,表明数据接收正确,向主机发送成功信号,函数返回0x00

TB8=0;

SBUF=_SUCC_;

while(!TI);

TI=0;

return 0;

}

void main()

{

serial_init();

while(1)

{

SM2=1; //接收地址帧

while(aa!=addr) //从机等待主机请求自己的地址

{

RI=0;

while(!RI);

aa=SBUF;

RI=0;

}

TI=0; //一旦被请求,从机返回自己的地址作为应答,等待接收数据

TB8=0;

SBUF=addr;

while(!TI);

TI=0;

SM2=0; //接收数据帧

aa=0xff; //从机接收数据,并将数据保存到数据缓冲区

while(aa==0xff)

{

aa=RECE_data(Buff);

}

if(aa==0xfe)

continue;

P1=Buff[1]; //查看接收到的数据

}

}

(3)从机2程序

#include

#include

#defineaddr 0x02//从机2的地址

#define _SUCC_ 0x0f//数据传送成功

#define _ERR_ 0xf0//数据传送失败

unsigned char aa=0xff;//主机与从机之间通信标志

unsigned char Buff[20];//数据缓冲区

//串口初始化函数

void serial_init()

{

TMOD=0x20; //定时器1工作于方式2

TH1=0xfd;

TL1=0xfd; //波特率为9600

PCON=0;

SCON=0xd0; //串口工作于方式3

TR1=1; //开启定时器

TI=0;

RI=0;

}

//接收数据函数

unsigned char RECE_data(unsigned char *Buff)

{

unsigned char i,temp;

unsigned char lenth;

unsigned char check;

RI=0; //接收数据长度

while(!RI);

if(RB8==1) //若接收到地址帧,则返回0xfe

return 0xfe;

lenth=SBUF;

RI=0;

check=lenth;

for(i=0;i

{

while(!RI);

if(RB8==1) //若接收到地址帧,则返回0xfe

return0xfe;

Buff=SBUF;

check=check^(Buff);

RI=0;

}

while(!RI); //接收校验字节

if(RB8==1) //若接收到地址帧,则返回0xfe

return 0xfe;

temp=SBUF;

RI=0;

check=temp^check; //将从主机接收到的校验码与自己计算的校验码比对

if(check!=0) //校验码不一致,表明数据接收错误,向主机发送错误信号,函数返回0xff

{

TI=0;

TB8=0;

SBUF=_ERR_;

while(!TI);

TI=0;

return 0xff;

}

TI=0; //校验码一致,表明数据接收正确,向主机发送成功信号,函数返回0x00

TB8=0;

SBUF=_SUCC_;

while(!TI);

TI=0;

return 0;

}

void main()

{

serial_init();

while(1)

{

SM2=1; //接收地址帧

while(aa!=addr) //从机等待主机请求自己的地址

{

RI=0;

while(!RI);

aa=SBUF;

RI=0;

}

TI=0; //一旦被请求,从机返回自己地址作为应答,等待接收数据

TB8=0;

SBUF=addr;

while(!TI);

TI=0;

SM2=0; //接收数据帧

aa=0xff; //从机接收数据,并将数据保存到数据缓冲区

while(aa==0xff)

{

aa=RECE_data(Buff);

}

if(aa==0xfe)

continue;

P1=Buff[2]; //查看接收到的数据

}

}


关键字:51单片机  多机通信  C语言程序 引用地址:51单片机多机通信电路图及C语言程序

上一篇:51单片机8位数码管电子时钟仿真图及源代码
下一篇:51单片机双机通信硬件电路图及C程序

推荐阅读最新更新时间:2024-11-17 04:35

C51单片机中断函数的修饰方法
interruptm修饰符 C51中断函数必须通过该修饰符进行修饰。在C51程序设计中,当函数定义时用了interruptm修饰符,系统编译时把对应函数转化为中断函数,自动加上程序头段和尾段,并按51系统中断的处理方式自动把它安排在程序存储器中的相应位置。 在该修饰符中,m的取值为0~31,对应的中断情况如下: 0——外部中断0 1——定时/计数器T0 2——外部中断1 3——定时/计数器T1 4——串行口中断 5——定时/计数器T2 其它值预留。 C51编译器从绝对地址8m+3处产生一个中断向量,其中m为中断号,也即interrupt后面的数字。该向量包含一个到中断函数入口地址的绝对跳转。 【例】编写一个用于统计外中
[单片机]
C<font color='red'>51单片机</font>中断函数的修饰方法
51单片机中断处理函数能否影响全局变量
在用keil4.60写51的定时器中断处理函数的时候,遇到了这么一个纠结的问题,就是我定义了一个全局变量temp,一旦触发定时器中断进入处理函数时将temp=123,在开中断前temp=888,我本用keil调试看看里边的变量变化情况的,无意中发现keil竟然显示不了全局变量,只能显示当前调用函数的局部变量,这样一来靠调试我是完全不知道全局变量的值变成了什么。上网查了一下,竟然有人说中断处理函数不能改变全局变量的值,决定不相信,做个测试就知道了。昨天实验室没有51开发板来做测试,就只好先画了个proteus的仿真电路图,把程序下载进去看看运行的结果。 如果处理函数可以改变全局变量,led2亮,否则led1亮 #includ
[单片机]
<font color='red'>51单片机</font>中断处理函数能否影响全局变量
51单片机设计方案TOP10(六)
传统的伏特表在我们的日常生活及科学研究中起到了其独特的作用,但是在科学技术日新月异、集成芯片在日常生活中的应用越来越广泛的今天显得比较落伍:①它们的量程往往在出厂以前就限定好的,不能根据具体使用场合进行相应调整;②测量精度有限;③不能够将测量结果用语音播放出来。本文将介绍一种由单片机最小系统、模-数转换电路 、语音电路、LED显示电路组成的单片机式语音播报伏特表。    1、硬件设计   整个系统的组成可以分成四大部分:单片机、模-数转换电路、语音电路、LED显示电路。下面就主要的部分进行具体介绍。   1.1、单片机   目前流行的单片机很多,其中89C51自带有片内ROM和一定数量的RAM,一般不需要扩展片外的存储器,并且
[模拟电子]
<font color='red'>51单片机</font>设计方案TOP10(六)
51单片机】静态与动态点亮数码管
静态: span style= font-family:Microsoft YaHei;font-size:18px; #include reg52.h sbit dula=P2^6; //数码管寄存器 sbit wela=P2^7; //位寄存器 void main() { wela=1; P0=0xf8; wela=0; //确定几位数码管亮,且使其状态保持 dula=1; P0=0x3f; //显示的数字 dula=0; while(1); //保持常亮 } /span 动态: span style= font-family:Microsoft YaHei;font-si
[单片机]
详解基于51单片机的small rtos
陈明计,这个人有必要要认识下,因为small rtos 是他写的,他根据ucos的原理写的。这个small rtos是可以再51单片机上运行的,但是受ARM内存的限制。 作为单片机开发的时刻都得想着内存的问题,因为51芯片资源有限。 好了不废话开始学习的旅程 首先你可以在51hei下个small rtos 源代码 http://www.51hei.com/f/small_rtos1.12.1.zip 或者跟我一步一步写。 keil51的工具编译代码后会生成一个.m51的文件,这个文件要学会去看,因为他把你的一些内存分配的地址和函数的地址都会以列表显示出来。 操作系统的任务其实都是一个死循环。我们写的操作系统其实就是把$
[单片机]
详解基于<font color='red'>51单片机</font>的small rtos
51单片机利用74HC595驱动数码管
基于51单片机利用八个数码管花样显示如下: xxx11xxx→xx2222xx→x333333x→44444444→x555555x→xx6666xx→x777777x→88888888 每个状态各一秒,显示反复循环,其中x表示对应的数码管熄灭 问题补充:是共阳极的数码管,利用595芯片驱动的。采用C语言。 =================================================== 最佳答案:程序设计完毕,采用PROTEUS仿真截图如下。 程序如下: //============================================= #include reg52.h #define u
[单片机]
<font color='red'>51单片机</font>利用74HC595驱动数码管
基于51单片机定时器0计时计数器1计数的霍尔传感器精确测速
#include reg52.h #include intrins.h #define uchar unsigned char #define uint unsigned int sbit RW=P2^6;//定义LCD1602液晶显示器读写端 sbit RS=P2^5;//定义LCD1602液晶显示器数据命令端 sbit EN=P2^7;//定义LCD1602液晶显示器使能端 sbit qiting=P3^0;//定义启停按键 uchar qitingnum;//定义启停次数变量 uint count;//定时器0定时中断次数变量 unsigned long speed=0;//定义速度变量 uchar co
[单片机]
基于<font color='red'>51单片机</font>定时器0计时计数器1计数的霍尔传感器精确测速
51单片机(三)—— 51单片机集成开发环境介绍
一、开发环境安装 能够进行51单片机的集成开发环境比较多,比较常用的是Keil C51,可以从相应的网站上下载,如下图所示为Keil C51的安装文件 点击“Setup.exe”,出现如下图所示的窗口。 点击上图中的“Full Version”,出现如下图所示的窗口 点击“Next”,出现如下图所示的窗口。 点击“Yes”,出现如下图所示的窗口 这个页面用来设置Keil的安装位置,默认安装在C盘,如果需要安装在其它位置,请点击后边的“Browse…”来重新设置安装路径。选好之后点击“Next”,出现如下图所示的窗口。 这一步需要填写序列号。这个序列号,可以在网上找一下,在上图中填写好序列号,如下图所示。
[单片机]
<font color='red'>51单片机</font>(三)—— <font color='red'>51单片机</font>集成开发环境介绍
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved