找回密码
 立即注册

QQ登录

只需一步,快速开始

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

基于单片机+GPS定位系统程序+原理图PCB设计

  [复制链接]
跳转到指定楼层
楼主
单片机是通用的51,52,stc,原理图中的.ddb、.Bkp等格式是要用protelse打开
GPS数据含义简介
Recommended Minimum Specific GPS/TRANSIT Data(RMC)推荐定位信息

$GPRMC,<1>,<2>,<3>,<4>,<5>,<6>,<7>,<8>,<9>,<10>,<11>,<12>*hh<CR><LF>

<1> UTC时间,hhmmss(时分秒)格式
<2> 定位状态,A=有效定位,V=无效定位
<3> 纬度ddmm.mmmm(度分)格式(前面的0也将被传输)
<4> 纬度半球N(北半球)或S(南半球)
<5> 经度dddmm.mmmm(度分)格式(前面的0也将被传输)
<6> 经度半球E(东经)或W(西经)
<7> 地面速率(000.0~999.9节,前面的0也将被传输)
<8> 地面航向(000.0~359.9度,以真北为参考基准,前面的0也将被传输)
<9> UTC日期,ddmmyy(日月年)格式
<10> 磁偏角(000.0~180.0度,前面的0也将被传输)
<11> 磁偏角方向,E(东)或W(西)
<12> 模式指示(仅NMEA0183 3.00版本输出,A=自主定位,D=差分,E=估算,N=数据无效)

Altium Designer画的原理图和PCB图如下:(51hei附件中可下载工程文件)


海拔速度


显示时间经纬度



制作出来的实物图如下:



