找回密码
 立即注册

QQ登录

只需一步,快速开始

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

基于51单片机的烟雾传感器程序

[复制链接]
跳转到指定楼层
楼主
ID:258823 发表于 2017-12-7 21:06 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
这是我这次传感器课题设计
用了两周时间
主要内容包括设计的程序
下载:
单片机程序.doc (41.5 KB, 下载次数: 56)


单片机源程序如下:
  1. #include <reg52.h>
  2. #include <intrins.h>

  3. #define uchar unsigned char                // 以后unsigned char就可以用uchar代替
  4. #define uint  unsigned int                // 以后unsigned int 就可以用uint 代替

  5. sfr ISP_DATA  = 0xe2;                        // 数据寄存器
  6. sfr ISP_ADDRH = 0xe3;                        // 地址寄存器高八位
  7. sfr ISP_ADDRL = 0xe4;                        // 地址寄存器低八位
  8. sfr ISP_CMD   = 0xe5;                        // 命令寄存器
  9. sfr ISP_TRIG  = 0xe6;                        // 命令触发寄存器
  10. sfr ISP_CONTR = 0xe7;                        // 命令寄存器

  11. sbit ADC_CS   = P1^0;                         // ADC0832的CS引脚
  12. sbit ADC_CLK  = P1^1;                         // ADC0832的CLK引脚
  13. sbit ADC_DAT  = P1^2;                 // ADC0832的DI/DO引脚
  14. sbit LcdRs_P  = P2^7;                     // 1602液晶的RS管脚      
  15. sbit LcdRw_P  = P2^6;                     // 1602液晶的RW管脚
  16. sbit LcdEn_P  = P2^5;                     // 1602液晶的EN管脚
  17. sbit Key1_P   = P3^2;                        // 减按键
  18. sbit Key2_P   = P3^3;                        // 加按键
  19. sbit Buzzer_P = P2^0;                        // 蜂鸣器
  20. sbit Led_P    = P2^1;                        // LED灯

  21. uchar gAlarm;                                        // 报警值

  22. /*********************************************************/
  23. // 单片机内部EEPROM不使能
  24. /*********************************************************/
  25. void ISP_Disable()
  26. {
  27.         ISP_CONTR = 0;
  28.         ISP_ADDRH = 0;
  29.         ISP_ADDRL = 0;
  30. }

  31. /*********************************************************/
  32. // 从单片机内部EEPROM读一个字节,从0x2000地址开始
  33. /*********************************************************/
  34. unsigned char EEPROM_Read(unsigned int add)
  35. {
  36.         ISP_DATA  = 0x00;
  37.         ISP_CONTR = 0x83;
  38.         ISP_CMD   = 0x01;
  39.         ISP_ADDRH = (unsigned char)(add>>8);
  40.         ISP_ADDRL = (unsigned char)(add&0xff);
  41.         // 对STC89C51系列来说,每次要写入0x46,再写入0xB9,ISP/IAP才会生效
  42.         ISP_TRIG  = 0x46;           
  43.         ISP_TRIG  = 0xB9;
  44.         _nop_();
  45.         ISP_Disable();
  46.         return (ISP_DATA);
  47. }

  48. /*********************************************************/
  49. // 往单片机内部EEPROM写一个字节,从0x2000地址开始
  50. /*********************************************************/
  51. void EEPROM_Write(unsigned int add,unsigned char ch)
  52. {
  53.         ISP_CONTR = 0x83;
  54.         ISP_CMD   = 0x02;
  55.         ISP_ADDRH = (unsigned char)(add>>8);
  56.         ISP_ADDRL = (unsigned char)(add&0xff);
  57.         ISP_DATA  = ch;
  58.         ISP_TRIG  = 0x46;
  59.         ISP_TRIG  = 0xB9;
  60.         _nop_();
  61.         ISP_Disable();
  62. }

  63. /*********************************************************/
  64. // 擦除单片机内部EEPROM的一个扇区
  65. // 写8个扇区中随便一个的地址,便擦除该扇区,写入前要先擦除
  66. /*********************************************************/
  67. void Sector_Erase(unsigned int add)         
  68. {
  69.         ISP_CONTR = 0x83;
  70.         ISP_CMD   = 0x03;
  71.         ISP_ADDRH = (unsigned char)(add>>8);
  72.         ISP_ADDRL = (unsigned char)(add&0xff);
  73.         ISP_TRIG  = 0x46;
  74.         ISP_TRIG  = 0xB9;
  75.         _nop_();
  76.         ISP_Disable();
  77. }



  78. /*********************************************************/
  79. // 毫秒级的延时函数,time是要延时的毫秒数
  80. /*********************************************************/
  81. void DelayMs(uint time)
  82. {
  83.         uint i,j;
  84.         for(i=0;i<time;i++)
  85.                 for(j=0;j<112;j++);
  86. }

  87. /*********************************************************/
  88. // 1602液晶写命令函数,cmd就是要写入的命令
  89. /*********************************************************/
  90. void LcdWriteCmd(uchar cmd)
  91. {
  92.         LcdRs_P = 0;
  93.         LcdRw_P = 0;
  94.         LcdEn_P = 0;
  95.         P0=cmd;
  96.         DelayMs(2);
  97.         LcdEn_P = 1;   
  98.         DelayMs(2);
  99.         LcdEn_P = 0;        
  100. }

  101. /*********************************************************/
  102. // 1602液晶写数据函数,dat就是要写入的数据
  103. /*********************************************************/
  104. void LcdWriteData(uchar dat)
  105. {
  106.         LcdRs_P = 1;
  107.         LcdRw_P = 0;
  108.         LcdEn_P = 0;
  109.         P0=dat;
  110.         DelayMs(2);
  111.         LcdEn_P = 1;   
  112.         DelayMs(2);
  113.         LcdEn_P = 0;
  114. }





  115. /*********************************************************/
  116. // 1602液晶初始化函数
  117. /*********************************************************/
  118. void LcdInit()
  119. {
  120.         LcdWriteCmd(0x38);        // 16*2显示,5*7点阵,8位数据口
  121.         LcdWriteCmd(0x0C);        // 开显示,不显示光标
  122.         LcdWriteCmd(0x06);        // 地址加1,当写入数据后光标右移
  123.         LcdWriteCmd(0x01);        // 清屏
  124. }

  125. /*********************************************************/
  126. // 液晶光标定位函数
  127. /*********************************************************/
  128. void LcdGotoXY(uchar line,uchar column)
  129. {
  130.         // 第一行
  131.         if(line==0)        
  132.                 LcdWriteCmd(0x80+column);
  133.          // 第二行
  134.         if(line==1)        
  135.                 LcdWriteCmd(0x80+0x40+column);
  136. }

  137. /*********************************************************/
  138. // 液晶输出字符串函数
  139. /*********************************************************/
  140. void LcdPrintStr(uchar *str)
  141. {
  142.         while(*str!='\0')
  143.                 LcdWriteData(*str++);
  144. }

  145. /*********************************************************/
  146. // 液晶输出数字
  147. /*********************************************************/
  148. void LcdPrintNum(uchar num)
  149. {
  150.         LcdWriteData(num/100+48);                        // 百位
  151.         LcdWriteData(num%100/10+48);                // 十位
  152.         LcdWriteData(num%10+48);                         // 个位
  153. }


  154. /*********************************************************/
  155. // 液晶显示初始化
  156. /*********************************************************/
  157. void LcdShowInit()
  158. {
  159.         LcdGotoXY(0,0);                                                // 液晶光标定位到第0行
  160.         LcdPrintStr("  MQ-2 System   ");                // 液晶第0行显示"  MQ-2 System   "
  161.         LcdGotoXY(1,0);                                                // 液晶光标定位到第1行
  162.         LcdPrintStr("Gas:    Warn:   ");                        // 液晶第1行显示"Gas:    Warn:   "
  163. }

  164. /*********************************************************/
  165. // ADC0832的时钟脉冲
  166. /*********************************************************/
  167. void WavePlus()
  168. {
  169.         _nop_();
  170.         ADC_CLK = 1;
  171.         _nop_();
  172.         ADC_CLK = 0;
  173. }

  174. /*********************************************************/
  175. // 获取指定通道的A/D转换结果
  176. /*********************************************************/
  177. uchar Get_ADC0832()
  178. {
  179.         uchar i;
  180.         uchar dat1=0;
  181.         uchar dat2=0;
  182.         ADC_CLK = 0;                                // 电平初始化
  183.         ADC_DAT = 1;
  184.         _nop_();
  185.         ADC_CS = 0;
  186.         WavePlus();                                // 起始信号
  187.         ADC_DAT = 1;
  188.         WavePlus();                                // 通道选择的第一位
  189.         ADC_DAT = 0;      
  190.         WavePlus();                                // 通道选择的第二位
  191.         ADC_DAT = 1;
  192.         for(i=0;i<8;i++)                        // 第一次读取
  193.         {
  194.                 dat1<<=1;
  195.                 WavePlus();
  196.                 if(ADC_DAT)
  197.                         dat1=dat1|0x01;
  198.                 else
  199.                         dat1=dat1|0x00;
  200.         }
  201.         for(i=0;i<8;i++)                        // 第二次读取
  202.         {
  203.                 dat2>>= 1;
  204.                 if(ADC_DAT)
  205.                         dat2=dat2|0x80;
  206.                 else
  207.                         dat2=dat2|0x00;
  208.                 WavePlus();
  209.         }
  210.         _nop_();                                        // 结束此次传输
  211.         ADC_DAT = 1;
  212.         ADC_CLK = 1;
  213.         ADC_CS  = 1;   
  214.         if(dat1==dat2)                                // 返回采集结果
  215.                 return dat1;
  216.         else
  217.                 return 0;
  218. }

  219. /*********************************************************/
  220. // 按键扫描
  221. /*********************************************************/
  222. void KeyScanf()
  223. {
  224.         if(Key1_P==0)                                                        // 如果减按键被按下        
  225.         {
  226.                 if(gAlarm>1)                                                // 只有gAlarm大于1才能减1                                                               
  227.                         gAlarm--;                                
  228.                 LcdGotoXY(1,13);                                        // 液晶光标定位到第1行第13列
  229.                 LcdPrintNum(gAlarm);                                // 刷新改变后的报警值
  230.                 DelayMs(250);                                                // 延时一下
  231.                 Sector_Erase(0x2000);                                // 存入EEPROM前先擦除
  232.                 EEPROM_Write(0x2000,gAlarm);                // 报警值存入EEPROM的0x2000这个地址
  233.         }
  234.         if(Key2_P==0)                                                        // 如果加按键被按下        
  235.         {
  236.                 if(gAlarm<100)                                        // 只有gAlarm小于100才能加1
  237.                         gAlarm++;                                
  238.                 LcdGotoXY(1,13);                                        // 液晶光标定位到第1行第13列
  239.                 LcdPrintNum(gAlarm);                                // 刷新改变后的报警值
  240.                 DelayMs(250);                                                // 延时一下
  241.                 Sector_Erase(0x2000);                                // 存入EEPROM前先擦除
  242.                 EEPROM_Write(0x2000,gAlarm);                // 报警值存入EEPROM的0x2000这个地址
  243.         }
  244. }

  245. /*********************************************************/
  246. // 报警判断
  247. /*********************************************************/
  248. void AlarmJudge(uchar dat)
  249. {
  250.         if(dat>gAlarm)
  251.         {
  252.                 Buzzer_P=0;                // 蜂鸣器报警
  253.                 Led_P=0;                        // LED灯亮
  254.         }
  255.         else
  256.         {
  257.                 Buzzer_P=1;                // 蜂鸣器停止报警
  258. ……………………

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


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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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