找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2456|回复: 6
收起左侧

基于51智能小车死机问题求教

[复制链接]
ID:265318 发表于 2018-2-9 11:19 | 显示全部楼层 |阅读模式
目前加的有的超声波模块,红外遥控模块,lcd1602模块,温湿度检测模块,以及俩电机。其中前三个模块虽然略有问题好歹还能运行,加入温湿度后程序就会卡在某一值的界面无法动弹。顺便求教如何优化代码已让其更平稳的运行
回复

使用道具 举报

ID:265318 发表于 2018-2-9 11:20 | 显示全部楼层
  1. #include"reg51.h"
  2. #include<intrins.h>       
  3. #define LCD1602_DB P2      //data bus 数据总线
  4. typedef unsigned int u16;          //对数据类型进行声明定义
  5. typedef unsigned char u8;

  6. sbit in1=P3^0;
  7. sbit in2=P3^1;
  8. sbit in3=P3^3;
  9. sbit in4=P3^4;
  10. sbit RX=P0^0;
  11. sbit TX=P0^1;
  12. sbit IRIN=P3^2;
  13. unsigned int  time=0;
  14. unsigned int  timer=0;
  15. unsigned int  DH=0;
  16. unsigned char posit=0;
  17. unsigned long S=0;
  18. unsigned long S1=0;
  19. unsigned long S2=0;
  20. unsigned char Zhu[]=" by Zhu JUN ";
  21. unsigned char Pow[]=" Powed ";
  22. unsigned char code ASCII[17] =    {'0','1','2','3','4','5','6','7','8','9','.','-','M','T','H'};
  23. static unsigned char DisNum = 0; //显示用指针                                  
  24. unsigned char disbuff[4]           ={ 0,0,0,0,};
  25. unsigned char IrValue[6];
  26. unsigned char Tim;
  27. sbit LCD1602_RS = P0^4;
  28. sbit LCD1602_RW = P0^3;
  29. sbit LCD1602_EN = P0^2;
  30. sbit Data=P3^6;
  31. bit flag=0;

  32. /*******************************************************************************
  33. * 函 数 名         : delay
  34. * 函数功能                   : 延时函数,i=1时,大约延时10us
  35. *******************************************************************************/
  36. void delay(u16 i)
  37. {
  38.         while(i--);       
  39. }
  40. void DHT11_delay_us(unsigned char n)
  41. {
  42.     while(--n);
  43. }

  44. void DHT11_delay_ms(unsigned int z)
  45. {
  46.    unsigned int i,j;
  47.    for(i=z;i>0;i--)
  48.       for(j=110;j>0;j--);
  49. }
  50. unsigned char rec_dat[9];   //用于显示的接收数据数组}

  51. void DHT11_start()
  52. {
  53.    Data=1;
  54.    DHT11_delay_us(2);
  55.    Data=0;
  56.    DHT11_delay_ms(20);   //延时18ms以上
  57.    Data=1;
  58.    DHT11_delay_us(30);
  59. }

  60. unsigned char DHT11_rec_byte()      //接收一个字节
  61. {
  62.   unsigned int i,dat=0;
  63.   for(i=0;i<8;i++)    //从高到低依次接收8位数据
  64.    {         
  65.       while(!Data);   ////等待50us低电平过去
  66.       DHT11_delay_us(8);     //延时60us,如果还为高则数据为1,否则为0
  67.       dat<<=1;           //移位使正确接收8位数据,数据为0时直接移位
  68.       if(Data==1)    //数据为1时,使dat加1来接收数据1
  69.          dat+=1;
  70.       while(Data);  //等待数据线拉低   
  71.     }  
  72.     return dat;
  73. }

  74. void DHT11_receive()
  75.       //接收40位的数据
  76. {
  77.     unsigned int R_H,R_L,T_H,T_L,RH,RL,TH,TL,revise;
  78.     DHT11_start();
  79.     if(Data==0)
  80.     {
  81.         while(Data==0);   //等待拉高     
  82.         DHT11_delay_us(40);  //拉高后延时80us
  83.         R_H=DHT11_rec_byte();    //接收湿度高八位  
  84.         R_L=DHT11_rec_byte();    //接收湿度低八位  
  85.         T_H=DHT11_rec_byte();    //接收温度高八位  
  86.         T_L=DHT11_rec_byte();    //接收温度低八位
  87.         revise=DHT11_rec_byte(); //接收校正位

  88.         DHT11_delay_us(25);    //结束

  89.         if((R_H+R_L+T_H+T_L)==revise)      //校正
  90.         {
  91.             RH=R_H;
  92.             RL=R_L;
  93.             TH=T_H;
  94.             TL=T_L;
  95.         }
  96.         /*数据处理,方便显示*/
  97.         rec_dat[0]='0'+(RH/10);
  98.         rec_dat[1]='0'+(RH%10);
  99.         rec_dat[2]='R';
  100.         rec_dat[3]='H';
  101.         rec_dat[4]=' ';
  102.         rec_dat[5]=' ';
  103.         rec_dat[6]='0'+(TH/10);
  104.         rec_dat[7]='0'+(TH%10);
  105.         rec_dat[8]='C';
  106.     }
  107. }

  108. void Read_Busy()           //忙检测函数,判断bit7是0,允许执行;1禁止
  109. {
  110.     unsigned char sta;      //
  111.     LCD1602_DB = 0xff;
  112.     LCD1602_RS = 0;
  113.     LCD1602_RW = 1;
  114.     do
  115.     {
  116.         LCD1602_EN = 1;
  117.         sta = LCD1602_DB;
  118.         LCD1602_EN = 0;    //使能,用完就拉低,释放总线
  119.     }while(sta & 0x80);
  120. }

  121. void Lcd1602_Write_Cmd(unsigned char cmd)     //写命令
  122. {
  123.     Read_Busy();
  124.     LCD1602_RS = 0;
  125.     LCD1602_RW = 0;
  126.     LCD1602_DB = cmd;
  127.     LCD1602_EN = 1;
  128.     LCD1602_EN = 0;   
  129. }

  130. void Lcd1602_Write_Data(unsigned char dat)   //写数据
  131. {
  132.       Read_Busy();
  133.       LCD1602_RS = 1;
  134.       LCD1602_RW = 0;
  135.       LCD1602_DB = dat;
  136.       LCD1602_EN = 1;
  137.       LCD1602_EN = 0;
  138. }

  139. void LcdSetCursor(unsigned char x,unsigned char y)  //坐标显示
  140. {
  141.     unsigned char addr;
  142.     if(y == 0)
  143.         addr = 0x00 + x;
  144.     else
  145.         addr = 0x40 + x;
  146.    
  147.     Lcd1602_Write_Cmd(addr|0x80);
  148. }

  149. //按指定位置显示一个字符
  150. void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData)
  151. {
  152.         Y &= 0x1;
  153.         X &= 0xF; //限制X不能大于15,Y不能大于1
  154.         if (Y) X |= 0x40; //当要显示第二行时地址码+0x40;
  155.         X |= 0x80; //算出指令码
  156.         Lcd1602_Write_Cmd(X); //发命令字
  157.         Lcd1602_Write_Data(DData); //发数据
  158. }


  159. void LcdShowStr(unsigned char x,unsigned char y,unsigned char *str)     //显示字符串
  160. {
  161.     LcdSetCursor(x,y);      //当前字符的坐标
  162.     while(*str != '\0')
  163.     {
  164.         Lcd1602_Write_Data(*str++);
  165.     }
  166. }

  167. void InitLcd1602()              //1602初始化
  168. {
  169.     Lcd1602_Write_Cmd(0x38);    //打开,5*8,8位数据
  170.     Lcd1602_Write_Cmd(0x0c);
  171.     Lcd1602_Write_Cmd(0x06);
  172.     Lcd1602_Write_Cmd(0x01);    //清屏   
  173. }

  174. void control()
  175. {if(IrValue[2]==0x18)
  176. {in1=1;
  177.   in2=0;
  178.   in3=0;
  179.   in4=1;//qian
  180.   

  181. }
  182. if(IrValue[2]==0x1c)
  183. {in1=1;
  184. in2=1;
  185. in3=1;
  186. in4=1;//ting
  187. }
  188. if(IrValue[2]==0x08)
  189. {in1=1;
  190. in2=0;
  191. in3=0;
  192. in4=0;//zuo

  193.   
  194. }
  195. if(IrValue[2]==0x5a)
  196. {in1=0;
  197. in2=0;
  198. in3=0;
  199. in4=1;//you

  200.    
  201. }

  202. if(IrValue[2]==0x52)
  203. {in1=0;
  204. in2=1;
  205. in3=1;
  206. in4=0;//dao
  207. //if(IrValue[2]==0x09)
  208. //P3^3=0;}
  209. }

  210. }


  211. /*******************************************************************************
  212. * 函数名         :DigDisplay()
  213. * 函数功能                 :数码管显示函数
  214. * 输入           : 无
  215. * 输出                  : 无
  216. *******************************************************************************/
  217. /*******************************************************************************
  218. * 函数名         : IrInit()
  219. * 函数功能                   : 初始化红外线接收
  220. * 输入           : 无
  221. * 输出                  : 无
  222. *******************************************************************************/
  223. /*******************************************************************************
  224. * 函 数 名       : main
  225. * 函数功能                 : 主函数
  226. * 输    入       : 无
  227. * 输    出             : 无
  228. *******************************************************************************/
  229. /*********************************************************/
  230. void Count()
  231. {    time=TH0*256+TL0;
  232.          S1=(time*1.7)/100;     //算出来是CM
  233.      S2=(time*1.7)/100;
  234.      TH0=0x00;
  235.      TL0=0x00;
  236.      S=(S1+S2)/2;
  237.      disbuff[0]=S%1000/100;
  238.      disbuff[1]=S%1000%100/10;
  239.          disbuff[2]=S%1000%10%10;
  240.          DisplayOneChar(0, 1, ASCII[disbuff[0]]);
  241.          DisplayOneChar(1, 1, ASCII[10]);        //显示点
  242.          DisplayOneChar(2, 1, ASCII[disbuff[1]]);
  243.          DisplayOneChar(3, 1, ASCII[disbuff[2]]);
  244.          DisplayOneChar(4, 1, ASCII[12]);        //显示M
  245.          DisplayOneChar(0, 0, rec_dat[0]);
  246.          DisplayOneChar(1, 0, rec_dat[1]);
  247.          DisplayOneChar(2, 0, rec_dat[2]);
  248.          DisplayOneChar(3, 0, rec_dat[3]);
  249.          DisplayOneChar(4, 0, rec_dat[4]);
  250.          DisplayOneChar(5, 0, rec_dat[5]);
  251.          DisplayOneChar(6, 0, rec_dat[6]);
  252.          DisplayOneChar(7, 0, rec_dat[7]);
  253.      DisplayOneChar(8, 0, rec_dat[8]);
  254.          delay(1);
  255.          
  256.         }
  257. /********************************************************/
  258.       void hr()
  259.     {  TX=1;                                        //800MS  启动一次模块
  260.           _nop_();
  261.           _nop_();
  262.           _nop_();
  263.           _nop_();
  264.           _nop_();
  265.           _nop_();
  266.           _nop_();
  267.           _nop_();
  268.           _nop_();
  269.           _nop_();
  270.           _nop_();
  271.           _nop_();
  272.           _nop_();
  273.           _nop_();
  274.           _nop_();
  275.           _nop_();
  276.           _nop_();
  277.           _nop_();
  278.           _nop_();
  279.           _nop_();
  280.           _nop_();
  281.           TX=0;
  282.          while(!RX);                //当RX为零时等待
  283.          TR0=1;                            //开启计数
  284.          while(RX);                        //当RX为1计数并等待
  285.          TR0=0;
  286.                                          //关闭计数
  287.          }


  288. void main()
  289. {       
  290.    
  291.         EX0=1;//打开中断0允许
  292.         IRIN=1;//初始化端口;
  293.         TMOD=0x11;                   //设T0为方式1,GATE=1;
  294.         IT0=1;//下降沿触发
  295.         IT1=1;
  296.         IE0=1;
  297.         IE1=1;
  298.         TH0=0;
  299.         TL0=0;         
  300.         TH1=0xf8;                   //2MS定时
  301.         TL1=0x30;
  302.         ET0=1;             //允许T0中断
  303.         ET1=1;                           //允许T1中断
  304.         TR1=1;                           //开启定时器
  305.         EA=1;
  306.         TR0=0;                           //开启总中断
  307.     InitLcd1602();
  308.         LcdShowStr(9,0,Pow);
  309.         LcdShowStr(5,1,Zhu);
  310.         while(1)
  311.         {if(timer>=100)
  312.             {hr();
  313.              }
  314.          if(DH>=200)
  315.            { DHT11_receive();
  316.                 DH=0;
  317.                 }
  318.          Count();

  319.                 
  320.         }               
  321. }

  322. /*******************************************************************************
  323. * 函数名         : ReadIr()
  324. * 函数功能                   : 读取红外数值的中断函数
  325. * 输入           : 无
  326. * 输出                  : 无
  327. *******************************************************************************/

  328. void ReadIr() interrupt 0
  329. {
  330.         u8 j,k;
  331.         u16 err;
  332.         Tim=0;                                         
  333.         delay(700);        //7ms
  334.         if(IRIN==0)                //确认是否真的接收到正确的信号
  335.         {         
  336.                
  337.                 err=1000;                                //1000*10us=10ms,超过说明接收到错误的信号
  338.                 /*当两个条件都为真是循环,如果有一个条件为假的时候跳出循环,免得程序出错的时
  339.                 侯,程序死在这里*/       
  340.                 while((IRIN==0)&&(err>0))        //等待前面9ms的低电平过去                 
  341.                 {                       
  342.                         delay(1);
  343.                         err--;
  344.                 }
  345.                 if(IRIN==1)                        //如果正确等到9ms低电平
  346.                 {
  347.                         err=500;
  348.                         while((IRIN==1)&&(err>0))                 //等待4.5ms的起始高电平过去
  349.                         {
  350.                                 delay(1);
  351.                                 err--;
  352.                         }
  353.                         for(k=0;k<4;k++)                //共有4组数据
  354.                         {                               
  355.                                 for(j=0;j<8;j++)        //接收一组数据
  356.                                 {

  357.                                         err=60;               
  358.                                         while((IRIN==0)&&(err>0))//等待信号前面的560us低电平过去
  359.                                         {
  360.                                                 delay(1);
  361.                                                 err--;
  362.                                         }
  363.                                         err=500;
  364.                                         while((IRIN==1)&&(err>0))         //计算高电平的时间长度。
  365.                                         {
  366.                                                 delay(10);         //0.1ms
  367.                                                 Tim++;
  368.                                                 err--;
  369.                                                 if(Tim>30)
  370.                                                 {
  371.                                                         return;
  372.                                                 }
  373.                                         }
  374.                                         IrValue[k]>>=1;         //k表示第几组数据
  375.                                         if(Tim>=8)                        //如果高电平出现大于565us,那么是1
  376.                                         {
  377.                                                 IrValue[k]|=0x80;
  378.                                         }                  
  379.                                         Tim=0;                //用完时间要重新赋值                                                       
  380.                                 }
  381.                         }
  382.                 }
  383.                 control();

  384. }                               
  385. }


  386. /*
  387.    void  zd3()  interrupt 3                  //T1中断用来扫描数码管和计800MS启动模块
  388.   {
  389.          TH1=0xf8;
  390.          TL1=0x30;
  391.          timer++;
  392.          DH++;
  393.          if(timer>=100)
  394.          {
  395.           timer=0;
  396.           TX=1;                                        //800MS  启动一次模块
  397.           _nop_();
  398.           _nop_();
  399.           _nop_();
  400.           _nop_();
  401.           _nop_();
  402.           _nop_();
  403.           _nop_();
  404.           _nop_();
  405.           _nop_();
  406.           _nop_();
  407.           _nop_();
  408.           _nop_();
  409.           _nop_();
  410.           _nop_();
  411.           _nop_();
  412.           _nop_();
  413.           _nop_();
  414.           _nop_();
  415.           _nop_();
  416.           _nop_();
  417.           _nop_();
  418.           TX=0;
  419.          }
  420.          
  421.          
  422.           
  423.   }
  424. */
  425. void  zd3()  interrupt 3                  //T1中断用来扫描数码管和计800MS启动模块
  426.   {
  427.          TH1=0xf8;
  428.          TL1=0x30;
  429.          timer++;
  430.          DH++;
  431.          }
  432.   