单片机源程序如下:
  1. //  工程名称:        GPS模块测试程序
  2. //  文件名称:  main.c
  3. //  功能描述:         GPS模块接收定位信息,在LCD上显示
  4. //  组成文件:        main.c LCD.c GPS.c display.c  
  5. //    头文件:  LCD.h GPS.h display.h
  6. //  程序分析:        GPS模块通过串口向单片机发送固定格式的数据
  7. //                                单片机的串口接收到数据后,进行解析,在LCD上显示
  8. //                                定位信息包括:日期时间,经纬度,速度,角度,高度                                
  9. //======================================================================

  10. #include <reg52.h>
  11. #include <stdio.h>
  12. #include <string.h>
  13. #include "GPS.h"
  14. #include "LCD.h"
  15. #include "display.h"

  16. sbit led1 = P2^3;  //接收数据指示灯
  17. sbit led2 = P2^4;  //GPRMC数据有效指示灯
  18. sbit led3 = P2^5;  //GPGGA数据有效指示灯

  19. #define   REV_YES    led1 = 0
  20. #define   REV_NO     led1 = 1
  21. #define   RMC_YES    led2 = 0
  22. #define   RMC_NO     led2 = 1
  23. #define   GGA_YES    led3 = 0
  24. #define   GGA_NO     led3 = 1

  25. char   xdata rev_buf[80];        //接收缓存
  26. uchar  xdata rev_start = 0;     //接收开始标志
  27. uchar  xdata rev_stop  = 0;     //接收停止标志
  28. uchar  xdata gps_flag = 0;      //GPS处理标志
  29. uchar  xdata change_page = 0;   //换页显示标志
  30. uchar  xdata num = 0;           //

  31. extern GPS_INFO   GPS;  //在display.c中定义,使用时要加extern


  32. void Uart_Init(void)
  33. {
  34.         TMOD = 0x21;        //0010 0001
  35.         PCON=0X00;
  36.         TH0=0x3c;
  37.         TL0=0xb0;
  38.         TH1=0xfd;                 //
  39.         TL1=0xfd;                //

  40.         TR1=1;     //开启定时器1
  41.         REN=1;     //允许接收数据   
  42.         SM0=0;
  43.         SM1=1;

  44.         TI=0;
  45.         RI=0;

  46.         EA=1;    //开总中断
  47.         ES=1;    //串口1中断允许
  48.         ET0 = 1; //定时器1中断允许
  49. }

  50. /****************************************
  51. 主函数        
  52. /****************************************/
  53. void main(void)
  54. {
  55.         uchar error_num = 0;

  56.         Uart_Init();  //初始化串口
  57.         Lcd_Init();          //初始化LCD
  58.         GPS_Init();   //初始化GPS
  59.         rev_stop=0;
  60.         REV_NO;
  61.         while(1)
  62.         {
  63.                 if (rev_stop)   //如果接收完一行
  64.                 {
  65.                         TR0 = 1;   //开启定时器
  66.                         REV_YES;
  67.                         if (change_page % 2 == 1)  //换页
  68.                         {
  69.                                 if (GPS_GGA_Parse(rev_buf, &GPS))  //解析GPGGA
  70.                                 {
  71.                                         GGA_YES;
  72.                                         GPS_DisplayTwo();  //显示第二页信息
  73.                                         error_num = 0;
  74.                                         gps_flag = 0;
  75.                                         rev_stop  = 0;
  76.                                         REV_NO;
  77.                                 }
  78.                                 else
  79.                                 {
  80.                                         error_num++;
  81.                                         if (error_num >= 20) //如果数据无效超过20次
  82.                                         {
  83.                                                 GGA_NO;
  84.                                                 error_num = 20;
  85.                                                 GPS_Init();     //返回初始化
  86.                                         }
  87.                                         gps_flag = 0;
  88.                                         rev_stop  = 0;
  89.                                         REV_NO;
  90.                                 }

  91.                         }
  92.                         else
  93.                         {
  94.                                 if (GPS_RMC_Parse(rev_buf, &GPS)) //解析GPRMC
  95.                                 {
  96.                                         RMC_YES;

  97.                                         GPS_DisplayOne();          //显示GPS第一页信息
  98.                                         error_num = 0;
  99.                                         gps_flag = 0;
  100.                                         rev_stop  = 0;
  101.                                         led1 = 1;        
  102.                                 }
  103.                                 else
  104.                                 {
  105.                                         error_num++;
  106.                                         if (error_num >= 20) //如果数据无效超过20次
  107.                                         {
  108.                                                 RMC_NO;
  109.                                                 error_num = 20;
  110.                                                 GPS_Init();     //返回初始化
  111.                                         }
  112.                                         gps_flag = 0;
  113.                                         rev_stop  = 0;
  114.                                         REV_NO;
  115.                                 }
  116.                         }
  117.                 }
  118.         }
  119. }


  120. void timer0(void) interrupt 1
  121. {
  122.         static uchar count = 0;
  123.         TH0 = 0x3c;
  124.         TL0 = 0xb0;
  125.         count++;
  126.         if (count == 200)  //2*5秒钟
  127.         {
  128.                 count = 0;
  129.                 change_page++;  //换页
  130.                 if (change_page == 10)
  131.                         change_page = 0;
  132.         }        
  133. }


  134. void Uart_Receive(void) interrupt 4
  135. {
  136.         uchar ch;
  137.         ES = 0;
  138.         if (RI)
  139.         {
  140.                 ch = SBUF;
  141.                 if ((ch == '


  142. ) && (gps_flag == 0))  //如果收到字符'


  143. ,便开始接收
  144.                 {
  145.                         rev_start = 1;
  146.                         rev_stop  = 0;
  147.                 }
  148.         
  149.                 if (rev_start == 1)  //标志位为1,开始接收
  150.                 {
  151.                         rev_buf[num++] = ch;  //字符存到数组中
  152.                         if (ch == '\n')     //如果接收到换行
  153.                         {
  154.                                 rev_buf[num] = '\0';
  155.                                 rev_start = 0;
  156.                                 rev_stop  = 1;
  157.                                 gps_flag = 1;
  158.                                 num = 0;
  159.                         }
  160.                 }
  161.         }
  162.         RI = 0;         //RI清0,重新接收
  163.         ES = 1;        
  164. }
复制代码

所有资料51hei提供下载:
基于单片机GPS定位系统设计.7z (812.52 KB, 下载次数: 572)

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

使用道具 举报

来自 2#
ID:16255 发表于 2022-5-10 23:39 | 只看该作者
发现一个小Bug。下面这个是楼主GPS.c中的代码,作用是将读到的字符串转换成浮点数。(浮点数可表示正负小数)
static float Str_To_Float(char *buf)
{
        float rev = 0;
        float dat;
        int integer = 1;
        char *str = buf;
        int i;
        while(*str != '\0')
        {
                switch(*str)
                {
                        case '0':
                                dat = 0;
                                break;
                        case '1':
                                dat = 1;
                                break;
                        case '2':
                                dat = 2;
                                break;               
                        case '3':
                                dat = 3;
                                break;
                        case '4':
                                dat = 4;
                                break;
                        case '5':
                                dat = 5;
                                break;
                        case '6':
                                dat = 6;
                                break;
                        case '7':
                                dat = 7;
                                break;
                        case '8':
                                dat = 8;
                                break;
                        case '9':
                                dat = 9;
                                break;
                        case '.':
                                dat = '.';
                                break;
                }
                if(dat == '.')
                {
                        integer = 0;
                        i = 1;
                        str ++;
                        continue;
                }
                if( integer == 1 )
                {
                        rev = rev * 10 + dat;
                }
                else
                {
                        rev = rev + dat / (10 * i);
                        i = i * 10 ;
                }
                str ++;
        }
        return rev;
}
代码中似乎没有考虑到负数的情况。
而实际上是有可能出现负数的。如下图,读出来-5.7m。


