arduino 2560GPS数据的处理

最近做个小项目,需要实现导航功能,用到了GPS模块,需要对GPS数据进行处理,但是arduino2560的float和double都只是4个字节,无法用一个变量存储经纬度数据,于是想用一个float数组分开存储,具体实现如下!

我的GPS模块返回的经纬度使用一个字符数组存储的,长度是11,下面是GPS模块返回数据的结构体:

struct
{
  char GPS_Buffer[80];
  bool isGetData;        //是否获取到GPS数据
  bool isParseData;    //是否解析完成
  char UTCTime[11];        //UTC时间
  char latitude[11];        //纬度
  char N_S[2];        //N/S
  char longitude[12];        //经度
  char E_W[2];        //E/W
  char vec[11];
  char yaw[11];
  bool isUsefull;        //定位信息是否有效
}

我直接贴出代码:

/*****************************************************************************

将字符串表示的字符串转换为两个浮点型数据,保存在一个浮点型数组内

*****************************************************************************/

void la_lo(char *str1, char *str2)
{
  int si, i;
  char **ret = explode('.', str1, &si);
  for (i = 0; i < si; i++)
  {
    x[i] = atof(ret[i]);
    free(ret[i]);
  }
  free (ret);
  while (x[1] / 10000 < 1)
  {
    x[1] = x[1] * 10;
  }
  int i1 = search(str1, '.');
  //Serial.print("weizhi: ");
  //Serial.print(i1);
  if ((str1[i1 + 1] == '0') && (str1[i1 + 2] != '0'))
  {
    x[1] = x[1] / 10;
  }
  else if (((str1[i1 + 1] == '0') && (str1[i1 + 2] == '0')) && (str1[i1 + 3] != '0'))
  {
    x[1] = x[1] / 100;
  }
  else if (((str1[i1 + 1] == '0') && (str1[i1 + 2] == '0') && (str1[i1 + 3] == '0')) && (str1[i1 + 4] != '0'))
  {
    x[1] = x[1] / 1000;
  }
  else if (((str1[i1 + 1] == '0') && (str1[i1 + 2] == '0') && (str1[i1 + 3] == '0') && (str1[i1 + 4] == '0')) && (str1[i1 + 5] != '0'))
  {
    x[1] = x[1] / 10000;
  }
  else if ((str1[i1 + 1] == '0') && (str1[i1 + 2] == '0') && (str1[i1 + 3] == '0') && (str1[i1 + 4] == '0') && (str1[i1 + 5] == '0'))
  {
    x[1] = 0;
  }
  //分离整数部分,小数部分
  int tem1 = int(x[0]) % 10;
  int tem2 = tem1 + (int(x[0]) - tem1) % 100;
  x[0] = x[0] / 100.0;
  x[1] = tem2 / 100.0 + x[1] / 10000000.0;
  x[0] = int(x[0]);

  **ret = explode('.', str2, &si);
  for (i = 0; i < si; i++)
  {
    x[i + 2] = atof(ret[i]);
    free(ret[i]);
  }
  free (ret);
  while (x[3] / 10000 < 1)
  {
    x[3] = x[3] * 10;
  }
  int i2 = search(str2, '.');
  //Serial.print("weizhi: ");
  //Serial.print(i1);
  if ((str2[i2 + 1] == '0') && (str2[i2 + 2] != '0'))
  {
    x[3] = x[3] / 10;
  }
  else if ((str2[i2 + 1] == '0') && (str2[i2 + 2] == '0') && (str2[i2 + 3] != '0'))
  {
    x[3] = x[3] / 100;
  }
  else if ((str2[i2 + 1] == '0') && (str2[i2 + 2] == '0') && (str2[i2 + 3] == '0') && (str2[i2 + 4] != '0'))
  {
    x[3] = x[3] / 1000;
  }
  else if ((str2[i2 + 1] == '0') && (str2[i2 + 2] == '0') && (str2[i2 + 3] == '0') && (str2[i2 + 4] == '0') && (str2[i2 + 5] != '0'))
  {
    x[3] = x[3] / 10000;
  }
  else if ((str2[i2 + 1] == '0') && (str2[i2 + 2] == '0') && (str2[i2 + 3] == '0') && (str2[i2 + 4] == '0') && (str2[i2 + 5] == '0'))
  {
    x[3] = 0;
  }
  //分离整数部分,小数部分
  tem1 = int(x[2]) % 10;
  tem2 = tem1 + (int(x[2]) - tem1) % 100;
  x[2] = x[2] / 100.0;
  x[3] = tem2 / 100.0 + x[3] / 10000000.0;
  x[2] = int(x[2]);
}

