找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 7686|回复: 4
打印 上一主题 下一主题
收起左侧

单片机双机主从串行通信源程序 注释详细 原理图与Proteus仿真

[复制链接]
跳转到指定楼层
楼主
下面是单片机主从串行通信的proteus仿真原理图(工程文件可到本帖附件中下载):

以上仿真完美实现了单片机双机通信.

单片机主从串行通信源程序如下:
  1. # include <STC12C5A60S2.h>
  2. # define uchar unsigned char
  3. # define vtime        10000   //定时3ms,一帧8*3=24ms,频率=40Hz
  4. # define sub1        0x11   
  5. # define sub2        0x12  
  6. sbit LED11=P1^0;
  7. sbit LED12=P1^1;
  8. sbit LED21=P1^2;
  9. sbit LED22=P1^3;
  10. uchar        over_t=0;
  11. uchar        addr=sub1,sum=0,count1,count2;
  12. //uchar i,j,temp,m=0x01;

  13. uchar        tn=0;  //发送循环变量
  14. uchar        txdv[3];
  15. void main( )
  16. {
  17. /*        //m1:m0  00=标准;   01=推挽;   10=输入;   11=开漏输出
  18.         P2M1 = 0X00;
  19.         P2M0 = 0Xff;
  20.         P1M1 = 0X00;
  21.         P1M0 = 0Xff; //设定P2,P1推挽输出
  22.         P3M1 = 0X0d;
  23.         P3M0 = 0X02;
  24. */        //
  25.         SCON = 0xd0;
  26.         PCON = 0X80;
  27.         TH1 = 0XFd;
  28.         TL1 = 0XFd;
  29.         IT0 = 1; //        中断0为边沿触发
  30.           IT1 = 1; //        中断1为边沿触发
  31.        
  32.           TMOD = 0X21; //设定定时器0为16位计数方式
  33.           TH0 = (65536-vtime )/256;
  34.           TL0 = (65536-vtime )%256; //赋定时器0初值
  35.           ET0 = 1;  //开定时器0中断
  36.            TR0 = 1;  //启动定时器0计数
  37.            TR1 = 1;  //启动定时器1计数
  38.         EX1 = 1;  //开中断1
  39.         EX0 = 1;  //开中断0
  40.         SM2 = 0;
  41.         TB8 = 1;
  42.         ES = 1;
  43.           EA = 1;          //开总中断

  44.         while(1);
  45. }
  46. void t0_isp() interrupt 1
  47. {
  48.         //uchar dm,wx;
  49.           TH0 = (65536-vtime )/256;
  50.           TL0 = (65536-vtime )%256; //赋定时器0初值
  51.         if(over_t != 0)
  52.         {
  53.                 over_t--;
  54.                 if(over_t==30)
  55.                 {       
  56.                         if(addr==sub1)
  57.                         {
  58.                         //        LED11=1;
  59.                                 IE0 = 0;   //中断0的源清零
  60.                                  EX0 = 1;  //开中断0
  61.                         }
  62.                         else
  63.                         {
  64.                         //        LED21=1;
  65.                                 IE1 = 0;   //中断0的源清零
  66.                                  EX1 = 1;  //开中断0
  67.                         }

  68.                 }
  69.                 if(over_t==0)
  70.                 {       
  71.                         if(addr==sub1)
  72.                         {
  73.                                 LED11=1;
  74.                                 //IE0 = 0;   //中断0的源清零
  75.                                  //EX0 = 1;  //开中断0
  76.                         }
  77.                         else
  78.                         {
  79.                                 LED21=1;
  80.                                 //IE1 = 0;   //中断0的源清零
  81.                                  //EX1 = 1;  //开中断0
  82.                         }

  83.                 }

  84.         }
  85. }
  86. void int0_isp() interrupt 0
  87. {
  88.         count1++;
  89.         addr=sub1;
  90.         sum=sub1;
  91.         txdv[0]=addr;
  92.         sum=sum+count1;
  93.         txdv[1]=count1;
  94.         txdv[2]=sum;
  95.         TB8=1;
  96.         SBUF=sub1;
  97.         LED12=1;
  98.         LED11=0;
  99.         EX0 = 0;          //  关中断0
  100.         over_t=50;
  101. }
  102. void int1_isp() interrupt 2
  103. {
  104.         count2++;
  105.         addr=sub2;
  106.         sum=sub2;
  107.         txdv[0]=addr;
  108.         sum=sum+count2;
  109.         txdv[1]=count2;
  110.         txdv[2]=sum;
  111.         TB8=1;
  112.         SBUF=sub2;
  113.         LED22=1;
  114.         LED21=0;
  115.         EX1 = 0;          //  关中断0
  116.         over_t=50;
  117. }
  118. void         uart_isp( ) interrupt 4
  119. {
  120.         if(RI==1)
  121.         {
  122.                 RI=0;
  123.                 if(addr==SBUF)
  124.                 {
  125.                         if(addr==sub1)
  126.                         {
  127.                                 LED12=0;
  128.                         }
  129.                         if(addr==sub2)
  130.                         {
  131.                                 LED22=0;
  132.                         }
  133.                 }
  134.         }
  135.         if(TI==1)
  136.         {
  137.                 tn++;
  138.                 TI=0;
  139.                 if(tn>=3)
  140.                 {
  141.                         tn=0;
  142.                         TB8=1;
  143.                         return;
  144.                 }
  145.                 if(tn<3)
  146.                 {
  147.                         TB8=0;
  148.                         SBUF=txdv[tn];
  149.                 }
  150.         }
  151. }  
  152.   