复制代码

回复

使用道具 举报

ID:282877 发表于 2018-2-9 18:21 | 显示全部楼层
你这单片机接的外设模块太多,还都得用到定时器,还开两个中断,相互之间会有干扰。特别是温湿度模块需要精准定时,建议在执行温湿度模块程序时关闭所有中断。
回复

使用道具 举报

ID:265318 发表于 2018-2-9 18:38 | 显示全部楼层
eeeewu 发表于 2018-2-9 18:21
你这单片机接的外设模块太多,还都得用到定时器,还开两个中断,相互之间会有干扰。特别是温湿度模块需要精 ...

可是超声波也得用中断额
回复

使用道具 举报

ID:282439 发表于 2018-2-9 20:11 | 显示全部楼层
内存堆栈是否溢出?换个内存大一点的芯片。
回复

使用道具 举报

ID:148552 发表于 2018-2-10 08:40 | 显示全部楼层
单片机运行卡死我教你一个简单的方法
找出单片机中所有的while循环,比如你程序里的
do
    {
        LCD1602_EN = 1;
        sta = LCD1602_DB;
        LCD1602_EN = 0;    //使能,用完就拉低,释放总线
    }while(sta & 0x80);
假如sta最高位一直为1,是不是单片机要在这死等?
这里我就是举个例子,不一定是这里的问题。
单片机运行卡死一般都是进入了一个死循环,特别是while这种,for循环也有可能
比如for(i=0;i<5;j++),这里由于粗心写成了j,是不是程序一直在这里循环?


所以 建议程序多用中断
回复

使用道具 举报

ID:265318 发表于 2018-2-10 10:09 | 显示全部楼层
你像风儿来了 发表于 2018-2-10 08:40
单片机运行卡死我教你一个简单的方法:
找出单片机中所有的while循环,比如你程序里的
do

好的大佬,我再看看,感谢
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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