找回密码
 立即注册

QQ登录

只需一步,快速开始

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

超声波测距C#上位机源码与单片机下位机PCB原理图

[复制链接]
跳转到指定楼层
楼主
上位机(含源码):


Altium Designer画的原理图和PCB图如下:(51hei附件中可下载工程文件)


单片机源程序如下:
  1. #include <reg52.h>                 //调用单片机头文件
  2. #define uchar unsigned char  //无符号字符型 宏定义        变量范围0~255
  3. #define uint  unsigned int         //无符号整型 宏定义        变量范围0~65535
  4. #include <intrins.h>
  5. uchar a_a;

  6. //数码管段选定义      0     1    2    3    4    5        6         7          8           9       
  7. uchar code smg_du[]={0x28,0xee,0x32,0xa2,0xe4,0xa1,0x21,0xea,0x20,0xa0,
  8.                                            0x60,0x25,0x39,0x26,0x31,0x71,0xff};         //断码


  9. uchar dis_smg[8]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8};

  10. //数码管位选定义
  11. sbit smg_we1 = P3^4;            //数码管位选定义
  12. sbit smg_we2 = P3^5;
  13. sbit smg_we3 = P3^6;
  14. sbit smg_we4 = P3^7;

  15. sbit c_send   = P3^2;                //超声波发射
  16. sbit c_recive = P3^3;                //超声波接收

  17. sbit beep = P2^3;   //蜂鸣器IO口定义
  18. uchar smg_i = 3;    //显示数码管的个位数
  19. bit flag_300ms ;

  20. long distance;                //距离
  21. uint set_d;                    //距离
  22. uchar flag_csb_juli;    //超声波超出量程
  23. uint  flag_time0;       //用来保存定时器0的时候的

  24. uchar menu_1;           //菜单设计的变量
  25. //串口接收用到的变量
  26. uchar receive[4];   
  27. uchar jyh=0;                       
  28. uchar uart_flag=0;   
  29. uchar count=0;
  30. //串口发送用到的变量
  31. uchar send[6];
  32. uchar fsxb=0;
  33. /***********************1ms延时函数*****************************/
  34. void delay_1ms(uint q)
  35. {
  36.         uint i,j;
  37.         for(i=0;i<q;i++)
  38.                 for(j=0;j<120;j++);
  39. }

  40. /***********************处理距离函数****************************/
  41. void smg_display()
  42. {
  43.         dis_smg[0] = smg_du[distance % 10];
  44.         dis_smg[1] = smg_du[distance / 10 % 10];
  45.         dis_smg[2] = smg_du[distance / 100 % 10] & 0xdf;
  46. }


  47. #define RdCommand 0x01 //定义ISP的操作命令
  48. #define PrgCommand 0x02
  49. #define EraseCommand 0x03
  50. #define Error 1
  51. #define Ok 0
  52. #define WaitTime 0x01 //定义CPU的等待时间
  53. sfr ISP_DATA=0xe2;  //寄存器申明
  54. sfr ISP_ADDRH=0xe3;
  55. sfr ISP_ADDRL=0xe4;
  56. sfr ISP_CMD=0xe5;
  57. sfr ISP_TRIG=0xe6;
  58. sfr ISP_CONTR=0xe7;

  59. /* ================ 打开 ISP,IAP 功能 ================= */
  60. void ISP_IAP_enable(void)
  61. {
  62.          EA = 0;       /* 关中断   */
  63.          ISP_CONTR = ISP_CONTR & 0x18;       /* 0001,1000 */
  64.          ISP_CONTR = ISP_CONTR | WaitTime; /* 写入硬件延时 */
  65.          ISP_CONTR = ISP_CONTR | 0x80;       /* ISPEN=1  */
  66. }
  67. /* =============== 关闭 ISP,IAP 功能 ================== */
  68. void ISP_IAP_disable(void)
  69. {
  70.          ISP_CONTR = ISP_CONTR & 0x7f; /* ISPEN = 0 */
  71.          ISP_TRIG = 0x00;
  72.          EA   =   1;   /* 开中断 */
  73. }
  74. /* ================ 公用的触发代码 ==================== */
  75. void ISPgoon(void)
  76. {
  77.          ISP_IAP_enable();   /* 打开 ISP,IAP 功能 */
  78.          ISP_TRIG = 0x46;  /* 触发ISP_IAP命令字节1 */
  79.          ISP_TRIG = 0xb9;  /* 触发ISP_IAP命令字节2 */
  80.          _nop_();
  81. }
  82. /* ==================== 字节读 ======================== */
  83. unsigned char byte_read(unsigned int byte_addr)
  84. {
  85.         EA = 0;
  86.          ISP_ADDRH = (unsigned char)(byte_addr >> 8);/* 地址赋值 */
  87.          ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
  88.          ISP_CMD   = ISP_CMD & 0xf8;   /* 清除低3位  */
  89.          ISP_CMD   = ISP_CMD | RdCommand; /* 写入读命令 */
  90.          ISPgoon();       /* 触发执行  */
  91.          ISP_IAP_disable();    /* 关闭ISP,IAP功能 */
  92.          EA  = 1;
  93.          return (ISP_DATA);    /* 返回读到的数据 */
  94. }
  95. /* ================== 扇区擦除 ======================== */
  96. void SectorErase(unsigned int sector_addr)
  97. {
  98.          unsigned int iSectorAddr;
  99.          iSectorAddr = (sector_addr & 0xfe00); /* 取扇区地址 */
  100.          ISP_ADDRH = (unsigned char)(iSectorAddr >> 8);
  101.          ISP_ADDRL = 0x00;
  102.          ISP_CMD = ISP_CMD & 0xf8;   /* 清空低3位  */
  103.          ISP_CMD = ISP_CMD | EraseCommand; /* 擦除命令3  */
  104.          ISPgoon();       /* 触发执行  */
  105.          ISP_IAP_disable();    /* 关闭ISP,IAP功能 */
  106. }
  107. /* ==================== 字节写 ======================== */
  108. void byte_write(unsigned int byte_addr, unsigned char original_data)
  109. {
  110.          EA  = 0;
  111. //         SectorErase(byte_addr);
  112.          ISP_ADDRH = (unsigned char)(byte_addr >> 8);  /* 取地址  */
  113.          ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
  114.          ISP_CMD  = ISP_CMD & 0xf8;    /* 清低3位 */
  115.          ISP_CMD  = ISP_CMD | PrgCommand;  /* 写命令2 */
  116.          ISP_DATA = original_data;   /* 写入数据准备 */
  117.          ISPgoon();       /* 触发执行  */
  118.          ISP_IAP_disable();     /* 关闭IAP功能 */
  119.          EA =1;
  120. }


  121. /******************把数据保存到单片机内部eeprom中******************/
  122. void write_eeprom()
  123. {
  124.         SectorErase(0x2000);
  125.         byte_write(0x2000, set_d % 256);
  126.         byte_write(0x2001, set_d / 256);
  127.         byte_write(0x2058, a_a);       
  128. }

  129. /******************把数据从单片机内部eeprom中读出来*****************/
  130. void read_eeprom()
  131. {
  132.         set_d  = byte_read(0x2001);
  133.         set_d <<= 8;
  134.         set_d  |= byte_read(0x2000);
  135.         a_a      = byte_read(0x2058);
  136. }

  137. /**************开机自检eeprom初始化*****************/
  138. void init_eeprom()
  139. {
  140.         read_eeprom();                //先读
  141.         if(a_a != 1)                //新的单片机初始单片机内问eeprom
  142.         {
  143.                 set_d = 50;
  144.                 a_a = 1;
  145.                 write_eeprom();           //保存数据
  146.         }       
  147. }

  148. /********************独立按键程序*****************/
  149. uchar key_can;         //按键值

  150. void key()         //独立按键程序
  151. {
  152.         static uchar key_new;
  153.         key_can = 20;                   //按键值还原
  154.         P2 |= 0x07;
  155.         if((P2 & 0x07) != 0x07)                //按键按下
  156.         {
  157.                 delay_1ms(1);                     //按键消抖动
  158.                 if(((P2 & 0x07) != 0x07) && (key_new == 1))
  159.                 {                                                //确认是按键按下
  160.                         key_new = 0;
  161.                         switch(P2 & 0x07)
  162.                         {
  163.                                 case 0x06: key_can = 3; break;           //得到k2键值
  164.                                 case 0x05: key_can = 2; break;           //得到k3键值
  165.                                 case 0x03: key_can = 1; break;           //得到k4键值
  166.                         }
  167.                 }                       
  168.         }
  169.         else
  170.                 key_new = 1;       
  171. }

  172. /****************按键处理显示函数***************/
  173. void key_with()
  174. {
  175.         if(key_can == 1)                //设置键
  176.         {
  177.                 menu_1 ++;
  178.                 if(menu_1 >= 2)
  179.                 {
  180.                         menu_1 = 0;
  181.                         smg_i = 3;                //只显示3位数码管
  182.                 }
  183.                 if(menu_1 == 1)
  184.                 {
  185.                         smg_i = 4;           //只显示4位数码管
  186.                 }
  187.         }
  188.         if(menu_1 == 1)                        //设置报警
  189.         {
  190.                 if(key_can == 2)
  191.                 {
  192.                         set_d ++ ;                //加1
  193.                         if(set_d > 400)
  194.                                 set_d = 400;
  195.                 }
  196.                 if(key_can == 3)
  197.                 {
  198.                         set_d -- ;                //减1
  199.                         if(set_d <= 1)
  200.                                 set_d = 1;
  201.                 }
  202.                 dis_smg[0] = smg_du[set_d % 10];                   //取小数显示
  203.                 dis_smg[1] = smg_du[set_d / 10 % 10] ;         //取个位显示
  204.                 dis_smg[2] = smg_du[set_d / 100 % 10] & 0xdf ; //取十位显示
  205.                 dis_smg[3] = 0x60;                //a
  206.                 write_eeprom();                           //保存数据
  207.         }       
  208. }  

  209. /****************报警函数***************/
  210. void clock_h_l()
  211. {
  212.         static uchar value;
  213.         if(distance <= set_d)
  214.         {
  215.                 value ++;  //消除实际距离在设定距离左右变化时的干扰
  216.                 if(value >= 2)
  217.                 {
  218.                         beep = ~beep;           //蜂鸣器报警       
  219.                 }
  220.         }
  221.         else
  222.         {
  223.                 value = 0;
  224.                 beep = 1;                //取消报警
  225.         }       
  226. }

  227. /***********************数码位选函数*****************************/
  228. void smg_we_switch(uchar i)
  229. {
  230.         switch(i)
  231.         {
  232.                 case 0: smg_we1 = 0;  smg_we2 = 1; smg_we3 = 1;  smg_we4 = 1; break;
  233.                 case 1: smg_we1 = 1;  smg_we2 = 0; smg_we3 = 1;  smg_we4 = 1; break;
  234.                 case 2: smg_we1 = 1;  smg_we2 = 1; smg_we3 = 0;  smg_we4 = 1; break;
  235.                 case 3: smg_we1 = 1;  smg_we2 = 1; smg_we3 = 1;  smg_we4 = 0; break;
  236.         }       
  237. }

  238. /***********************数码显示函数*****************************/
  239. void display()
  240. {
  241.         static uchar i;   
  242.         i++;
  243.         if(i >= smg_i)
  244.         i = 0;       
  245.         smg_we_switch(i);                 //位选
  246.         P1 = dis_smg[i];                 //段选                
  247. }

  248. /******************小延时函数*****************/
  249. void delay()
  250. {
  251.         _nop_();                            //执行一条_nop_()指令就是1us
  252.         _nop_();
  253.         _nop_();
  254.         _nop_();
  255.         _nop_();
  256.         _nop_();
  257.         _nop_();
  258.         _nop_();
  259.         _nop_();  
  260.         _nop_();
  261. }


  262. /*********************超声波测距程序*****************************/
  263. void send_wave()
  264. {
  265.         unsigned char flag=0;
  266.         c_send = 1;                           //10us的高电平触发
  267.         delay();
  268.         c_send = 0;         
  269.         //TH0 = 0;                          //给定时器0清零
  270.         TL0 = 0;
  271.         TR0 = 0;                                  //关定时器0定时
  272.         while(!c_recive);                  //当c_recive为零时等待
  273.         TR0=1;
  274.         while(c_recive)                      //当c_recive为1计数并等待
  275.         {
  276.                 while(!TF0);
  277.                 TF0=0;
  278.                 flag++;
  279.         }
  280.         flag_time0 = 256*flag+TL0;
  281.         flag=0;
  282.         if((flag_time0 > 29000))      //当超声波超过测量范围时,显示3个888
  283.                 {
  284.                         TR0 = 0;
  285.                         flag_csb_juli = 2;
  286.                         distance = 888;
  287. //                break;               
  288.                 }
  289.                 else
  290.                 {
  291.                         flag_csb_juli = 1;       
  292.                 }
  293.         if(flag_csb_juli == 1)
  294.         {       
  295.                 TR0=0;                                                         //关定时器0定时
  296.                 distance =flag_time0;                         //读出定时器0的时间
  297.                 distance *= 0.017;               // 0.017 = 340M / 2 = 170M = 0.017M 算出来是米
  298.                 if((distance > 500))                                 //距离 = 速度 * 时间
  299.                 {       
  300.                         distance = 888;                                 //如果大于3.8m就超出超声波的量程
  301.                 }
  302.         }  
  303. }


  304. /*********************定时器0、定时器1初始化******************/
  305. void time_init()          
  306. {
  307.         EA  = 1;                   //开总中断
  308.         TMOD = 0X23;          //定时器0、定时器1工作方式1
  309.         TH0=0x06;
  310.         TL0=0x00;
  311.         TCON=0x50;
  312.         TH1 = 0xF3;
  313.   TL1 = TH1;
  314.         ES=1;
  315.         ET0 = 0;                  //关定时器0中断
  316.         TR0 = 1;                  //允许定时器0定时
  317.         ET1 = 1;                  //开定时器1中断
  318.         TR1 = 1;                  //允许定时器1定时       
  319. }

  320. /******************串口初始化****************************/
  321. void UsartInit()
  322. {
  323.           SCON=0X50;
  324.          
  325.           PCON=0x80;
  326.          
  327.         ES=1;   
  328.           
  329. }
  330. /***************主函数*****************/
  331. void main()
  332. {
  333.         beep = 0;                 //开机叫一声   
  334.         delay_1ms(150);
  335.         P0 = P1 = P2 = P3 = 0xff;           //初始化单片机IO口为高电平
  336.         send_wave();        //测距离函数
  337.         smg_display();        //处理距离显示函数
  338.         time_init();        //定时器初始化程序
  339.         UsartInit();    //串口初始化
  340.         init_eeprom();  //开始初始化保存的数据
  341.         send_wave();        //测距离函数
  342.         send_wave();        //测距离函数
  343.         while(1)
  344.         {                  
  345.                 if(uart_flag==1)
  346.                 {
  347.                         uart_flag=0;
  348.                         if(receive[2]==0x01)
  349.                         {
  350.                                 send[0]=0xaa;
  351.                                 send[1]=0x55;
  352.                                 send[2]=distance % 10;
  353.                                 send[3]=distance / 10 % 10;
  354.                                 send[4]=distance / 100 % 10;
  355.                                 send[5]=send[2]+send[3]+send[4];
  356.                                 fsxb=0;
  357.                                 SBUF=send[fsxb];
  358.                                 fsxb++;
  359.                         }
  360.                        
  361.                 }
  362.                 if(flag_300ms == 1)
  363.                 {               
  364.                         flag_300ms = 0;
  365.                         clock_h_l();    //报警函数
  366.                         if(beep == 1)
  367.                                 send_wave();        //测距离函数
  368.                         if(menu_1 == 0)
  369.                                 smg_display();         //处理距离显示函数
  370.                 }
  371.                 key();                                         //按键函数
  372.                 if(key_can < 10)
  373.                 {
  374.                         key_with();                         //按键处理函数
  375.                 }
  376.         }
  377. }

  378. /*********************定时器1中断服务程序************************/
  379. void time1_int() interrupt 3
  380. {       
  381.         static int value;                         //定时2ms中断一次
  382.         TH0 = 0x06;
  383.         display();                //数码管显示函数
  384.         value++;
  385.         if(value >= 1200)
  386. ……………………

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

所有资料51hei提供下载:
超声波测距pcb.rar (523.39 KB, 下载次数: 34)
超声波测距.zip (2.32 MB, 下载次数: 46)


评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

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

使用道具 举报

沙发
ID:650421 发表于 2020-3-22 00:12 | 只看该作者
很好,拿来学习一下!
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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