51单片机解析卫星定位数据源码+DHT11,1602显示

什么都不多说,直接看代码:

#include <REG52.H>
#include <stdio.h>
#include <intrins.h>
////////////========================================================================================

/*
1602A(16*2)模拟口线接线方式
连接线图:   
       ---------------------------------------------------
       |LCM-----51   |  LCM-----51   |  LCM------51      |
       --------------------------------------------------|
       |DB0-----P0.0 |  DB4-----P0.4 |  RW-------P1.5    |
       |DB1-----P0.1 |  DB5-----P0.5 |  RS-------P1.4    |
       |DB2-----P0.2 |  DB6-----P0.6 |  E--------P1.6    |
       |DB3-----P0.3 |  DB7-----P0.7 |  V0接10K电阻到GND |
       ---------------------------------------------------
温湿度DHT11接线:模块data接   P3^2
******************/
sbit Data= P3^2;   //定义数据线      温湿度传感器DHT11数据接入
unsigned char rec_dat[9];   //用于显示的接收数据数组  

unsigned char  flag_rec=0;     
unsigned char  num_rec=0;
   
unsigned char code kaijihuamian[]="BeiDou_DHT11";    //开机显示
unsigned char code kaijihuamian2[]="AnShan_17-12-01";
unsigned char code receiving[]="Receiving!";
unsigned char code nodata[]="No BD data!";
unsigned char code LCD_161[]="BD_HeGuang Studios";
unsigned char code LCD_162[]="TEL:15117255049";

char code TIME_AREA= 8;     //时区
unsigned char flag_data;    //数据标志位

//BD数据存储数组
unsigned char JD[10];       //经度
unsigned char JD_a;     //经度方向
unsigned char WD[9];        //纬度
unsigned char WD_a;     //纬度方向
unsigned char date[6];      //日期
unsigned char time[6];      //时间
unsigned char time1[6];     //时间
unsigned char speed[5]={'0','0','0','0','0'};       //速度
unsigned char high[6];      //高度
unsigned char angle[5];     //方位角
unsigned char use_sat[2];   //使用的卫星数
unsigned char total_sat[2]; //天空中总卫星数
unsigned char lock;         //定位状态

//串口中断需要的变量
unsigned char seg_count;    //逗号计数器
unsigned char dot_count;    //小数点计数器
unsigned char byte_count;   //位数计数器
unsigned char cmd_number;   //命令类型
unsigned char mode;         //0:结束模式,1:命令模式,2:数据模式
unsigned char buf_full;     //1:整句接收完成,相应数据有效。0:缓存数据无效。
unsigned char cmd[5];       //命令类型存储数组

sbit rs = P1^4;
sbit rw = P1^5;
sbit ep = P1^6;

//-----------------------------------------------------------------------------------------------
//延时子程序
void delayms(unsigned char ms)
{
    unsigned char i;
    while(ms--)
    {
    for(i = 0; i < 120; i++);
    }
}


bit lcd_bz(void)
{                           // 测试LCD忙碌状态
    bit result;
    rs = 0;
    rw = 1;
    ep = 1;
    _nop_();
    _nop_();
    _nop_();
    _nop_();
    result = (bit)(P0 & 0x80);
    ep = 0;
    return result;  
}

void lcd_wcmd(unsigned char cmd)
{                           // 写入指令数据到LCD
    while(lcd_bz());
    rs = 0;
    rw = 0;
    ep = 0;
    _nop_();
    _nop_();    
    P0 = cmd;
    _nop_();
    _nop_();
    _nop_();
    _nop_();
    ep = 1;
    _nop_();
    _nop_();
    _nop_();
    _nop_();
    ep = 0;     
}

void lcd_pos(unsigned char pos)
{                           //设定显示位置
    lcd_wcmd(pos | 0x80);
}

void lcd_pos_2(unsigned char pos)
{                           //设定显示位置
    lcd_wcmd(pos | 0xC0);
}

