怎样写一个GPS应用程序——介绍
一句重要的语句
第一部分将探讨描述GPS原始数据的工作。实际上这个工作已经被国家舰船电子协会(下面简称NMEA,www.nmea.org)给简化了许多,该网站介 绍了一个正广泛应用于GPS主流设备的工业标准。为了给广大的开发人员一个良好的开端,我选择使用的一些VS.NET的源代码来自我的“GPS.NET Global Position SDK”组件。(为了简短这些代码,我去掉了诸如多线程和错误处理的部分。)
NMEA数据通过一个“逗号分隔的语句”来传递,这个语句包含的信息都基于语句的第一个单词。这里有五十种以上类型的语句,不过真正的一个描述仅仅需要处理少量的采集数据。最终常用的NMEA语句是“推荐最小”语句,它以“$GPRMC.”开头。这里有个一例子:
$GPRMC,040302.663,A,3939.7,N,10506.6,W,0.27,358.86,200804,,*1A
这条语句基本上包含了GPS应用程序所需的全部数据:纬度、经度、速度、方向、卫星时间、状态以及磁场变量。
第一步就是要写一个方法来解释NMEA数据,这个方法需要完成两件事:将每条语句分解为独立的单词,和检查首单词是否有效。清单 1-1 就是这个描述类的开始部分。
清单 1-1:一个NMEA描述的核心功能是将NMEA语句分解成单个的单词。
'** Listing 1-1. The core of an NMEA interpreter
'*******************************************************
Public Class NmeaInterpreter
' 处理来自GPS接收器的信息
Public Function Parse(ByVal sentence As String) As Boolean
' 将语句分解为单词
Dim Words() As String = GetWords(sentence)
' 通过匹配首单词来决定下一步的工作
Select Case Words(0)
Case "$GPRMC" ' 一条“推荐最小”的语句被找到!
' 标示这条语句为可用
Return True
Case Else
' 标示这条语句为不可用
Return False
End Select
End Function
' 将语句分解为单词
Public Function GetWords(ByVal sentence As String) As String()
Return sentence.Split(","c)
End Function
End Class
接下来一步就是要分析提取出来的信息了,让我们从纬度和经度开始吧。纬度和经度存储格式为“DDD°MM’SS.S,”,其中D表示时(也可以叫 “度”),M表示分,S表示秒。坐标可以简单表示为“DD°MM.M’”或者直接就表示为“DD°.”语句中第四个单词(“3939.7,”)以度和分的 形式表示当前的纬度为39°39.7’。头两个字符(39)表示度,余下的部分(39.7)表示分。经度和纬度是相同的结构,从这个语句来看要注意,头三 个字符表示经度的度(105°06.6’)。第五和第七个单词标示“半球”,其中“N”代表“北半球”,“W”代表“西经”。半球信息放在数字部分后面组 成完整的测量信息。
我还发现NMEA描述在事件驱动下使工作更容易完成。这是因为并不是按照特定的顺序来获取数据。一个事件驱动类为一应用程序给出了它最具灵活性和响应性的 描述。因此,我也将使用事件来设计描述,从而获取信息。PositionReceived是第一个事件,它将记录当前获得的纬度和经度。清单 1-2 扩展这个描述器,来报告当前位置。
清单 1-2:这个描述器可以报告当前的纬度和经度。
'*******************************************************
'************* 清单 1-2. 从语句中提取信息 *************
'*******************************************************
Public Class NmeaInterpreter
' 当前位置变化时,记录新位置
Public Event PositionReceived(ByVal latitude As String, _
ByVal longitude As String)
' 处理GPS接收器的信息
Public Function Parse(ByVal sentence As String) As Boolean
' 通过匹配首单词来决定下一步的工作
Select Case GetWords(sentence)(0)
Case "$GPRMC" ' 一条“推荐最小”的语句被找到!
Return ParseGPRMC(sentence)
Case Else
' 标示这条语句为不可用
Return False
End Select
End Function
' 将语句分解为单词
Public Function GetWords(ByVal sentence As String) As String()
Return sentence.Split(","c)
End Function
' 描述 $GPRMC 消息
Public Function ParseGPRMC(ByVal sentence As String) As Boolean
' 将语句分解为单词
Dim Words() As String = GetWords(sentence)
' 我们是否有足够的数据来描述当前位置?
If Words(3) <> "" And Words(4) <> "" And Words(5) <> "" And _
Words(6) <> "" Then
' 是,则提取纬度和经度
Dim Latitude As String = Words(3).Substring(0, 2) & "°" ' 获取度
Latitude = Latitude & Words(3).Substring(2) & """" ' 获取分
Latitude = Latitude & Words(4) ' 获取半球
Dim Longitude As String = Words(5).Substring(0, 3) & "°" ' 获取度
Longitude = Longitude & Words(5).Substring(3) & """" ' 获取分
Longitude = Longitude & Words(6) ' 获取半球
' 将该变化通知应用程序
RaiseEvent PositionReceived(Latitude, Longitude)
End If
' 表示该语句可用
Return True
End Function
End Class
当然,我们肯定注意到了当没有接受到任何信息时,有些GPS设备将报告空值。因此,在解意描述前先判断每个单词是一个很不错的想法。如果你需要打度的符号(°),那么按Alt健并且用数字键盘输入“0176”就OK了。
检验和(Checksum)就是采用XOR的方法来计算美元符($)和星号(*)之间的字节。这个检验和是用来与语句的检验和比较的。如果检验和不匹配, 语句将被弃用。这个做法是很有效的,因为GPS设备每隔很短的时间就返回一批相同的信息。利用检验和的比较能力,描述就可以剔除每一无效的语句。清单 1-3 扩展描述器类来完成这个功能。
清单 1-3:这个描述器可以检测错误并且仅仅解意释放错误NMEA数据
'** 清单 1-3. 检测并处理NMEA错误
'*******************************
Public Class NmeaInterpreter
' 当前位置变化时,记录新位置
Public Event PositionReceived(ByVal latitude As String, _
ByVal longitude As String)
' 处理GPS接收器的信息
Public Function Parse(ByVal sentence As String) As Boolean
' 如果检验和不能和我们计算的检验和匹配,那么丢弃它
If Not IsValid(sentence) Then Return False
' 通过匹配首单词来决定下一步的工作
Select Case GetWords(sentence)(0)
Case "$GPRMC" ' 一条“推荐最小”的语句被找到!
Return ParseGPRMC(sentence)
Case Else
' 标示这条语句为不可用
Return False
End Select
End Function
' 将语句分解为单词
Public Function GetWords(ByVal sentence As String) As String()
Return sentence.Split(","c)
End Function
' 描述 $GPRMC 消息
Public Function ParseGPRMC(ByVal sentence As String) As Boolean
' 将语句分解为单词
Dim Words() As String = GetWords(sentence)
' 我们是否有足够的数据来描述当前位置?
If Words(3) <> "" And Words(4) <> "" And Words(5) <> "" And _
Words(6) <> "" Then
' 是,则提取纬度和经度
Dim Latitude As String = Words(3).Substring(0, 2) & "°" ' 获取度
Latitude = Latitude & Words(3).Substring(2) & """" ' 获取分
Latitude = Latitude & Words(4) ' 获取半球
Dim Longitude As String = Words(5).Substring(0, 3) & "°" ' 获取度
Longitude = Longitude & Words(5).Substring(3) & """" ' 获取分
Longitude = Longitude & Words(6) ' 获取半球
' 将该变化通知应用程序
RaiseEvent PositionReceived(Latitude, Longitude)
End If
' 表示该语句可用
Return True
End Function
' 如果检验和匹配,则返回TRUE
Public Function IsValid(ByVal sentence As String) As Boolean
' 星号计算后比较字符
Return sentence.Substring(sentence.IndexOf("*") + 1) = GetChecksum(sentence)
End Function
' 计算语句的检验和
Public Function GetChecksum(ByVal sentence As String) As String
' 通过循环获得检验和
Dim Character As Char
Dim Checksum As Integer
For Each Character In sentence
Select Case Character
Case "$"c
' 忽略美元符号
Case "*"c
' 遇到星号则停止计算
Exit For
Case Else
' 是否为检验和的第一个值?
If Checksum = 0 Then
' 是。将该值赋给检验和
Checksum = Convert.ToByte(Character)
Else
' 否。采用XOR算法和当前检验和值计算,并将结构赋给检验和
Checksum = Checksum Xor Convert.ToByte(Character)
End If
End Select
Next
' 返回一个十六进制的检验和
Return Checksum.ToString("X2")
End Function
End Class
最新文章
- 怎样写一个GPS应用程序——介绍 [11-03]
- 浅谈GPS与GIS的学科关系 [07-19]
- 车载导航领域 四维地图便携式导航市 [03-15]
- 凯立德成立发展十周年 畅谈GPS发展 [12-28]
- 亚洲导航展览会将于2007年11月在中 [10-29]
- GPS编码格式分析及C语言的解码代码 [09-24]
- 全球定位系统的组成及应用基础 [09-18]
- 空间导航数据库技术的特点与要求 [08-08]
- GPS在大地控制测量中的应用与实践 [08-08]
- GPS全球卫星定位技术在摄影测量中有 [08-08]
推荐文章


热点文章
目前世界上的几种全球定位系统
编写你自己的GPS应用程序:第二部
全球卫星导航系统的发展概况
GPS名词解释词典
TITAN GPS车辆监控系统介绍
亚洲导航展览会将于2007年11月在
全球定位系统(GPS)介绍
GPS RTK技术在地籍测量中的应用
GPS技术在土地测绘地籍控制测量的
空间导航数据库技术的特点与要求
GPS定位原理概述
GPS在地形、地籍及房地产测量中的
GPS在林业面积测绘中的应用
GPS在大地控制测量中的应用与实践
GPS在公路工程控制测量中的应用
GPS编码格式分析及C语言的解码代
GPS全球卫星定位技术在摄影测量中
全球定位系统的组成及应用基础
怎样写一个GPS应用程序——介绍
浅谈GPS与GIS的学科关系
车载导航领域 四维地图便携式导航
凯立德成立发展十周年 畅谈GPS发

