找回密码
 立即注册

QQ登录

只需一步,快速开始

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

PM2.5检测Proteus仿真程序,用可调电阻代替

[复制链接]
跳转到指定楼层
楼主
仿真没有PM2.5传感器,用可调电阻代替
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)


单片机源程序如下:
  1. /*
  2. 51单片机 1602+ADC0832显示程序,用proteus 7.8仿真通过。
  3. */


  4. #include <reg52.h>
  5. #include<intrins.h>
  6. #include<math.h>
  7. #include <stdio.h>


  8. char data str[]="                ";
  9. /**********************************/
  10. /**********LCD1602接口程序**********/


  11. #define DD P2
  12. sbit Rs=P3^0;
  13. sbit Rw=P3^1;
  14. sbit E=P3^2;
  15. sbit qaz=P1^0;
  16. sbit wsx=P1^1;

  17. /********************************/
  18. void delay_1ms(unsigned char i)   //最小延时1ms
  19. {
  20. unsigned char j;
  21. while(i--)
  22. for(j=0;j<125; j++);
  23. }
  24. void delay_10us(unsigned char i) //最小延时10us
  25. {
  26. unsigned char j;
  27. while(i--)
  28. for(j=0;j<10; j++);
  29. }


  30. void write_com(unsigned char com)   //写指令
  31. {
  32. delay_10us(5);
  33. E=0;
  34. Rs=0;
  35. Rw=0;
  36. DD=com;
  37. delay_10us(50); //>40us
  38. E=1;
  39. delay_1ms(2); //>150us
  40. E=0;
  41. delay_10us(4); //>25+10us
  42. }


  43. void write_data(unsigned char DATA)   //写数据
  44. {
  45. delay_10us(50);
  46. E=0;
  47. Rs=1;
  48. Rw=0;
  49. DD=DATA;
  50. delay_10us(50);
  51. E=1;
  52. delay_10us(50);
  53. E=0;
  54. delay_10us(4);
  55. }




  56. void addr_x_y(unsigned char x,bit y)   //写坐标,定位置
  57. {
  58. unsigned char temp=0x80; //默认最高位:D7为1 即以0x80开始。  
  59. if(y) //y :0为第一行  1为第二行
  60.   {
  61.   temp|=0x40;
  62.   }
  63.   temp|=x;
  64. write_com(temp);
  65. }




  66. void Show_Char(unsigned char x,bit y,unsigned char p)


  67. //在指定位置显示一个字符。
  68. {
  69. addr_x_y(x,y);
  70. write_data(p);
  71. }


  72. void Show_String(unsigned char x,bit y,char *ptr)
  73. {
  74.   unsigned char i;
  75. for (i=0;i<16;i++)
  76.   Show_Char(x++,y,*(ptr+i));//循环显示16个字符
  77. }




  78. void init(void) //1602初始化代码
  79. {
  80. delay_1ms(1500);
  81. write_com(0x38);
  82. delay_1ms(5);
  83. write_com(0x38);
  84. delay_1ms(5);
  85. write_com(0x38);
  86. delay_1ms(5);
  87. write_com(0x38);
  88. write_com(0x08);
  89. write_com(0x06);
  90. write_com(0x0c);
  91. write_com(0x01);
  92. }
  93. void xs_int(unsigned int shuju,bit t)   //数据显示
  94. {
  95. unsigned char huancun[6]={0};
  96. unsigned char biaozhi=0,i;
  97. if   (shuju < 10) biaozhi = 1;
  98. else if(shuju < 100) biaozhi = 2;
  99. else if(shuju < 1000) biaozhi = 3;
  100. else if(shuju < 10000) biaozhi = 4;
  101. else if(shuju < 65535) biaozhi = 5;
  102. switch(biaozhi) //这里没有break,因此从标识匹配的入口直接执行到最后,完成整数各位的提取到数组。
  103.   {
  104.   case 5:huancun[5] = shuju/10000;
  105.    case 4:huancun[4] = shuju%10000/1000;
  106.    case 3:huancun[3] = shuju%1000/100;
  107.    case 2:huancun[2] = shuju%100/10;
  108.    case 1:huancun[1] = shuju%10; break;
  109.    default:break;
  110. }
  111. for(i=0;i<6;i++)
  112.   {
  113.   if(i==3)Show_Char(i,1,'.'); //加入小数点,缩小了10000倍,因此AD采样后的值需要乘上10000*5V/256=196(V)
  114.        else Show_Char(i,t,0x30+huancun[6-i-1]);
  115. }
  116. Show_Char(6,t,'0');
  117. }




  118. /************************************************************/
  119. /**********ADC0832接口程序************************************/
  120. sbit ADC_CS =P3^4;
  121. sbit ADC_CLK=P3^5;
  122. sbit ADC_DO =P3^6;
  123. sbit ADC_DI =P3^7;
  124. /*******************************************************************/
  125. void Delay(unsigned char j)
  126. {
  127. unsigned char i;
  128. for(i=0;i<j;i++); //延时,脉冲一位持续的时间
  129. }


  130. unsigned char ADC0832(void) //把模拟电压值转换成8位二进制数并返回
  131. {
  132. unsigned char i,data_c;
  133. data_c=0;
  134. ADC_CS=0;
  135. ADC_CLK=1;
  136. ADC_DO=0;//片选,DO为高阻态
  137. //ADC_DI=0; //先拉低

  138. for(i=0;i<10;i++)
  139.     {;}

  140. ADC_CLK=0;
  141. Delay(2);
  142. ADC_CLK=1;
  143. ADC_DI=1; //启始位  
  144. Delay(2);

  145. ADC_CLK=0; //第一个脉冲下降沿,DI=1为起始位
  146. Delay(2);
  147. ADC_CLK=1;
  148. ADC_DI=1;   //1
  149. Delay(2);

  150. ADC_CLK=0;  //第二个脉冲下降沿,DI=1
  151. Delay(2);
  152. ADC_DI=1;  //1 若为0则选择CH0
  153. ADC_CLK=1;
  154. Delay(2);

  155. ADC_CLK=0; //第三个脉冲下降沿,DI=1,选择 ADC0832 的CH1(1 1)
  156. Delay(2);
  157. ADC_DI=1;   //已不关心
  158. ADC_DO=1; //高阻
  159. ADC_CLK=1;
  160. Delay(2);

  161. for (i=0; i<8; i++)
  162.    {
  163. ADC_CLK=0; //第四个开始读数据
  164. Delay(2);
  165.    data_c=(data_c<<1)|ADC_DO;//在每个脉冲的下降沿DO输出一位数据,最终ch为8位二进制数
  166. ADC_CLK=1;
  167. Delay(2);
  168.   }
  169. ADC_CS=1;//取消片选,一个转换周期结束
  170. return(data_c);//把转换结果返回
  171. }


  172. void main(void)
  173. {
  174. unsigned int data_temp=0;
  175. qaz=0;
  176. wsx=0;

  177. while(1)
  178. {
  179. data_temp=ADC0832();
  180. sprintf(str,"Air quality is:"); //the first line
  181.     init();
  182. Show_String(0,0,str);
  183. xs_int(196*data_temp,1);   //10000*5V/256=196(V),把采样值放大10000倍后再处理,移动小数点4位。 4位有效数字,其实用不到,因为8位采样精度为:0.019V
  184. delay_1ms(500);
  185. if(196*data_temp>20000)
  186. qaz=1;
  187. else
  188. qaz=0;
  189. if(196*data_temp>40000)
  190. wsx=1;
  191. else
  192. wsx=0;


  193.    }
  194. }
复制代码

所有资料51hei提供下载:
f.zip (85.41 KB, 下载次数: 51)


评分

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

查看全部评分

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

使用道具 举报

沙发
ID:686752 发表于 2020-9-3 14:28 | 只看该作者
你好,我想问一下,这里PM2.5传感器你用电阻替代了,那么做实物的时候我直接把PM2.5传感器的电压输出引脚接到AD连接滑动变阻器的引脚是不是就行了。PM2.5还用另外的检测程序吗?
回复

使用道具 举报

板凳
ID:1012337 发表于 2022-3-27 20:22 | 只看该作者
梦醒了11 发表于 2020-9-3 14:28
你好,我想问一下,这里PM2.5传感器你用电阻替代了,那么做实物的时候我直接把PM2.5传感器的电压输出引脚接 ...

肯定不是的,做仿真用滑动变阻器替代是迫不得已,做实物直接用现成的传感器模块连接A/D模块就行,代码肯定要改动,要用传感器的代码
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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