找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開(kāi)始

搜索
查看: 4161|回復(fù): 4
收起左側(cè)

做基于單片機(jī)的pm2.5空氣質(zhì)量檢測(cè)儀仿真時(shí)程序出現(xiàn)問(wèn)題,求助

[復(fù)制鏈接]
ID:291036 發(fā)表于 2018-5-23 09:38 | 顯示全部樓層 |閱讀模式
我是用的51單片機(jī),傳感器用的夏普粉塵傳感器GP2Y1010AU0F,AD0832,LCD1602我一開(kāi)始在網(wǎng)上找的電路圖和程序,然后再proteus做的仿真,是可以出結(jié)果的,但因?yàn)槔蠋熞笪覀冇蒙a(chǎn)實(shí)習(xí)的板子,所以我把電路圖按照生產(chǎn)實(shí)習(xí)那個(gè)重新做的,就是一些地方的接口變了,然后復(fù)位電路變了。然后我把之前那個(gè)程序最上面定義的接口改了,別的沒(méi)動(dòng),仿真LCD就沒(méi)有數(shù)字了
這是我的畢業(yè)設(shè)計(jì),很急,我C語(yǔ)言真的一點(diǎn)不會(huì),出不來(lái)結(jié)果,求大神幫幫忙,看看我的程序怎么改。
我附上我的原圖,現(xiàn)在的圖,和現(xiàn)在的程序。

#include <reg52.h>
#include <intrins.h>
#define uint unsigned int
#define uchar unsigned char //宏定義
sbit RS=P3^7;//液晶接口
sbit EN=P2^7;

sbit LED = P3^4;//粉塵傳感器控制接口

sbit ADCS = P1^3;//AD0832接口
sbit ADCLK =P1^0;
sbit ADDI = P1^1;
sbit ADDO = P1^1;                  

sbit SET= P2^0;//按鍵接口
sbit ADD= P2^1;
sbit DEC= P2^2;

sbit BEEP=P2^4;//蜂鳴器接口

uchar set_st;
uchar tab[5];
uint DUST_SET=150; //固體顆粒的閾值

//bit shanshuo_st; //閃爍間隔標(biāo)志
bit beep_st; //蜂鳴器間隔標(biāo)志
uchar x=4; //計(jì)數(shù)器
//定義標(biāo)識(shí)
uchar FlagStart = 0;
float DUST_Value;
uint DUST;
uchar num=0;
uchar mm;
uchar abc;
uchar ADC_Get[10]={0}; //定義AD采樣數(shù)組
uchar str[5]={0};

/*****初始化定時(shí)器0*****/
void InitTimer(void)
{
        TMOD = 0x01;
        TL0 = (65536-10000)/256; //定時(shí)10ms
        TH0 = (65536-10000)%256;
        TR0 = 1;
        ET0 = 1;
        EA = 1;
}
/*************************lcd1602程序**************************/
void delay1ms(uint ms)//延時(shí)1毫秒
{
    uint i,j;
        for(i=0;i<ms;i++)
        for(j=0;j<100;j++);
}