/*****************************************************************************************

以特定字符分离字符数组,此处特定字符是‘.’,字符数组是经纬度

*****************************************************************************************/
char **explode(char sep, const char *str, int *size)
{
  int count = 0, i;
  for (i = 0; i < strlen(str); i++)
  {
    if (str[i] == sep)
    {
      count ++;
    }
  }

  char **ret = calloc(++count, sizeof(char *));

  int lastindex = -1;
  int j = 0;

  for (i = 0; i < strlen(str); i++)
  {
    if (str[i] == sep)
    {
      ret[j] = calloc(i - lastindex, sizeof(char)); //分配子串长度+1的内存空间
      memcpy(ret[j], str + lastindex + 1, i - lastindex - 1);
      j++;
      lastindex = i;
    }
  }
  //处理最后一个子串
  if (lastindex <= strlen(str) - 1)
  {
    ret[j] = calloc(strlen(str) - lastindex, sizeof(char));
    memcpy(ret[j], str + lastindex + 1, strlen(str) - 1 - lastindex);
    j++;
  }

  *size = j;

  return ret;
}

/**************************************************************************************

找到目标字符的位置

**************************************************************************************/

int search(const char *a, const char b)
{
  int i;
  for (i = 0; a[i]; i++)
    if (a[i] == b)
      return i;
}

核心就是这三个函数,可以实现经纬度的字符数组形式,转换为浮点型数据,进而可以直接用到你需要的地方。当然,这个方法适用于float类型变量长度不够的情况,如果你的运行环境浮点型类型长度够长,则可以不必使用这种方法。

这是一篇关于动手制作基于arduino 和12864液晶模块的图形化显示GPS导航系统的帖子,作品效果和成本可能无法和TB上的导航产品媲美,但是动手过程会带给你乐趣和知识。 该设计根据功能分为三个版本。(具体详见附件内容) 最终版本视频演示: 动手之前最好能具备一些背景知识: 1. arduino 相关基础; 2. 能使用12864 液晶模块; 3. 能够用arduino 通过串口通信获取GPS模块定位信息; 4. 能够用arduino 操作SD卡模块; 制作所需主要硬件: arduino UNO 1片; 12864 液晶模块 1片; GPS模块1片; SD卡模块及SD卡 1套; 如截图: 原理: 将地图数据依据瓦片算法存储在SD卡中,通过串口获取GPS定位信息并从中解析出经纬度坐标,依据经纬度坐标读取相应地图数据显示在12864液晶模块上,同时显示定位坐标点。 1. 地图存储算法——瓦片系统(Maps Tile System) 本制作采用的地图数据和地图存储算法来源于微软的bing maps并做了相应修改。 具体可参考: Bing Maps Tile System[1*] Virtual Earth Tile Image URI 参数解析 Goolge and Microsoft Map Url Parameters 在瓦片系统中地图采用金字塔式的分层存储结构,不同层具有不同级别的地图分辨率(地图精细程度),每一层地图被分割成等像素大小(256X256)的瓦片,算法要解决的问题就是给定经纬度坐标和缩放级别(层索引)得到具体相应的瓦片编号。 在连接[1*]的最后有算法实现的代码可共参考。 2. 针对12864液晶模块的设计 12864液晶模块是128像素宽64像素高的单色液晶显示模块,本制作为了适应模块显示做出了两个设计。 1). 将256X256像素的瓦片裁切成128X64像素大小的8份 子瓦片。 如下图所示: 2). 通过阈值方法将8位png索引图像(bing maps 的道路数据)转换成二进制地图数据文件,为了能够显示原图中的文字信息,采用多阈值提取求或方法提取原地图中背景、地物和标注文字数据,由于标注文字和背景之间的扰动,提取效果有待改进。 显示效果: 生成地图程序(需要连接互联网): 通过设置地图范围经纬度信息获取要使用定位的区域,可通过google earth 等能显示经纬度的软件或网页获取经纬度的最大最小值(上大下小,右大左小)。缩放级别建议设置范围1~15,较大地图范围和较高缩放级别会增加地图下载、显示加载的时间。 注意:上面是对GPS导航V0.1版本的具体介绍,其他的V0.2和V0.3版本详见附件内容。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值