void lcd_wdat(unsigned char dat)    
{                           //写入字符显示数据到LCD
    while(lcd_bz());
    rs = 1;
    rw = 0;
    ep = 0;
    P0 = dat;
    _nop_();
    _nop_();
    _nop_();
    _nop_();
    ep = 1;
    _nop_();
    _nop_();
    _nop_();
    _nop_();
    ep = 0; 
}
void lcd_string(unsigned char *ptr,unsigned char pos)
{

    unsigned char i;
    lcd_pos(pos);
    while(*(ptr+i) != '\0')
    {                       // 显示字符"welcome!"
        lcd_wdat(*(ptr+i));
        i++;
    }       

}

lcd_init()
{                           //LCD初始化设定
    lcd_wcmd(0x38);         //
    delayms(1);
    lcd_wcmd(0x0c);         //
    delayms(1);
    lcd_wcmd(0x06);         //
    delayms(1);
    lcd_wcmd(0x01);         //清除LCD的显示内容
    delayms(1);
}
//-----------------------------------------------------------------------------------------------
//延时子程序
void delaym(unsigned int ms)
{
    unsigned int i,j;
    for(i = 0;i < ms ;i++)
    {
        for(j = 0;j < 125 ;j++)
            ;
    }
}

void DHT11_delay_us(unsigned char n)
{
    while(--n);
}

void DHT11_delay_ms(unsigned int z)
{
   unsigned int i,j;
   for(i=z;i>0;i--)
      for(j=110;j>0;j--);
}

void DHT11_start()
{
   Data=1;
   DHT11_delay_us(2);
   Data=0;
   DHT11_delay_ms(20);   //延时18ms以上
   Data=1;
   DHT11_delay_us(30);
}
//-----------------------   温湿度读取   -----------------------------------

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

void DHT11_receive()      //接收40位的数据
{
    unsigned char R_H,R_L,T_H,T_L,RH,RL,TH,TL,revise; 
    DHT11_start();
    if(Data==0)
    {
        while(Data==0);   //等待拉高     
        DHT11_delay_us(40);  //拉高后延时80us
        R_H=DHT11_rec_byte();    //接收湿度高八位  
        R_L=DHT11_rec_byte();    //接收湿度低八位  
        T_H=DHT11_rec_byte();    //接收温度高八位  
        T_L=DHT11_rec_byte();    //接收温度低八位
        revise=DHT11_rec_byte(); //接收校正位

        DHT11_delay_us(25);    //结束

        if((R_H+R_L+T_H+T_L)==revise)      //校正
        {
            RH=R_H;
            RL=R_L;
            TH=T_H;
            TL=T_L;
        } 
        /*数据处理,方便显示*/
        rec_dat[0]='0'+(RH/10);
        rec_dat[1]='0'+(RH%10);
        rec_dat[2]='R';
        rec_dat[3]='H';
        rec_dat[4]=' ';
        rec_dat[5]=' ';
        rec_dat[6]='0'+(TH/10);
        rec_dat[7]='0'+(TH%10);
        rec_dat[8]='C';
    }
}