void wr_com(uchar com)//寫指令//
{
    delay1ms(1);
        RS=0;
//        RW=0;
        EN=0;
        P0=com;
        delay1ms(1);
        EN=1;
        delay1ms(1);
        EN=0;
}
void wr_dat(uchar dat)//寫數(shù)據(jù)//
{
    delay1ms(1);;
        RS=1;
//        RW=0;
        EN=0;
        P0=dat;
        delay1ms(1);
        EN=1;
        delay1ms(1);
        EN=0;
}
/*****************************液晶初始化
*********************************************/
void lcd_init()//初始化設(shè)置//
{
        delay1ms(15);
        wr_com(0x38);
        delay1ms(5);
        wr_com(0x01);
        delay1ms(5);
        wr_com(0x06);
        delay1ms(5);
        wr_com(0x0c);
        delay1ms(5);

        wr_com(0x80);
        wr_dat('P');//
        wr_com(0x81);
        wr_dat('M');//:
        wr_com(0x82);
        wr_dat('2');//
        wr_com(0x83);
        wr_dat('.');//:
        wr_com(0x84);
        wr_dat('5');//:
        wr_com(0x85);
        wr_dat(':');


        wr_com(0x8b);
        wr_dat('m');
        wr_com(0x8c);
        wr_dat('g');
        wr_com(0x8d);
        wr_dat('/');
        wr_com(0x8e);
        wr_dat('m');
        wr_com(0x8f);
        wr_dat('3');
       


/////////////////////////////////////

        wr_com(0xc0);
        wr_dat('A');
        wr_com(0xc1);
        wr_dat('l');
        wr_com(0xc2);
        wr_dat('a');
        wr_com(0xc3);
        wr_dat('r');
        wr_com(0xc4);
        wr_dat('m');
        wr_com(0xc5);
        wr_dat(':');

        wr_com(0xcb);
        wr_dat('m');
        wr_com(0xcc);
        wr_dat('g');
        wr_com(0xcd);
        wr_dat('/');
        wr_com(0xce);
        wr_dat('m');
        wr_com(0xcf);
        wr_dat('3');
}
/*****************顯示函數(shù)******************************/
void disp(unsigned int Data)//PM2.5值顯示
{
        uint Temp;
        Temp=Data%10000;
        str[0]=Temp/1000+0x30; //千位
        Temp%=1000;
        str[1]='.';
        str[2]=Temp/100+0x30; //百位
        Temp%=100;
        str[3]=Temp/10+0x30; //十位
        str[4]=Temp%10+0x30; //個(gè)位
        wr_com(0x86);
        wr_dat(str[0]);
        wr_com(0x87);
        wr_dat(str[1]);
        wr_com(0x88);
        wr_dat(str[2]);
        wr_com(0x89);
        wr_dat(str[3]);
        wr_com(0x8a);
        wr_dat(str[4]);

}
/************************報(bào)警值顯示************************************/
void baojing()
{
        wr_com(0xc6);
        wr_dat(tab[0]+0x30);
        wr_com(0xc7);
        wr_dat(tab[1]);
        wr_com(0xc8);
        wr_dat(tab[2]+0x30);
        wr_com(0xc9);
        wr_dat(tab[3]+0x30);
        wr_com(0xca);
        wr_dat(tab[4]+0x30);
}
/*****延時(shí)子程序*****/
void Delay(uint num)
{
while( --num );
}
/**************************按鍵檢測(cè)
*******************************************/
void checkkey()
{
        if(SET==0)
        {
        Delay(2000);
        do{}while(SET==0);
        set_st++;
        if(set_st>1)set_st=0;
        }
        if(set_st==0)
        {
        }
        else if(set_st==1)
        {
        if(DEC==0)
        {
        Delay(2000);
        do{}while(DEC==0);
        if(DUST_SET>0)DUST_SET--;
        if(DUST_SET==0)DUST_SET=0;
        }
        if(ADD==0)
        {
        Delay(2000);
        do{}while(ADD==0);
        DUST_SET++;
        if(DUST_SET>800)DUST_SET=800;
        }
        }
        tab[0]=DUST_SET/1000;
        tab[1]='.';
        tab[2]=DUST_SET%1000/100;
        tab[3]=DUST_SET%100/10;
        tab[4]=DUST_SET%10;
}
/*****報(bào)警子程序*****/
void Alarm()
{
        if(x>=10){beep_st=~beep_st;x=0;}
        if(DUST>DUST_SET&&beep_st==1)BEEP=0;
        else BEEP=1;
//        if(DUST>0&&DUST<100){LED2=0;LED3=1;LED4=1;}
//        if(DUST>=10&&DUST<300){LED2=1;LED3=0;LED4=1;}
//        if(DUST>=300){LED2=1;LED3=1;LED4=0;}
        }