复制代码
  1. //#include<reg51.h>
  2. # include <STC12C5A60S2.h>
  3. # define uchar unsigned char
  4. # define vtime        6000   //定时3ms,一帧8*3=24ms,频率=40Hz
  5. # define sub1        0x11   
  6. # define sub2        0x12  
  7. sbit key1=P3^2;
  8. sbit key2=P3^3;
  9. uchar        over_t=0;
  10. uchar        addr=sub1,sum=0;
  11. //uchar i,j,temp,m=0x01;

  12. uchar code distable[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0x00,0x40,0x09};
  13. //位选码表
  14. uchar  code numi[]={0xfe,0xfd,0xfb,0xf7};
  15. //显示缓存
  16. uchar V_ram[]={17,16,16,0};
  17. uchar        wi=0;  //位选循环变量
  18. uchar        rn=0;  //接收循环变量
  19. uchar        rxdv[3];
  20. void main( )
  21. {
  22. /*        //m1:m0  00=标准;   01=推挽;   10=输入;   11=开漏输出
  23.         P2M1 = 0X00;
  24.         P2M0 = 0Xff;
  25.         P1M1 = 0X00;
  26.         P1M0 = 0Xff; //设定P2,P1推挽输出
  27.         P3M1 = 0X0d;
  28.         P3M0 = 0X02;
  29. */        //
  30.         SCON = 0xd0;
  31.         PCON = 0X80;
  32.         TH1 = 0XFd;
  33.         TL1 = 0XFd;
  34.         IT0 = 1; //        中断0为边沿触发
  35.           IT1 = 1; //        中断1为边沿触发
  36.        
  37.           TMOD = 0X21; //设定定时器0为16位计数方式
  38.           TH0 = (65536-vtime )/256;
  39.           TL0 = (65536-vtime )%256; //赋定时器0初值
  40.           ET0 = 1;  //开定时器0中断
  41.            TR0 = 1;  //启动定时器0计数
  42.            TR1 = 1;  //启动定时器1计数
  43.         SM2=1;
  44.         if(key1==0)
  45.         {
  46.                 addr=sub1;
  47.                  IE1 = 0;  //中断1的源清零
  48.                   EX1 = 1; //        开启中断1
  49.         }
  50.         if(key2==0)
  51.         {
  52.                 addr=sub2;
  53.                  IE0 = 0;   //中断0的源清零
  54.                 EX0 = 1; //        开启中断0
  55.         }
  56.         V_ram[0]=addr;
  57.         ES=1;
  58.           EA = 1;          //开总中断
  59.         while(1);
  60. }
  61. void t0_isp() interrupt 1
  62. {
  63.         uchar dm,wx;
  64.           TH0 = (65536-vtime )/256;
  65.           TL0 = (65536-vtime )%256; //赋定时器0初值
  66.           dm=distable[V_ram[wi]];        //取显示段码
  67.         wx=numi[wi];                           //取位选码
  68.         P1=0xff;                           //关显示
  69.         P2=dm;                                   //段码赋给P0口
  70.         P1=wx;                                   //点亮位选的那个数码管
  71.         wi++;
  72.         if(wi==4)wi=0;
  73.         if(over_t != 0)over_t--;
  74. }
  75. void int0_isp() interrupt 0
  76. {
  77.         addr=sub1;
  78.         V_ram[0]=addr;
  79.         EX0 = 0;          //  关中断0
  80.         IE1 = 0;  //中断1的源清零
  81.         EX1 = 1;  //开中断1
  82. }
  83. void int1_isp() interrupt 2
  84. {
  85.         addr=sub2;
  86.         V_ram[0]=addr;
  87.         EX1 = 0;          // 关中断1
  88.         IE0 = 0;   //中断0的源清零
  89.         EX0 = 1;  //开中断0
  90. }
  91. void         uart_isp( ) interrupt 4
  92. {
  93.         if(RI==1)
  94.         {
  95.                 over_t=100;
  96.                 RI=0;
  97.                 rxdv[rn]=SBUF;
  98.                 //V_ram[1]=rn;
  99.                 if((rn==0)&&(addr==SBUF))
  100.                 {
  101.                         SM2=0;
  102.                         rn++;
  103. //                        V_ram[1]=rn;
  104.                         sum=SBUF;
  105.                         return;
  106.                 }
  107.                 if(rn==2)
  108.                 {
  109.                         if(sum==SBUF)
  110.                         {
  111.                                 V_ram[1]=rxdv[1]/100;
  112.                                 if(V_ram[1]==0)V_ram[1]=0x10;
  113.                                 V_ram[2]=(rxdv[1]%100)/10;
  114.                                 if((V_ram[1]==16)&&(V_ram[2]==0x00))V_ram[2]=0x10;
  115.                                 V_ram[3]=(rxdv[1]%100)%10;
  116.                                 TB8=0;
  117.                                 SBUF=addr;
  118.                         }
  119.                 //        V_ram[1]=rn;
  120.                         rn=0;
  121.                         SM2=1;
  122.                         return;
  123.                 }
  124.                 if(rn==1)
  125.                 {
  126.                         rn++;
  127.                         sum=sum+SBUF;
  128.                         V_ram[1]=SBUF;


  129. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

最后给大家分享一些我们老师给的一些经典的单片机程序源码, 一共有十多个.都有详细的注释,然大家快速的理解每一行代码的意思。而且有proteus仿真原理图。大家可以直接验证程序的对错.


本系列所有源码打包下载地址(含proteus仿真工程文件和源程序):
http://www.51hei.com/bbs/dpj-82474-1.html

本例程下载:
主从串行通信.rar (88.38 KB, 下载次数: 47)


分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:161951 发表于 2017-5-8 22:36 | 只看该作者
51黑有你更精彩,向你学习,尊敬。
回复

使用道具 举报

板凳
ID:253421 发表于 2017-11-25 22:11 | 只看该作者
有没有汇编程序
回复

使用道具 举报

地板
ID:257286 发表于 2017-12-4 20:46 | 只看该作者
虚心求教学习
回复

使用道具 举报

5#
ID:247161 发表于 2017-12-14 20:16 | 只看该作者
资料很好,
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表