但是,我并没有打算改它,因为我用不着这个海拔数据。
而且我把数据都以int的形式储存,更方便处理数据。GPS报文中的数据都是两位小数的,将其乘上100就都能变成整数,就不用以浮点数储存了(int占两个字节,float占四个字节,double占八个字节)。

评分

参与人数 1黑币 +80 收起 理由
admin + 80 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

板凳
ID:626079 发表于 2020-3-30 10:39 | 只看该作者
历害,向你学习…… GPS是不是要退休了, 能搞一个我们的北斗不嘛
回复

使用道具 举报

地板
ID:523982 发表于 2020-4-10 16:33 | 只看该作者
xmfjfhcel 发表于 2020-3-30 10:39
历害,向你学习…… GPS是不是要退休了, 能搞一个我们的北斗不嘛

有北斗GPS双模的,网络上都有教程。
回复

使用道具 举报

5#
ID:523982 发表于 2020-4-10 16:34 | 只看该作者
不错不错,去年做过STC12+GPS+LCD12864+语音播报的,最近整理下,也发出来。
回复

使用道具 举报

6#
ID:723764 发表于 2020-4-12 20:29 | 只看该作者
jakcypater 发表于 2020-4-10 16:34
不错不错,去年做过STC12+GPS+LCD12864+语音播报的,最近整理下,也发出来。

蹲一个大神
回复

使用道具 举报

7#
ID:727275 发表于 2020-4-13 13:08 | 只看该作者
感谢楼主分享基于单片机GPS定位系统设计.
回复

使用道具 举报

8#
ID:731957 发表于 2020-4-18 12:39 | 只看该作者
jakcypater 发表于 2020-4-10 16:33
有北斗GPS双模的,网络上都有教程。

网上哪里有资源呀
回复

使用道具 举报

9#
ID:631770 发表于 2020-4-19 10:43 来自手机 | 只看该作者
jakcypater 发表于 2020-4-10 16:34
不错不错,去年做过STC12+GPS+LCD12864+语音播报的,最近整理下,也发出来。

大佬你好,能给个球球号问些问题吗……
回复

使用道具 举报

10#
ID:731957 发表于 2020-4-20 09:59 | 只看该作者
这个代码改一下可以用于BDS嘛?
回复

使用道具 举报

11#
ID:709306 发表于 2020-4-26 18:02 | 只看该作者
jakcypater 发表于 2020-4-10 16:34
不错不错,去年做过STC12+GPS+LCD12864+语音播报的,最近整理下,也发出来。

能不能发一下程序还有实物图或者资料包啊?
回复

使用道具 举报

12#
ID:64253 发表于 2020-8-12 10:51 | 只看该作者
先MARK一下,坐等北斗的资料,学习
回复

使用道具 举报

13#
ID:650894 发表于 2021-3-24 16:27 来自手机 | 只看该作者
gps用的哪个芯片
回复

使用道具 举报

14#
ID:284050 发表于 2021-7-7 22:34 | 只看该作者
确定GPS准确吗(能转换地图中准确定位吗)
回复

使用道具 举报

15#
ID:99130 发表于 2022-2-20 16:02 | 只看该作者
GPS的型号是哪种?
回复

使用道具 举报

16#
ID:1019040 发表于 2022-4-17 01:47 来自手机 | 只看该作者
头文件在附件里面可以下载
回复

使用道具 举报

17#
ID:16255 发表于 2022-5-11 16:49 | 只看该作者
xiamo123 发表于 2020-4-20 09:59
这个代码改一下可以用于BDS嘛?

完全可以。如果单纯是定位,不用改代码,GPS模块从硬件上就支持北斗定位的。
只是卫星列表有点差异,GPS是用$GPGSV报文,北斗是用$BDGSV报文,但他们的格式是一样的,稍稍改一下就能读取北斗列表。
回复

使用道具 举报

18#
ID:1086148 发表于 2023-7-17 21:44 | 只看该作者
赞楼主! 你好! 非常感兴趣另一位工程师说的GPS语音播报!期待看到你发表作品!
回复

使用道具 举报

19#
ID:1085441 发表于 2023-7-18 01:41 | 只看该作者
给大家推荐一个北斗GPS双模模块,HT1612,几年前买的时候是十几元,挺好用的。
回复

使用道具 举报

20#
ID:840718 发表于 2023-8-2 21:42 | 只看该作者
历害,向你学习…… GPS是不是要退休了, 能搞一个我们的北斗?
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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