当前位置:   article > 正文

qgis 源码学习之core的gps模块_qgis源码解析

qgis源码解析

我已经编译出了qgis2.4的源码,进行了一些简单的二次开发,但是仍然感觉不爽,这么优秀的软件就单单在它上层做开发,太浪费了,我想学习qgis的源码的架构,顺便学习qgis的二次开发,为我将来的自己架构自己的软件吸取些经验。

qgis的文档还是比较少的,并且1.0版本和2.0以上的版本差别很大,而中文的文档绝大部分针对1.0版本,如果照搬的话会导致编译出现问题。我以下的博文是边看编写。

1.qgis的源码文档如何找到?

http://qgis.org/api/2.4/files.html 这里是找到的。

2.qgis的源码的核心部分主要分为:analysis;core;gui;plugin;4个部分。


上面4个模块的依赖关系依次是:

analysis模块:

gui模块:

Plugin模块:

core模块:

上面4个模块的core和plugin模块均是独立存在的,而其余两个模块均依赖于core模块,所以我打算首先读core模块。

从上面的core模块的内部图可以看出core模块中的gps模块是比较独立的,首先读gps模块。

gps模块里面有一个很重要的概念就是“NMEA”,NMEA是美国国家海洋电子协会(National Marine Electronics Association)的简称,它规定了gps数据的协议,如果解析过gps数据就知道“#BESPOSA"之类的东西就是由它规定的,gps模块规定

config.h里面规定了NMEA协议的版本、NMEA_VERSION_PATCH 补充协议版本,产生调试”assert“的一些宏NMEA_ASSERT(x)。

contex.h定义了输出调试信息的函数:nmea_trace、nmea_trace_buff、nmea_error;对应的.c文件就是对应的实现。

info.h定义了各种NMEA用到的各种数据结构。_nmeaPOS 位置信息里面只有两个变量一个经度另一个是纬度,所以qgis只能是二维的不可能是三维的地图软件。nmeaSATELLITE卫星信息。nmeaSATINFO是nmeaSATELLITE的包装:里面描述了有多少卫星是可见的,有多少卫星是可用的。;nmeaINFO是gps解析数据的结构体。

下面的gmath.h使用了这个info.h文件。这个文件在gps模块是个基础的头文件。

gmath.h定义了数学函数例如定义了PI、地球的平均半径等宏;角度弧度转化函数(nmea_degree2radian);NMEA degree(NMEA度)和弧度、角度转化。计算位置精度因子(pdop)、精度衰减因子(PDP)函数:nmea_calc_pdop等。位置转化函数:主要是由原始的gps字符串转化为逻辑量,例如:nmea_info2pos是由NMEA的原始数据解析后倒入POS结构中里里面的实现也很简单:

  1. pos->lat = nmea_ndeg2radian( info->lat );
  2. pos->lon = nmea_ndeg2radian( info->lon );

nmeatime.h 定义了一个“nmeaTIME"的结构体,和一个获取当前时间的函数nmea_time_now。

nmea_time_now函数在time.c中定义,调用Windows的GetSystemTime函数,所以这个函数是调用的系统时间。

parse.h解析gps数据包用的各个函数

  1. <span style="font-size:10px;">int nmea_pack_type( const char *buff, int buff_sz );//包的类型
  2. int nmea_find_tail( const char *buff, int buff_sz, int *res_crc );//寻找包尾,并进行crc校验
  3. int nmea_parse_GPGGA( const char *buff, int buff_sz, nmeaGPGGA *pack );//解析GPGGA包
  4. int nmea_parse_GPGSA( const char *buff, int buff_sz, nmeaGPGSA *pack );
  5. int nmea_parse_GPGSV( const char *buff, int buff_sz, nmeaGPGSV *pack );
  6. int nmea_parse_GPRMC( const char *buff, int buff_sz, nmeaGPRMC *pack );
  7. int nmea_parse_GPVTG( const char *buff, int buff_sz, nmeaGPVTG *pack );
  8. void nmea_GPGGA2info( nmeaGPGGA *pack, nmeaINFO *info );//从pack填充info结构,下同
  9. void nmea_GPGSA2info( nmeaGPGSA *pack, nmeaINFO *info );
  10. void nmea_GPGSV2info( nmeaGPGSV *pack, nmeaINFO *info );
  11. void nmea_GPRMC2info( nmeaGPRMC *pack, nmeaINFO *info );
  12. void nmea_GPVTG2info( nmeaGPVTG *pack, nmeaINFO *info );</span>

parser.h定义了一个解析结构体,但是这里面的函数从来没有在qgis里面用过,主要使用parse.h的函数,所以这个文件可以略过。。。。

unites.h定义了3个单位与米进行换算的公式:分别是

  1. #define NMEA_TUD_YARDS (1.0936) /**< Yeards, meter * NMEA_TUD_YARDS = yard 码*/
  2. #define NMEA_TUD_KNOTS (1.852) /**< Knots, kilometer / NMEA_TUD_KNOTS = knot节 */
  3. #define NMEA_TUD_MILES (1.609) /**< Miles, kilometer / NMEA_TUD_MILES = mile 英里 */

tok.h定义了几个函数,主要有格式化输入输出,字符串转int,字符串转float。

  1. int nmea_calc_crc( const char *buff, int buff_sz );//计算buffer中的crc,并返回crc值
  2. int nmea_atoi( const char *str, size_t str_sz, int radix );//转化成int
  3. double nmea_atof( const char *str, int str_sz );//字符串转float
  4. int nmea_printf( char *buff, int buff_sz, const char *format, ... );//格式化输出
  5. int nmea_scanf( const char *buff, int buff_sz, const char *format, ... );//格式化输出

