微波EDA网,见证研发工程师的成长!
首页 > 研发问答 > 嵌入式设计讨论 > MCU和单片机设计讨论 > 51单片机连接红外测温模块代码解释问题

51单片机连接红外测温模块代码解释问题

时间:10-02 整理:3721RD 点击:

  1. #include"at89x52.h"
  2. #include"intrins.h"
  3. //************************************
  4. #define  uint  unsigned int
  5. #define  uchar unsigned char
  6. #define  Nack_counter  10
  7. //************** 端口定义**************
  8. //LCD 控制线接口                          !缺少了剩下的引脚定义
  9. uchar flag1;
  10. sbit  RS=P0^7;
  11. sbit  RW=P0^6;
  12. sbit  LCDE=P0^5;
  13. //mlx90614 端口定义
  14. sbit  SCL=P1^6;// 时钟线
  15. sbit  SDA=P1^7;// 数据线
  16. //************ 数据定义****************
  17. bdata uchar flag;//可位寻址数据
  18. sbit bit_out=flag^7;
  19. sbit bit_in=flag^0;
  20. uchar DataH,DataL,Pecreg;
  21. //************ 函数声明*****************************************
  22. void   start_bit();                 //MLX90614 发起始位子程序
  23. void   stop_bit();                 //MLX90614发结束位子程序
  24. uchar  rx_byte(void);              //MLX90614 接收字节子程序
  25. void   send_bit(void);             //MLX90614发送位子程序
  26. void   tx_byte(uchar dat_byte);     //MLX90614 接收字节子程序
  27. void   receive_bit(void);           //MLX90614接收位子程序
  28. void   delay(uint N);              //延时程序
  29. uint   memread(void);             // 读温度数据
  30. void   init1602(void);        //LCD 初始化子程序
  31. void   chk_busy_flg(void);         //LCD 判断忙子程序
  32. void   dis_cmd_wrt(uchar cmd);     //LCD 写命令子程序
  33. void   dis_dat_wrt(uchar dat);       //LCD 写数据子程序
  34. void   display(uint Tem);           // 显示子程序
  35. //*************主函数*******************************************
  36. void main()
  37. {
  38. uint Tem;
  39. //函数部分
  40. SCL=1;SDA=1;_nop_();
  41. _nop_();_nop_();_nop_();
  42. SCL=0;
  43. delay(1000);
  44. SCL=1;
  45. init1602();
  46. while(1)
  47. {
  48.    Tem=memread();
  49.    display(Tem);
  50.    delay(20);
  51. }
  52. }
  53. //*********输入转换并显示*********
  54. void display(uint Tem)
  55. {
  56. uint T,a,b;
  57. T=Tem*2;
  58.   dis_cmd_wrt(0x01);//清屏
  59. if(T>=27315)
  60. {
  61.    T=T-27315;
  62.    a=T/100;
  63.    b=T-a*100;
  64. //---------------------------
  65.    if(a>=100)
  66.    {
  67.     dis_dat_wrt(0x30+a/100);
  68.     a=a%100;
  69.     dis_dat_wrt(0x30+a/10);
  70.     a=a%10;
  71.     dis_dat_wrt(0x30+a);
  72.    }
  73.    else if(a>=10)
  74.    {
  75.     dis_dat_wrt(0x30+a/10);
  76.     a=a%10;
  77.     dis_dat_wrt(0x30+a);
  78.    }
  79.    else
  80.    {
  81.     dis_dat_wrt(0x30+a);
  82.    }
  83.    dis_dat_wrt(0x2e);// 显示点
  84.    //---------------------------
  85.   if(b>=10)
  86.   {
  87.     dis_dat_wrt(0x30+b/10);
  88. //    b=b%10;
  89. //    dis_dat_wrt(0x30+b);
  90.   }
  91.   else
  92.   {
  93.     dis_dat_wrt(0x30);
  94. //    dis_dat_wrt(0x30+b);
  95.   }
  96. }
  97. //==========
  98. else
  99. {
  100.   T=27315-T;
  101.   a=T/100;
  102.     b=T-a*100;
  103.   dis_dat_wrt(0x2d);
  104. //--------------------------
  105.   if(a>=10)
  106.   {
  107.     dis_dat_wrt(0x30+a/10);
  108.     a=a%10;
  109.     dis_dat_wrt(0x30+a);
  110.   }
  111.   else
  112.   {
  113.     dis_dat_wrt(0x30+a);
  114.   }
  115.   dis_dat_wrt(0x2e);//显示点
  116. //--------------------------
  117.   if(b>=10)
  118.   {
  119.     dis_dat_wrt(0x30+b/10);
  120.     b=b%10;
  121.     dis_dat_wrt(0x30+b);
  122.   }
  123.   else
  124.   {
  125.     dis_dat_wrt(0x30);
  126.     dis_dat_wrt(0x30+b);
  127.   }
  128. }
  129. }

  130. //************************************
  131. void   start_bit(void)
  132. {
  133.    SDA=1;
  134.    _nop_();_nop_();_nop_();_nop_();_nop_();
  135.    SCL=1;
  136.    _nop_();_nop_();_nop_();_nop_();_nop_();
  137.    SDA=0;
  138.    _nop_();_nop_();_nop_();_nop_();_nop_();
  139.    SCL=0;
  140.    _nop_();_nop_();_nop_();_nop_();_nop_();

  141. }
  142. //------------------------------
  143. void   stop_bit(void)
  144. {

  145.    SCL=0;
  146.    _nop_();_nop_();_nop_();_nop_();_nop_();
  147.    SDA=0;
  148.    _nop_();_nop_();_nop_();_nop_();_nop_();
  149.    SCL=1;
  150.    _nop_();_nop_();_nop_();_nop_();_nop_();
  151.    SDA=1;
  152. }
  153. //--------- 发送一个字节---------
  154. void  tx_byte(uchar dat_byte)
  155. {
  156.    char i,n,dat;
  157.    n=Nack_counter;
  158. TX_again:
  159.    dat=dat_byte;
  160.    for(i=0;i<8;i++)
  161.    {
  162.      if(dat&0x80)
  163.       bit_out=1;
  164.      else
  165.       bit_out=0;
  166.      send_bit();
  167.      dat=dat<<1;  //右移一位继续运算
  168.    }
  169.    
  170.       receive_bit();
  171.    if(bit_in==1)
  172.    {
  173.     stop_bit();
  174.     if(n!=0)
  175.     {n--;goto Repeat;}
  176.     else
  177.      goto exit;
  178.    }
  179.    else
  180.     goto exit;
  181. Repeat:
  182.     start_bit();
  183.     goto TX_again;
  184. exit: ;
  185. }
  186. //-----------发送一个位---------
  187. void  send_bit(void)
  188. {
  189.   if(bit_out==0)
  190.   
  191.        SDA=0;
  192.   else
  193.      SDA=1;
  194.   _nop_();
  195.   SCL=1;
  196.   _nop_();_nop_();_nop_();_nop_();
  197.   _nop_();_nop_();_nop_();_nop_();
  198.   SCL=0;
  199.   _nop_();_nop_();_nop_();_nop_();
  200.   _nop_();_nop_();_nop_();_nop_();
  201. }
  202. //---------- 接收一个字节--------
  203. uchar rx_byte(void)
  204. {
  205.   uchar i,dat;
  206.   dat=0;
  207.   for(i=0;i<8;i++)
  208.   {
  209.     dat=dat<<1;
  210.     receive_bit();
  211.     if(bit_in==1)
  212.      dat=dat+1;
  213.   }
  214.   send_bit();
  215.   return dat;
  216. }

  217. //---------- 接收一个位----------
  218. void receive_bit(void)
  219. {
  220.   SDA=1;bit_in=1;
  221.   SCL=1;
  222.   _nop_();_nop_();_nop_();_nop_();
  223.   _nop_();_nop_();_nop_();_nop_();
  224.   bit_in=SDA;
  225.   _nop_();
  226.   SCL=0;
  227.   _nop_();_nop_();_nop_();_nop_();
  228.   _nop_();_nop_();_nop_();_nop_();
  229. }
  230. //------------ 延时--------------
  231. void   delay(uint N)
  232. {
  233.   uint i;
  234.   for(i=0;i<N;i++)
  235.   
  236.      _nop_();
  237. }
  238. //------------------------------
  239. uint memread(void)
  240. {
  241.   start_bit();
  242.   tx_byte(0xB4);  //Send SlaveAddress ==============================
  243.   //tx_byte(0x00);
  244.   tx_byte(0x07);  //Send Command
  245.   //------------
  246.   start_bit();
  247.   tx_byte(0x01);
  248.   bit_out=0;
  249.   DataL=rx_byte();
  250.   bit_out=0;
  251.   DataH=rx_byte();
  252.   bit_out=1;
  253.   Pecreg=rx_byte();
  254.   stop_bit();
  255.   return(DataH*256+DataL);
  256. }
  257. //******************LCD 显示子函数***********************
  258. void init1602(void)        // 初始化LCD
  259. {
  260.    dis_cmd_wrt(0x01);
  261.    dis_cmd_wrt(0x0c);
  262.    dis_cmd_wrt(0x06);
  263.    dis_cmd_wrt(0x38);
  264.    
  265.    }

  266. void chk_busy_flg(void) //LCD 忙标志判断
  267. {
  268.    flag1=0x80;
  269.   while(flag1&0x80)
  270. {
  271.    P2=0xff;
  272.    RS=0;
  273.    RW=1;
  274.    LCDE=1;
  275.    flag1=P2;
  276.    LCDE=0;
  277. }
  278. }

  279. void dis_cmd_wrt(uchar cmd)    // 写命令子函数
  280. {
  281.    chk_busy_flg();
  282.    P2=cmd;
  283.    RS=0;
  284.    RW=0;
  285.    LCDE=1;
  286.    LCDE=0;
  287. }

  288. void dis_dat_wrt(uchar dat) // 写数据子函数
  289. {
  290.   chk_busy_flg();
  291.   if(flag1==16)
  292.   {
  293.    P2=0XC0;
  294.    RS=0;
  295.    RW=0;
  296.    LCDE=1;
  297.    LCDE=0;
  298.   }
  299.    P2=dat;
  300.    RS=1;
  301.    RW=0;
  302.    LCDE=1;
  303.    LCDE=0;
  304. }

复制代码

代码中uint memread(void)
{
  start_bit();
  tx_byte(0xB4);  //Send SlaveAddress ==============================
  //tx_byte(0x00);
  tx_byte(0x07);  //Send Command
  //------------
  start_bit();
  tx_byte(0x01);
  bit_out=0;
  DataL=rx_byte();
  bit_out=0;
  DataH=rx_byte();
  bit_out=1;
  Pecreg=rx_byte();
  stop_bit();
  return(DataH*256+DataL);
}这个函数为什么是16位的数据?为什么要获取两次数据才等于一次?
谢谢了,我看了很久没看懂,翻了DATASHEET由于是英文的也不是很懂

这是因为存储的数据是超过8位,小于16位的,而存储器是以8位的结构存储的,所以存储或读取时都需要操作2次,然后再合并成一个完整的数据,这个在程序里也有。DataH*256+DataL

看下寄存器长度

哦哦,懂了

一直显示382.1是怎么回事啊,52单片机12M晶振

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top