/**************************AD0832轉(zhuǎn)換程序
***********************************************/
uchar ADC0832(bit mode,bit channel) //AD轉(zhuǎn)換,返回結(jié)果
{
        uchar i,dat,ndat;
        ADCS = 0;//拉低CS端
        _nop_();
        _nop_();
        ADDI = 1; //第1個(gè)下降沿為高電平
        ADCLK = 1;//拉高CLK端
        _nop_();
        _nop_();
        ADCLK = 0;//拉低CLK端,形成下降沿1
        _nop_();
        _nop_();
        ADDI = mode; //低電平為差分模式,高電平為單通道模式。
        ADCLK = 1;//拉高CLK端
        _nop_();
        _nop_();
        ADCLK = 0;//拉低CLK端,形成下降沿2
        _nop_();
        _nop_();
        ADDI = channel; //低電平為CH0,高電平為CH1
        ADCLK = 1;//拉高CLK端
        _nop_();
        _nop_();
        ADCLK = 0;//拉低CLK端,形成下降沿3
        ADDI = 1;//控制命令結(jié)束(經(jīng)試驗(yàn)必需)
        dat = 0;
        //下面開(kāi)始讀取轉(zhuǎn)換后的數(shù)據(jù),從最高位開(kāi)始依次輸出(D7~D0)
        for(i = 0;i < 8;i++)
        {
        dat <<= 1;
        ADCLK=1;//拉高時(shí)鐘端
        _nop_();
        _nop_();
        ADCLK=0;//拉低時(shí)鐘端形成一次時(shí)鐘脈沖
        _nop_();
        _nop_();
        dat |= ADDO;
        }
        ndat = 0; //記錄D0
        if(ADDO == 1)
        ndat |= 0x80;
        //下面開(kāi)始繼續(xù)讀取反序的數(shù)據(jù)(從D1到D7)
        for(i = 0;i < 7;i++)
        {
        ndat >>= 1;
        ADCLK = 1;//拉高時(shí)鐘端
        _nop_();
        _nop_();
        ADCLK=0;//拉低時(shí)鐘端形成一次時(shí)鐘脈沖
        _nop_();
        _nop_();
        if(ADDO==1)
        ndat |= 0x80;
        }
        ADCS=1;//拉高CS端,結(jié)束轉(zhuǎn)換
        ADCLK=0;//拉低CLK端
        ADDI=1;//拉高數(shù)據(jù)端,回到初始狀態(tài)
        if(dat==ndat)
        return(dat);
        else
        return 0;
}
/*****定時(shí)器0中斷服務(wù)程序*****/
void timer0(void) interrupt 1
{
        uint j;
        TL0 = (65536-10000)/256; //定時(shí)10ms
        TH0 = (65536-10000)%256;
        LED=1; //開(kāi)啟傳感器的LED
        x++;
        for (j=0;j<30;j++); //0.28ms //延時(shí)0.28ms
        abc=ADC0832(1,0); //開(kāi)啟ADC采集
        FlagStart=1;
        TR0 = 0; //先關(guān)閉定時(shí)器0
        EA = 0;
        LED=0;//關(guān)閉傳感器LED
}
//中值濾波
//算法:先進(jìn)行排序,然后將數(shù)組的中間值作為當(dāng)前值返回。
uchar Error_Correct(uchar *str,uchar num)
{
        unsigned char i=0;
        unsigned char j=0;
        unsigned char Temp=0;
        //排序
        for(i=0;i<num-1;i++)
        {
        for(j=i+1;j<num;j++)
        {
           if(str[i]<str[j])
                {
                Temp=str[i];
                str[i]=str[j];
                str[j]=Temp;
                }
        }
}

        //去除誤差,取中間值
        return str[num/2];
}
/*****主函數(shù)*****/
void main(void)
{
        InitTimer(); //初始化定時(shí)器
        BEEP=1;
        lcd_init();//初始化顯示
        delay1ms(500);
        while(1)
        {

         checkkey();//按鍵檢測(cè)
        if(set_st==0)
        {
                //wr_com(0x0c);
                if(FlagStart==1) //1次數(shù)據(jù)采集完成
                {
                num++;
                ADC_Get[num]=abc;
                if(num>9)
                {
                num=0;
//                DUST=Error_Correct(ADC_Get,10); //求取10次AD采樣的值
//                DUST_Value=(DUST/256.0)*5000; //轉(zhuǎn)化成電壓值MV
//                DUST_Value=DUST_Value*0.17-100; //固體懸浮顆粒濃度計(jì)算 Y=0.17*X-0.1 X--采樣電壓V
       
                DUST=Error_Correct(ADC_Get,10);
            DUST_Value=(DUST/256.0)*5;//轉(zhuǎn)化成電壓值
            DUST_Value=(DUST_Value*0.17-0.1)*1000;//固體懸浮顆粒濃度計(jì)
       
                if(DUST_Value<0) DUST_Value=0;
                if(DUST_Value>760) DUST_Value=760; //限位
                DUST=(uint)DUST_Value;
                }
                TL0 = (65536-10000)/256;
                TH0 = (65536-10000)%256;
                TR0 = 1; //開(kāi)啟定時(shí)器0
                EA = 1;
                FlagStart=0;
                }
                Alarm(); //報(bào)警檢測(cè)
        }
        disp(DUST);//顯示粉塵濃度值
        baojing();//顯示報(bào)警值

                if(set_st==1)//報(bào)警值閃動(dòng)
                {
//                wr_com(0xca);
//                wr_com(0x0d);
//                delay1ms(150);
                }
        }
}




原圖仿真

原圖仿真

現(xiàn)在的圖仿真

現(xiàn)在的圖仿真
回復(fù)

使用道具 舉報(bào)

ID:303383 發(fā)表于 2018-5-23 09:58 | 顯示全部樓層
提示: 作者被禁止或刪除 內(nèi)容自動(dòng)屏蔽
回復(fù)

使用道具 舉報(bào)

ID:96682 發(fā)表于 2018-5-23 11:20 | 顯示全部樓層
RW 得接 GND
回復(fù)

使用道具 舉報(bào)

ID:291036 發(fā)表于 2018-5-23 13:10 | 顯示全部樓層

那您知道我這個(gè)復(fù)位電路由之前的電容復(fù)位變成按鍵復(fù)位之后,程序怎么改嗎
回復(fù)

使用道具 舉報(bào)

ID:96682 發(fā)表于 2018-5-23 14:26 | 顯示全部樓層
在 U1 ⑨ 的C3兩端并個(gè)復(fù)位按鍵就可以了,圖上 C3 R1 按鍵少個(gè)節(jié)點(diǎn),程序不用改

回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表