sentence.h定义了gps的rawData的集中数据类型和各种结构nmeaPACKTYPE等。和info.h一样这也是很基础的文件。


下面的文件是qgs独有的文件,上面的文件是nmea库的,是qgis借用nmea的。

qgsgpsconnection.h里面定义了自己的QgsSatelliteInfo(gps卫星信息)和QgsGPSInformation(gps信息)这两个struct。这两个struct与nmea在info.h定义的两个struct类似。QgsGPSConnection是一个抽象的gps设备类,定义了一个parseData接口,在使用这个类的时候,首先传入一个QIODevice设备,这类调用parseData接口然后就可以解析数据了。很牛逼的写法,值得学习。实现了硬件层和软件层的隔离。connect和close函数用来打开关闭设备。它是如何实现设备与解析结合的呢?

  1. QgsGPSConnection::QgsGPSConnection( QIODevice* dev ): QObject( 0 ), mSource( dev ), mStatus( NotConnected )
  2. {
  3. clearLastGPSInformation();
  4. QObject::connect( dev, SIGNAL( readyRead() ), this, SLOT( parseData() ) );
  5. }

它是通过调用一个qt的connect来实现的。

这个类里其他主要的函数有

  1. currentGPSInformation//得到当前的gps状态
  2. status//得到当前硬件的状态,是否正常工作等。

QgsGPSConnectionRegistry类用来注册QgsGPSConnection类。这个类是一个单例的。里面采用Qset来实现。instance函数采用一个static的变量来实现。这种写法要比定义一个成员变量要简洁,并且不用使用锁。
  1. QgsGPSConnectionRegistry* QgsGPSConnectionRegistry::instance()
  2. {
  3.   static QgsGPSConnectionRegistry mInstance;
  4.   return &mInstance;
  5. }
QgsNMEAConnection 类是QgsGPSConnection的一个具体的实现,作用是解析来自gps的NMEA数据,主要是实现了QgsGPSConnection的parseData函数。QgsGpsdConnection类继承自QgsNMEAConnection类,主要是实现对gpsd守护进程的监视。gpsd是监视来自usb或者串口的gps数据,然后通过tCP 的2947 端口把数据发送出去,gpsd的网址:http://www.catb.org/gpsd
  1. class CORE_EXPORT QgsGPSDetector : public QObject
  2. {
  3. Q_OBJECT
  4. public:
  5. QgsGPSDetector( QString portName );
  6. ~QgsGPSDetector();
  7. /*通过调用qt提供的函数枚举所有的串口
  8. */
  9. static QList< QPair<QString, QString> > availablePorts();
  10. public slots:
  11. /*
  12. 函数里面定义了一个timmer时间为2000毫秒,每2000秒调用一次advance函数。
  13. 尝试打开mPortList里面存储的所有的串口,每个串口尝试使用mBaudList里面存储的波特率。
  14. 如果执行完串口的open函数后,把detected函数connect给gps信息改变slot。
  15. */
  16. void advance();
  17. void detected( const QgsGPSInformation& );
  18. void connDestroyed( QObject * );
  19. signals:
  20. void detected( QgsGPSConnection * );
  21. void detectionFailed();
  22. private:
  23. int mPortIndex;
  24. int mBaudIndex;
  25. QList< QPair< QString, QString > > mPortList;
  26. QList<BaudRateType> mBaudList;
  27. QgsGPSConnection *mConn;
  28. };

QgsQtLocationConnection类从QgsGPSConnection继承。

  1. class CORE_EXPORT QgsQtLocationConnection: public QgsGPSConnection
  2. {
  3.     Q_OBJECT
  4.   public:
  5. //调用startMonitor函数和strtGPS函数,开启一个0.5秒的timer,timer不停的调用broadcastConnectionAvailable,来广播最后一次的gps状态
  6.     QgsQtLocationConnection();
  7.     ~QgsQtLocationConnection();
  8.   protected slots:
  9.     /**Needed to make QtLocation detected*/
  10.     void broadcastConnectionAvailable( );
  11.     /**Parse available data source content*/
  12.     void parseData();
  13.     /**Called when the position updated.
  14.       * @note not available in python binding
  15.       */
  16.     void positionUpdated( const QGeoPositionInfo &info );
  17.     /**Called when the number of satellites in view is updated.
  18.       * @note not available in python bindings on android
  19.       */
  20.     void satellitesInViewUpdated( const QList<QGeoSatelliteInfo>& satellites );
  21.     /**Called when the number of satellites in use is updated.
  22.       * @note not available in python bindings on android
  23.       */
  24.     void satellitesInUseUpdated( const QList<QGeoSatelliteInfo>& satellites );
  25.   private:
  26.     void startGPS();
  27.     void startSatelliteMonitor();
  28.     QString mDevice;
  29.     QGeoPositionInfo mInfo;
  30.     QPointer<QGeoPositionInfoSource> locationDataSource;
  31.     QPointer<QGeoSatelliteInfoSource> satelliteInfoSource;
  32. };

gps模块总结:

里面分为两个部分前缀不带qgs和前缀带qgs的。不带qgs的为其他的开源软件中抠出来的代码,代码可能全部来自nmea的库。前缀带qgs的为qgs的代码。

最基础的类为QgsGPSConnection,它是:QgsQtLocationConnection、QgsNMEAConnection 的父类。QgsGPSConnectionRegistry用来注册QgsGPSConnection类。总之QgsGPSConnection是这个gps模块里的核心。

gps模块里还有两个比较基础的数据结构:QgsSatelliteInfo(gps卫星信息)和QgsGPSInformation(gps信息)它是上面QgsGPSConnection向外发送signal的载体。

gps模块完毕2015.1.23







声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/羊村懒王/article/detail/117580
推荐阅读
相关标签
  

闽ICP备14008679号