//-------------------------   <初始化设置 结束>   -------------------------------------
void DHT11_main(unsigned int zxi){      
    lcd_init();            //初始化1602
    lcd_wcmd(0x01);     //清除LCD的显示内容
    delaym(1);
    //============温湿度读取============================================================
    DHT11_delay_ms(1500);    //DHT11上电后要等待1S以越过不稳定状态在此期间不能发送任何指令
  DHT11_receive();  
    lcd_pos(3);         // 设置显示位置
    for(zxi = 0;zxi<6;zxi++)
    {
        lcd_wdat(kaijihuamian[zxi+6]);  // 显示字符dht11
    }
    zxi = 0;
    delayms(100);
    lcd_pos_2(2);              
    while(rec_dat[zxi] != '\0'){
        lcd_wdat(rec_dat[zxi]); //显示
        zxi++;
    }
    delaym(1000);               
}
//==================================================================================
//发送字节
void sendbyte(unsigned char ptr)
{
    SBUF=ptr;
    while(TI==0);
    TI=0;
}
//---------------------------------------------------------------------------
//发送数据流
void sendstring(unsigned char *ptr,unsigned char len)
{

    int i=0;
    while(i<len)
    {   
        SBUF=*(ptr+i);               //SUBF接受/发送缓冲器
        while(TI==0);
        TI=0;
        i++;    
    }
    num_rec=0;  

}
void init_all(void) //9600  11.05926
{

    SCON = 0x50;      //REN=1允许串行接受状态,串口工作模式1                  
    TMOD|= 0x20;      //定时器工作方式2                    
    PCON|= 0x80;                                                          
    TH1 = 0xFa;     //baud*2  /* reload value 9600、数据位8、停止位1。效验位无 (11.0592)             
  TL1 = 0xF3;         
    TR1  = 1;                                                             
    ES   = 1;        //开串口中断                  
    EA   = 1;        // 开总中断
    lcd_init() ;

} 
//将UTC时间转成BJ时间
void trans_time(void)
{
    unsigned char temp,hour_shi,hour_ge;
    temp=(time[0]-'0')*10+(time[1]-'0');
    if(temp<=16)
    {
        temp=temp+8;
        hour_shi=temp/10;
        hour_ge=temp%10;
        time[0]=hour_shi+'0';
        time[1]=hour_ge+'0';    
    }
    else
    {
        temp=temp+8-24;
        time[0]='0';
        time[1]=temp%10+'0';    
    }
} 
//判断是否有BD数据 有1,无0
bit BD_data(void)
{
    if(buf_full&0x01!=0)
        return 1;
    else 
        return 0;   
}
//----------------------------------------------------------------------------
void main () 
{   
    unsigned int X;
    unsigned char i;
    init_all(); 
    lcd_wcmd(0x01);         //清除LCD的显示内容
    delayms(10);
    i = 0;
    X = 0;
    lcd_pos(1);         // 设置显示位置
    while(kaijihuamian[i] != '\0')
    {
        lcd_wdat(kaijihuamian[i]);  // 显示字符时间
        i++;
    }
    delayms(10);
    i=0  ;
    lcd_pos_2(0);           // 设置显示位置
    while(kaijihuamian2[i] != '\0')
    {
        lcd_wdat(kaijihuamian2[i]); // 显示字符时间
        i++;
    }
    i = 0;
    delayms(5000);
    lcd_wcmd(0x01);         //清除LCD的显示内容
    delayms(10);
    while(1)
    {
        if(flag_data==0)// 如果没有数据
        {
            lcd_wcmd(0x01);         //清除LCD的显示内容
            delayms(10);
            i=0  ;
            lcd_pos(2);         // 设置显示位置
            while(nodata[i] != '\0')
            {
                lcd_wdat(nodata[i]);    // 显示字符
                i++;
            }
            delayms(1000);
            lcd_wcmd(0x01);         //清除LCD的显示内容
            delayms(10);    
            //第三页开始==================================================================
            DHT11_main(i);// 显示字符DHT11 温度 湿度 信息
            delaym(3000);
            i = 0;
            lcd_wcmd(0x01);         
            delaym(1);
            //第四页开始==================================================================
            lcd_pos(0);         // 设置显示位置
            while(LCD_161[i] != '\0'){
                lcd_wdat(LCD_161[i]);   // 显示字符禾灮信息
                i++;
            }
            delayms(1);
            i=0  ;
            lcd_pos_2(0);           // 设置显示位置
            while(LCD_162[i] != '\0'){
                lcd_wdat(LCD_162[i]);   // 显示字符禾灮信息
                i++;
            }   
            //结束==================================================================
            delayms(5000);
            lcd_wcmd(0x01);         //清除LCD的显示内容
            delayms(1);
        }
        if(flag_rec==1)             //data 解码完毕
        {
            flag_rec=0;  //清数据有效标志位         
            if (lock==1) //如果已经定位
            {

                for(X;X <10;X++){           
                    trans_time();//UTC--BJ TIME转换为北京时间                      
                    lcd_pos_2(0);           // 设置显示位置     
                    lcd_wdat(' ');         //显示 北京时间        
                    lcd_wdat(time[5]);  
                }
                    delayms(1);          //第一页显示完毕======经纬度+北京时间====定位+授时
                    lcd_wcmd(0x01);         //清除LCD的显示内容
                    delayms(1);
                    X = 0;
                for(X;X <10;X++){
                    //第二页开始=================================================================================
                    lcd_pos(0);         // 设置显示位置
                    lcd_wdat('H');              // 显示海拔 卫星个数
                    delayms(100);
                    lcd_pos(1);            
                    while(high[i] != '\0'){
                        lcd_wdat(high[i]);  //显示海拔
                        i++;
                    }
                    delayms(100);
                    while(use_sat[i] != '\0'){
                        lcd_wdat(use_sat[i]);   // // 显示 卫星个数
                        i++;
                    }
                    delayms(100);
                    i=0  ;  
                    
                    //delayms(5000);         //第二页显示完毕======显示海拔+卫星个数===速度+方位
                    //lcd_wcmd(0x01);           //清除LCD的内容
                            
                    while(angle[i] != '\0'){
                        lcd_wdat(angle[i]); // // 显示 方位角    
                        i++;
                    }
                    i=0  ;  
                }
                delayms(10);         //第二页显示完毕======显示海拔+卫星个数===速度+方位
                lcd_wcmd(0x01);         //清除LCD的显示内容
                delayms(1);
                X = 0;

                //第三页开始==================================================================
                DHT11_main(i);// 显示字符DHT11 温度 湿度 信息
                delaym(3000);
                i = 0;
                lcd_wcmd(0x01);         
                delaym(1);
                //第四页开始==================================================================
                lcd_pos(0);         // 设置显示位置
                while(LCD_161[i] != '\0'){
                    lcd_wdat(LCD_161[i]);   // 显示字符禾灮信息
                    i++;
                }
                delayms(1);
                i=0  ;
                lcd_pos_2(0);           // 设置显示位置
                while(LCD_162[i] != '\0'){
                    lcd_wdat(LCD_162[i]);   // 显示字符禾灮信息
                    i++;
                }   
                //结束==================================================================
                delayms(5000);

                delayms(1);         //显示完毕======显示禾灮信息
                lcd_wcmd(0x01);         //清除LCD的显示内容
                delayms(1); 
            }
            else if(lock==0)
            {      //未定位    //display relative message              
                lcd_wcmd(0x01);         //清除LCD的显示内容
                delayms(10);
                i=0  ;
                lcd_pos(3);         // 设置显示位置
                while(receiving[i] != '\0')
                {
                    lcd_wdat(receiving[i]); // 显示字符时间
                    i++;
                }
                delaym(1000);
                lcd_wcmd(0x01);         //清除LCD的显示内容
                delaym(10);                 
                X = 0;
                //第三页开始==================================================================
                DHT11_main(i);// 显示字符DHT11 温度 湿度 信息
                delaym(3000);
                i = 0;
                lcd_wcmd(0x01);         
                delaym(1);
                //第四页开始==================================================================
                lcd_pos(0);         // 设置显示位置
                while(LCD_161[i] != '\0'){
                    lcd_wdat(LCD_161[i]);   // 显示字符禾灮信息
                    i++;
                }
                delayms(1);
                i=0  ;
                lcd_pos_2(0);           // 设置显示位置
                while(LCD_162[i] != '\0'){
                    lcd_wdat(LCD_162[i]);   // 显示字符禾灮信息
                    i++;
                }   
                //结束==================================================================
                delayms(5000);
                delayms(1);                         
            }                   
        }           
    }
}

//--------------------------------------------------------------------
//串口中断程序
void ser_int (void) interrupt 4 using 1
{
    
    unsigned char tmp;
    if(RI)
    {
        RI=0;
        tmp=SBUF;
        switch(tmp)
        {
            case '$':
                cmd_number=0;       //命令类型清空
                mode=1;             //接收命令模式
                byte_count=0;       //接收位数清空
                flag_data=1;
                flag_rec=1;     //数据标志位置一
                break;
            case ',':
                seg_count++;        //逗号计数加1
                byte_count=0;
                break;
            case '*':
                switch(cmd_number)
                {
                    case 1:
                        buf_full|=0x01;
                        break;
                    case 2:
                        buf_full|=0x02;
                        break;
                    case 3:
                        buf_full|=0x04;
                        break;
                }
                mode=0;
                break;
            default:
                if(mode==1) //命令种类判断
                {
                    cmd[byte_count]=tmp;            //接收字符放入类型缓存
                    if(byte_count>=4)
                    {               //如果类型数据接收完毕,判断类型
                        if(cmd[0]=='G')
                        {
                            if(cmd[1]=='N')
                            {
                                if(cmd[2]=='G')
                                {
                                    if(cmd[3]=='G')
                                    {
                                        if(cmd[4]=='A')
                                        {
                                            cmd_number=1;      //data type
                                            mode=2;
                                            seg_count=0;
                                            byte_count=0;
                                        }
                                    }
                                    else if(cmd[3]=='S')
                                    {
                                        if(cmd[4]=='V')
                                        {
                                            cmd_number=2;
                                            mode=2;
                                            seg_count=0;
                                            byte_count=0;
                                        }
                                    }
                                }
                                
                    switch (cmd_number)
                    {
                        case 1:             //类型1数据接收。GNGGA————北斗与GPS组合数据   
                            switch(seg_count)
                            {
                                case 2:     //纬度处理
                                    if(byte_count<9)
                                    {
                                        WD[byte_count]=tmp;
                                    }
                                    break;
                                case 3:     //纬度方向处理
                                    if(byte_count<1)
                                    {
                                        WD_a=tmp;
                                    }
                                    break;
                                case 4:     //经度处理
                                    if(byte_count<10)
                                    {
                                        JD[byte_count]=tmp;
                                    }
                                    break;
                                case 5:     //经度方向处理
                                    if(byte_count<1)
                                    {
                                        JD_a=tmp;
                                    }
                                    break;
                                case 6:     //定位判断
                                    if(byte_count<1)
                                    {
                                        lock=tmp;
                                    }
                                    break;
                                case 7:     //定位使用的卫星数
                                    if(byte_count<2)
                                    {
                                        use_sat[byte_count]=tmp;
                                    }
                                    break;
                                case 9:     //高度处理
                                    if(byte_count<6)
                                    {
                                        high[byte_count]=tmp;
                                    }
                                    break;
                            }
                            break;                          
                }
                byte_count++;       //接收数位加1
                break;
        }
    }       
}

相关测试结果:


测试视频地址:点击进入


    感谢一直关注着禾灮成长进步的朋友们。你们的信任、支持和鼓励,鞭策着我们一路走到了今天。
    
    感谢所有的合作伙伴,我们相互促进,共同见证了彼此的成长。

    感谢所有曾经在禾灮彼此倚靠、相互鼓励、携手同心、砥砺同行的兄弟姐妹。这里承载了我们的青春与热血。

                禾灮,感谢有你。

    未来,我们将一如既往,砥砺前行。

                                        禾灮·小楊
                                       2018.08.10

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容

  • 60. 搜索插入位置 描述 笔记 数据 评测 给定一个排序数组和一个目标值,如果在数组中找到目标值则返回索引。如果...
    李清依阅读 178评论 0 0
  • 三月八日小谁逛崇文门新世界百货 风之声流过 丛林的花朵 蘑菇群点头微笑 你在彩灯下 四顾张望 向这个世界 问好 在...
    草田羽人阅读 339评论 0 1
  • 在这个世界上,无论怎样,无论在哪,你都可以遇见一两个足以让你感动至深的人,他或许真的卑微的如同尘埃,但他总能深深...
    沐一阙阅读 292评论 0 6