当前位置: 首页 > news >正文

C++(Qt)-GIS开发-简易瓦片地图下载器

Qt-GIS开发-简易瓦片地图下载器

文章目录

  • Qt-GIS开发-简易瓦片地图下载器
    • 1、概述
    • 2、安装openssl
    • 3、实现效果
    • 4、主要代码
      • 4.1 算法函数
      • 4.2 瓦片地图下载url拼接
      • 4.3 多线程下载
    • 5、源码地址
    • 6、参考

更多精彩内容
👉个人内容分类汇总 👈
👉GIS开发 👈

1、概述

  1. 支持单线程、多线程下载瓦片地图。
  2. 使用QNetworkAccessManager、QNetworkReply实现http、https下载功能;
  3. 支持下载多样式arcGis瓦片地图;
  4. 支持下载多样式高德瓦片地图;
  5. 支持多样式Bing地图下载;
  6. Qt中https下载功能需要安装openssl库。
  7. 本文中不会详细说瓦片地图的原理,写得好的文章太多了。

开发环境说明

  • 系统:Windows11、Ubuntu20.04
  • Qt版本:Qt 5.14.2
  • 编译器:MSVC2017-64、GCC/G++64

2、安装openssl

  • qt使用QNetworkReply/https下载瓦片地图需要ssl支持,qt默认是没有ssl库的;

  • 使用下列代码打印qt版本支持的ssl版本;

    qDebug() << "输出当前QT支持的openSSL版本: " << QSslSocket::sslLibraryBuildVersionString();
    qDebug() << "OpenSSL支持情况: " <<QSslSocket::supportsSsl();
    qDebug() << "OpenSSL运行时SSL库版本: " << QSslSocket::sslLibraryBuildVersionString();
    
  • windows可以下载对应版本的openssl,然后进行安装(轻量级就可以);

  • linux可以通过命令行安装,也可以下载源码自己编译。

  • openssl的github仓库

  • openssl官网

  • 安装后将openssl/bin文件夹下的libcrypto-1_1-x64.dll、libssl-1_1-x64.dll两个动态库拷贝到qt的编译器路径下,注意区分32和64位

    • D:\Qt\Qt5.14.2\5.14.2\msvc2017_64\bin
    • D:\Qt\Qt5.14.2\5.14.2\mingw73_64\bin

3、实现效果

  1. 无需注册、无需key进行瓦片地图下载;
  2. 地址可能会失效;
  3. 大量下载可能会限速;
  4. 仅作为学习使用。

在这里插入图片描述

4、主要代码

  • 项目文件结构
    在这里插入图片描述

4.1 算法函数

  • bingformula.h文件

    #ifndef BINGFORMULA_H
    #define BINGFORMULA_H
    #include <QPoint>
    #include <QtGlobal>namespace Bing {
    qreal clip(qreal n, qreal min, qreal max);
    qreal clipLon(qreal lon);   // 裁剪经度范围
    qreal clipLat(qreal lat);   // 裁剪纬度范围uint mapSize(int level);                        // 根据地图级别计算世界地图总宽高(以像素为单位)
    qreal groundResolution(qreal lat, int level);   // 计算地面分辨率
    qreal mapScale(qreal lat, int level, int screenDpi);   // 计算比例尺QPoint latLongToPixelXY(qreal lon, qreal lat, int level);               // 经纬度转像素 XY坐标
    void pixelXYToLatLong(QPoint pos, int level, qreal& lon, qreal& lat);   // 像素坐标转WGS-84墨卡托坐标QPoint pixelXYToTileXY(QPoint pos);    // 像素坐标转瓦片编号
    QPoint tileXYToPixelXY(QPoint tile);   // 瓦片编号转像素坐标QPoint latLongToTileXY(qreal lon, qreal lat, int level);   // 经纬度转瓦片编号
    QPointF tileXYToLatLong(QPoint tile, int level);           // 瓦片编号转经纬度QString tileXYToQuadKey(QPoint tile, int level);                             // 瓦片编号转QuadKey
    void quadKeyToTileXY(QString quadKey, int& tileX, int& tileY, int& level);   // QuadKey转瓦片编号、级别
    }   // namespace Bing
    #endif   // BINGFORMULA_H
  • bingformula.cpp文件

    /********************************************************************* 文件名: bingformula.cpp* 时间:   2024-04-05 21:36:16* 开发者:  mhf* 邮箱:   1603291350@qq.com* 说明:   适用于Bing瓦片地图的算法* ******************************************************************/
    #include "bingformula.h"
    #include <qstring.h>
    #include <QtMath>static const qreal g_EarthRadius = 6'378'137;   // 赤道半径/*** @brief      限定最小值,最大值范围* @param n    需要限定的值* @param min* @param max* @return*/
    qreal Bing::clip(qreal n, qreal min, qreal max)
    {n = qMax(n, min);n = qMin(n, max);return n;
    }/*** @brief      限定经度范围值,防止超限,经度范围[-180, 180]* @param lon  输入的经度* @return     裁剪后的经度*/
    qreal Bing::clipLon(qreal lon)
    {return clip(lon, -180.0, 180);
    }/*** @brief      限定纬度范围值,防止超限,经度范围[-85.05112878, 85.05112878]* @param lat  输入的纬度* @return     裁剪后的纬度*/
    qreal Bing::clipLat(qreal lat)
    {return clip(lat, -85.05112878, 85.05112878);
    }/*** @brief       根据输入的瓦片级别计算全地图总宽高,适用于墨卡托投影* @param level 1-23(bing地图没有0级别,最低级别为1,由4块瓦片组成)* @return      以像素为单位的地图宽度和高度。*/
    uint Bing::mapSize(int level)
    {uint w = 256;   // 第0级别为256*256return (w << level);
    }/*** @brief        计算指定纬度、级别的地面分辨率(不同纬度分辨率不同)* @param lat    纬度* @param level  地图级别 1-23(bing地图没有0级别,最低级别为1,由4块瓦片组成)* @return       地面分辨率 单位(米/像素)*/
    qreal Bing::groundResolution(qreal lat, int level)
    {lat = clipLat(lat);return qCos(lat * M_PI / 180) * 2 * M_PI * g_EarthRadius / mapSize(level);
    }/*** @brief           计算地图比例尺,地面分辨率和地图比例尺也随纬度而变化* @param lat       纬度* @param level     地图级别 1-23(bing地图没有0级别,最低级别为1,由4块瓦片组成)* @param screenDpi 屏幕分辨率,单位为点/英寸  通常为 96 dpi* @return          地图比例尺 1:N(地图上1厘米表示实际N厘米)*/
    qreal Bing::mapScale(qreal lat, int level, int screenDpi)
    {return groundResolution(lat, level) * screenDpi / 0.0254;   // 1英寸等于0.0254米
    }/*** @brief         将一个点从纬度/经度WGS-84墨卡托坐标(以度为单位)转换为指定细节级别的像素XY坐标。* @param lon     经度* @param lat     纬度* @param level   地图级别* @return        像素坐标*/
    QPoint Bing::latLongToPixelXY(qreal lon, qreal lat, int level)
    {lon = clipLon(lon);lat = clipLat(lat);qreal x = (lon + 180) / 360;qreal sinLat = qSin(lat * M_PI / 180);qreal y = 0.5 - qLn((1 + sinLat) / (1 - sinLat)) / (4 * M_PI);uint size = mapSize(level);qreal pixelX = x * size + 0.5;pixelX = clip(pixelX, 0, size - 1);qreal pixelY = y * size + 0.5;pixelY = clip(pixelY, 0, size - 1);return QPoint(pixelX, pixelY);
    }/*** @brief         将像素从指定细节级别的像素XY坐标转换为经纬度WGS-84坐标(以度为单位)* @param pos    像素坐标* @param level* @param lon* @param lat*/
    void Bing::pixelXYToLatLong(QPoint pos, int level, qreal& lon, qreal& lat)
    {uint size = mapSize(level);qreal x = (clip(pos.x(), 0, size - 1) / size) - 0.5;qreal y = 0.5 - (clip(pos.y(), 0, size - 1) / size);lon = x * 360;lat = 90 - (360 * qAtan(qExp(-y * 2 * M_PI)) / M_PI);
    }/*** @brief     像素坐标转瓦片编号* @param pos  像素坐标* @return    瓦片编号*/
    QPoint Bing::pixelXYToTileXY(QPoint pos)
    {int x = pos.x() / 256;int y = pos.y() / 256;return QPoint(x, y);
    }/*** @brief       瓦片编号转像素坐标* @param tile  瓦片编号* @return      像素坐标*/
    QPoint Bing::tileXYToPixelXY(QPoint tile)
    {int x = tile.x() * 256;int y = tile.y() * 256;return QPoint(x, y);
    }/*** @brief       经纬度转瓦片编号* @param lon* @param lat* @param level* @return*/
    QPoint Bing::latLongToTileXY(qreal lon, qreal lat, int level)
    {return pixelXYToTileXY(latLongToPixelXY(lon, lat, level));
    }/*** @brief         瓦片编号转经纬度* @param tile* @param level* @return       经纬度 x:经度  y纬度*/
    QPointF Bing::tileXYToLatLong(QPoint tile, int level)
    {qreal lon = 0;qreal lat = 0;QPoint pos = tileXYToPixelXY(tile);pixelXYToLatLong(pos, level, lon, lat);return QPointF(lon, lat);
    }/*** @brief         瓦片编号转 bing请求的QuadKey* @param tile   瓦片编号* @param level  瓦片级别* @return*/
    QString Bing::tileXYToQuadKey(QPoint tile, int level)
    {QString key;for (int i = level; i > 0; i--){char digit = '0';int mask = 1 << (i - 1);if ((tile.x() & mask) != 0){digit++;}if ((tile.y() & mask) != 0){digit += 2;}key.append(digit);}return key;
    }/*** @brief            将一个QuadKey转换为瓦片XY坐标。* @param quadKey* @param tileX      返回瓦片X编号* @param tileY      返回瓦片Y编号* @param level      返回瓦片等级*/
    void Bing::quadKeyToTileXY(QString quadKey, int& tileX, int& tileY, int& level)
    {tileX = 0;tileY = 0;level = quadKey.count();QByteArray buf = quadKey.toUtf8();for (int i = level; i > 0; i--){int mask = 1 << (i - 1);switch (buf.at(i - 1)){case '0':break;case '1':tileX |= mask;break;case '2':tileY |= mask;break;case '3':tileX |= mask;tileY |= mask;break;default:break;}}
    }

4.2 瓦片地图下载url拼接

  • mapinput.h

    #ifndef MAPINPUT_H
    #define MAPINPUT_H#include <QWidget>
    #include "mapStruct.h"namespace Ui {
    class MapInput;
    }class MapInput : public QWidget
    {Q_OBJECTpublic:explicit MapInput(QWidget *parent = nullptr);~MapInput();const QList<ImageInfo> &getInputInfo();       // 获取下载地图所需的输入信息private:// ArcGisvoid initArcGis();void getArcGisMapInfo();// 高德void initAMap();void getAMapInfo();// Bing地图void initBing();void getBingMapInfo();private:Ui::MapInput *ui;QList<ImageInfo> m_infos;                // 保存下载瓦片图片的信息
    };#endif // MAPINPUT_H
  • mapinput.cpp

    /********************************************************************* 文件名: mapinput.cpp* 时间:   2024-01-19 22:22:37* 开发者:  mhf* 邮箱:   1603291350@qq.com* 说明:   生成地图下载的输入信息* ******************************************************************/
    #include "mapinput.h"
    #include "bingformula.h"
    #include "formula.h"
    #include "ui_mapinput.h"
    #include <QDebug>MapInput::MapInput(QWidget* parent): QWidget(parent), ui(new Ui::MapInput)
    {ui->setupUi(this);initArcGis();initAMap();initBing();
    }MapInput::~MapInput()
    {delete ui;
    }/*** @brief 填入ArcGis下载地图类型*/
    void MapInput::initArcGis()
    {for (int i = 0; i < 23; i++){ui->com_z->addItem(QString("%1").arg(i), i);}ui->com_type->addItem("NatGeo_World_Map");ui->com_type->addItem("USA_Topo_Maps ");ui->com_type->addItem("World_Imagery");ui->com_type->addItem("World_Physical_Map");ui->com_type->addItem("World_Shaded_Relief");ui->com_type->addItem("World_Street_Map");ui->com_type->addItem("World_Terrain_Base");ui->com_type->addItem("World_Topo_Map");ui->com_type->addItem("Canvas/World_Dark_Gray_Base");ui->com_type->addItem("Canvas/World_Dark_Gray_Reference");ui->com_type->addItem("Canvas/World_Light_Gray_Base");ui->com_type->addItem("Canvas/World_Light_Gray_Reference");ui->com_type->addItem("Elevation/World_Hillshade_Dark");ui->com_type->addItem("Elevation/World_Hillshade");ui->com_type->addItem("Ocean/World_Ocean_Base");ui->com_type->addItem("Ocean/World_Ocean_Reference");ui->com_type->addItem("Polar/Antarctic_Imagery");ui->com_type->addItem("Polar/Arctic_Imagery");ui->com_type->addItem("Polar/Arctic_Ocean_Base");ui->com_type->addItem("Polar/Arctic_Ocean_Reference");ui->com_type->addItem("Reference/World_Boundaries_and_Places_Alternate ");ui->com_type->addItem("Reference/World_Boundaries_and_Places");ui->com_type->addItem("Reference/World_Reference_Overlay");ui->com_type->addItem("Reference/World_Transportation");ui->com_type->addItem("Specialty/World_Navigation_Charts");// 填入下载格式ui->com_format->addItem("jpg");ui->com_format->addItem("png");ui->com_format->addItem("bmp");
    }/*** @brief   计算并返回需要下载的瓦片地图信息* @return*/
    const QList<ImageInfo>& MapInput::getInputInfo()
    {m_infos.clear();   // 清除之前的内容switch (ui->tabWidget->currentIndex())   // 判断是什么类型的地图源{case 0:   // ArcGis{getArcGisMapInfo();   // 计算ArcGis下载信息break;}case 1:{getAMapInfo();   // 计算高德地图下载信息break;}case 2:{getBingMapInfo();   // 计算bing地图下载信息break;}default:break;}qDebug() << "瓦片数:" << m_infos.count();return m_infos;
    }/*** @brief   通过输入地图信息计算需要下载的瓦片图信息,下载ArcGIS地图,WGS84坐标系,Web墨卡托投影,z y x输入*/
    void MapInput::getArcGisMapInfo()
    {static QString url = "https://server.arcgisonline.com/arcgis/rest/services/%1/MapServer/tile/%2/%3/%4.%5";int z = ui->com_z->currentData().toInt();QString type = ui->com_type->currentText();QString format = ui->com_format->currentText();QStringList lt = ui->line_LTGps->text().trimmed().split(',');   // 左上角经纬度QStringList rd = ui->line_RDGps->text().trimmed().split(',');   // 右下角经纬度if (lt.count() != 2 || rd.count() != 2)return;                                    // 判断输入是否正确int ltX = lonTotile(lt.at(0).toDouble(), z);   // 计算左上角瓦片Xint ltY = latTotile(lt.at(1).toDouble(), z);   // 计算左上角瓦片Yint rdX = lonTotile(rd.at(0).toDouble(), z);   // 计算右下角瓦片Xint rdY = latTotile(rd.at(1).toDouble(), z);   // 计算右下角瓦片YImageInfo info;info.z = z;info.format = format;for (int x = ltX; x <= rdX; x++){info.x = x;for (int y = ltY; y <= rdY; y++){info.y = y;info.url = url.arg(type).arg(z).arg(y).arg(x).arg(format);m_infos.append(info);}}
    }/*** @brief 初始化高德地图下载选项信息*/
    void MapInput::initAMap()
    {for (int i = 1; i < 5; i++){ui->com_amapPrefix->addItem(QString("wprd0%1").arg(i));}for (int i = 1; i < 5; i++){ui->com_amapPrefix->addItem(QString("webst0%1").arg(i));}for (int i = 0; i < 19; i++){ui->com_amapZ->addItem(QString("%1").arg(i), i);}// 语言设置ui->com_amapLang->addItem("中文", "zh_cn");ui->com_amapLang->addItem("英文", "en");// 地图类型ui->com_amapStyle->addItem("卫星影像图", 6);ui->com_amapStyle->addItem("矢量路网", 7);ui->com_amapStyle->addItem("影像路网", 8);        // 支持png透明背景ui->com_amapStyle->addItem("卫星+影像路网", 9);   // 支持png透明背景// 图片尺寸,只在7 8生效ui->com_amapScl->addItem("256x256", 1);ui->com_amapScl->addItem("512x512", 2);// 填入下载格式ui->com_amapFormat->addItem("jpg");ui->com_amapFormat->addItem("png");ui->com_amapFormat->addItem("bmp");
    }/*** @brief 计算高德地图瓦片下载信息*/
    void MapInput::getAMapInfo()
    {static QString url = "https://%1.is.autonavi.com/appmaptile?";int z = ui->com_amapZ->currentData().toInt();QString format = ui->com_amapFormat->currentText();QStringList lt = ui->line_LTGps->text().trimmed().split(',');   // 左上角经纬度QStringList rd = ui->line_RDGps->text().trimmed().split(',');   // 右下角经纬度if (lt.count() != 2 || rd.count() != 2)return;                                    // 判断输入是否正确int ltX = lonTotile(lt.at(0).toDouble(), z);   // 计算左上角瓦片Xint ltY = latTotile(lt.at(1).toDouble(), z);   // 计算左上角瓦片Yint rdX = lonTotile(rd.at(0).toDouble(), z);   // 计算右下角瓦片Xint rdY = latTotile(rd.at(1).toDouble(), z);   // 计算右下角瓦片YImageInfo info;info.z = z;info.format = format;int style = ui->com_amapStyle->currentData().toInt();int count = 1;if (style == 9){count = 2;   // 如果是下载卫星图 + 路网图则循环两次}for (int i = 0; i < count; i++){if (count == 2){if (i == 0){style = 6;   // 第一次下载卫星图info.format = "jpg";}else{style = 8;             // 第二次下载路网图info.format = "png";   // 如果同时下载卫星图和路网图则路网图为透明png格式}}QString tempUrl = url.arg(ui->com_amapPrefix->currentText());                     // 设置域名tempUrl += QString("&style=%1").arg(style);                                       // 设置地图类型tempUrl += QString("&lang=%1").arg(ui->com_amapLang->currentData().toString());   // 设置语言tempUrl += QString("&scl=%1").arg(ui->com_amapScl->currentData().toInt());        // 设置图片尺寸,只在7 8生效tempUrl += QString("&ltype=%1").arg(ui->spin_amapLtype->value());                 // 设置图片中的信息,只有 7 8有效for (int x = ltX; x <= rdX; x++){info.x = x;for (int y = ltY; y <= rdY; y++){info.url = tempUrl + QString("&x=%1&y=%2&z=%3").arg(x).arg(y).arg(z);info.y = y;m_infos.append(info);}}}
    }/*** @brief 初始化Bing地图配置*/
    void MapInput::initBing()
    {// 服务器for (int i = 0; i < 8; i++){ui->com_bingPrefix->addItem(QString("%1").arg(i));}// 地图语言ui->com_bingLang->addItem("中文", "zh-cn");ui->com_bingLang->addItem("英语", "en-US");// 地图类型ui->com_bingType->addItem("卫星地图", "a");ui->com_bingType->addItem("普通地图", "r");ui->com_bingType->addItem("混合地图", "h");ui->com_bingCstl->addItem("默认", "w4c");ui->com_bingCstl->addItem("白天", "vb");    // 白天道路地图ui->com_bingCstl->addItem("夜晚", "vbd");   // 夜晚道路图// 瓦片等级for (int i = 1; i < 21; i++){ui->com_bingZ->addItem(QString("%1").arg(i));}// 填入下载格式ui->com_bingFormat->addItem("jpg");ui->com_bingFormat->addItem("png");ui->com_bingFormat->addItem("bmp");
    }/*** @brief 计算Bing地图的下载信息(这些url可能会失效,后续会使用其他方式下载)*  https://learn.microsoft.com/en-us/bingmaps/rest-services/directly-accessing-the-bing-maps-tiles*/
    void MapInput::getBingMapInfo()
    {//https://r1.tiles.ditu.live.com/tiles/r1321001.png?g=1001&mkt=zh-cn//http://dynamic.t2.tiles.ditu.live.com/comp/ch/r1321001.png?it=G,OS,L&mkt=en-us&cstl=w4c&ur=cn//http://ecn.t{0}.tiles.virtualearth.net/tiles/{1}{2}.png? g={4}//https://t0.dynamic.tiles.ditu.live.com/comp/ch/1320300313132?mkt=zh-CN&ur=CN&it=G,RL&n=z&og=894&cstl=vb//https://t1.dynamic.tiles.ditu.live.com/comp/ch/13203012200201?mkt=zh-CN&ur=cn&it=G,RL&n=z&og=894&cstl=vbd//https://dynamic.t1.tiles.ditu.live.com/comp/ch/1320300313313?it=Z,TF&L&n=z&key=AvquUWQgfy7VPqHn9ergJsp3Q_EiUft0ed70vZsX0_aqPABBdK07OkwrXWoGXsTG&ur=cn&cstl=vbd#define USE_URL 1
    #if (USE_URL == 0)// https://r1.tiles.ditu.live.com/tiles/r1321001.png?g=1001&mkt=zh-cnstatic QString url = "https://r%1.tiles.ditu.live.com/tiles/%2%3.%4?g=1001&mkt=%5";   // 街道图r支持中文
    #elif (USE_URL == 1)// http://dynamic.t2.tiles.ditu.live.com/comp/ch/r1321001.png?it=G,OS,L&mkt=en-us&cstl=w4c&ur=cnstatic QString url = "http://dynamic.t%1.tiles.ditu.live.com/comp/ch/%2%3.%4?it=G,OS,L&mkt=%5&cstl=%6&ur=cn";
    #endifint z = ui->com_bingZ->currentText().toInt();QStringList lt = ui->line_LTGps->text().trimmed().split(',');   // 左上角经纬度QStringList rd = ui->line_RDGps->text().trimmed().split(',');   // 右下角经纬度if (lt.count() != 2 || rd.count() != 2)return;                                    // 判断输入是否正确int ltX = lonTotile(lt.at(0).toDouble(), z);   // 计算左上角瓦片Xint ltY = latTotile(lt.at(1).toDouble(), z);   // 计算左上角瓦片Yint rdX = lonTotile(rd.at(0).toDouble(), z);   // 计算右下角瓦片Xint rdY = latTotile(rd.at(1).toDouble(), z);   // 计算右下角瓦片YQString format = ui->com_bingFormat->currentText();ImageInfo info;info.z = z;info.format = format;int prefix = ui->com_bingPrefix->currentIndex();QString lang = ui->com_bingLang->currentData().toString();   // 语言QString type = ui->com_bingType->currentData().toString();   // 类型QString cstl = ui->com_bingCstl->currentData().toString();   // 样式QPoint point;for (int x = ltX; x <= rdX; x++){info.x = x;point.setX(x);for (int y = ltY; y <= rdY; y++){info.y = y;point.setY(y);QString quadKey = Bing::tileXYToQuadKey(point, z);   // 将xy转为quadkey
    #if (USE_URL == 0)info.url = url.arg(prefix).arg(type).arg(quadKey).arg(format).arg(lang);
    #elif (USE_URL == 1)info.url = url.arg(prefix).arg(type).arg(quadKey).arg(format).arg(lang).arg(cstl);
    #endifm_infos.append(info);}}
    }

4.3 多线程下载

  • downloadthreads.h

    #ifndef DOWNLOADTHREADS_H
    #define DOWNLOADTHREADS_H#include "mapStruct.h"
    #include <QFutureWatcher>
    #include <QObject>class DownloadThreads : public QObject
    {Q_OBJECT
    public:explicit DownloadThreads(QObject* parent = nullptr);~DownloadThreads();// 传入需要下载的瓦片信息void getImage(QList<ImageInfo> infos);void quit();   // 退出下载signals:void finished(ImageInfo info);   // 返回下载后的瓦片,由于QImage为共享内存,所以传递不需要考虑太多性能private:QFuture<void> m_future;QList<ImageInfo> m_infos;
    };#endif   // DOWNLOADTHREADS_H
  • downloadthreads.cpp

    /********************************************************************* 文件名: downloadthreads.cpp* 时间:   2024-03-31 20:32:58* 开发者:  mhf* 邮箱:   1603291350@qq.com* 说明:   多线程下载瓦片地图* ******************************************************************/
    #include "downloadthreads.h"
    #include <QtConcurrent>
    #include <qnetworkaccessmanager.h>
    #include <qnetworkreply.h>static DownloadThreads* g_this = nullptr;
    DownloadThreads::DownloadThreads(QObject *parent) : QObject(parent)
    {g_this = this;  // 记录当前 this指针,用于传递信号
    }DownloadThreads::~DownloadThreads()
    {g_this = nullptr;quit();
    }/*** @brief       下载瓦片* @param info* @return*/
    void getUrl(ImageInfo info)
    {QNetworkAccessManager manager;QNetworkReply* reply = manager.get(QNetworkRequest(QUrl(info.url)));// 等待返回QEventLoop loop;QObject::connect(reply, &QNetworkReply::finished, &loop, &QEventLoop::quit);loop.exec();if(reply->error() == QNetworkReply::NoError){QByteArray buf = reply->readAll();info.img.loadFromData(buf);}else{info.count++;if(info.count < 3){getUrl(info);   // 下载失败重新下载return;}else{qWarning() << "下载失败:" << reply->errorString();}}if(g_this){emit g_this->finished(info);  // 通过信号将下载后的瓦片传出去}
    }/*** @brief         调用线程池下载瓦片* @param infos*/
    void DownloadThreads::getImage(QList<ImageInfo> infos)
    {m_infos = infos;    // 这里不能使用infos,因为会在函数退出释放
    #if 0   // 由于map使用的是全局线程池,所以可以查看、设置线程数qDebug() <<QThreadPool::globalInstance()->maxThreadCount();   // 查看最大线程数QThreadPool::globalInstance()->setMaxThreadCount(1);          // 设置最大线程数
    #endifm_future = QtConcurrent::map(m_infos, getUrl);
    }/*** @brief 退出下载*/
    void DownloadThreads::quit()
    {if(m_future.isRunning())   // 判断是否在运行{m_future.cancel();               // 取消下载m_future.waitForFinished();      // 等待退出}
    }

5、源码地址

  • github
  • gitee

6、参考

  • GIS开发一:OpenLayers在线瓦片数据源汇总_在线瓦片图数据-CSDN博客
  • Bing Maps Tile System - Bing Maps | Microsoft Learn

相关文章:

C++(Qt)-GIS开发-简易瓦片地图下载器

Qt-GIS开发-简易瓦片地图下载器 文章目录 Qt-GIS开发-简易瓦片地图下载器1、概述2、安装openssl3、实现效果4、主要代码4.1 算法函数4.2 瓦片地图下载url拼接4.3 多线程下载 5、源码地址6、参考 更多精彩内容&#x1f449;个人内容分类汇总 &#x1f448;&#x1f449;GIS开发 …...

誉天教育7月开班计划:为梦想插上腾飞的翅膀!

随着夏日的脚步渐近&#xff0c;誉天教育也迎来了新一轮的学习热潮。在这个充满活力和希望的季节里&#xff0c;我们精心策划了7月的开班计划&#xff0c;旨在为广大学子提供一个优质、高效的学习平台&#xff0c;助力他们追逐梦想&#xff0c;实现自我价值。 本月 Linux云计算…...

STM32基础篇:GPIO

GPIO简介 GPIO&#xff1a;即General Purpose Input/Output&#xff0c;通用目的输入/输出。就是一种片上外设&#xff08;内部模块&#xff09;。 对于STM32的芯片来说&#xff0c;周围有一圈引脚&#xff0c;有时需要对引脚进行读写&#xff08;读&#xff1a;从外部输入一…...

HTTPS 发送请求出现TLS握手失败

最近在工作中&#xff0c;调外部接口&#xff0c;发现在clientHello步骤报错&#xff0c;服务端没有返回serverHello。 从网上找了写方法&#xff0c;都没有解决&#xff1b; 在idea的vm options加上参数&#xff1a; -Djavax.net.debugSSL,handshake 把SSL和handshake的日…...

数字化精益生产系统--IFS财务管理系统

IFS财务管理系统是一款功能丰富、高效且灵活的企业财务管理软件&#xff0c;广泛应用于多个行业和不同规模的企业中。以下是对IFS财务管理系统的功能设计&#xff1a;...

基于SpringBoot的校园台球厅人员与设备管理系统

本系统是要设计一个校园台球厅人员与设备管理系统&#xff0c;这个系统能够满足校园台球厅人员与设备的管理及用户的校园台球厅人员与设备管理功能。系统的主要功能包括首页、个人中心、用户管理、会员账号管理、会员充值管理、球桌信息管理、会员预约管理、普通预约管理、留言…...

免杀笔记 ---> Session0--DLL注入

刚更新完上一篇&#xff0c;于是我们就马不停蹄的去跟新下一篇&#xff01;&#xff01; Session0注入 &#xff1a;&#xff1a; 各位看官如果觉得还不错的可以给博主点个赞&#x1f495;&#x1f495; 这次&#xff0c;我把这个脚本直接传到Github上了 喜欢的师傅点个Star噢…...

如何做好IT类的技术面试?

我们在找工作时&#xff0c;需要结合自己的现状&#xff0c;针对意向企业做好充分准备。作为程序员&#xff0c;你有哪些面试IT技术岗的技巧&#xff1f; 方向一&#xff1a;分享你面试IT公司的小技巧 我分享一些基于广泛观察和用户反馈的面试IT公司的小技巧&#xff1a; 技术准…...

A7 配置方式Master SPI如何更改位宽

在 FPGA 完成自初始化后&#xff0c;INIT 释放&#xff0c;FPGA 对模式引脚 (M[2:0]) 进行采样&#xff0c;以确定使用哪种配置模式。当模式引脚 M[2:0] 001 时&#xff0c;FPGA 开始以大约 3 MHz 的频率在 CCLK 上输出时钟。随后&#xff0c;FCS_B 驱动为低电平&#xff0c;紧…...

linux kthread任务管理

目录 一、linux 创建内核线程1.1 kthread_create1.2 kthread_create_worker kthread_queue_work 二、设置线程优先级和调度策略2.1 sched_setscheduler2.2 调度策略 一、linux 创建内核线程 1.1 kthread_create 在 linux 中&#xff0c;可以使用 kthread_create 接口创建内核…...

第一节 网络安全概述

一.网络空间安全 网络空间&#xff1a;一个由信息基础设施组成相互依赖的网络。 ---- 海陆空天&#xff08;大海、陆 地、天空、航天&#xff09; 通信保密阶段 ---- 计算机安全 ----- 信息系统安全 ----- 网络空间安全 计算机安全&#xff1a;开始秉持着“严于律己&#x…...

星光云VR全景系统源码

星光云VR全景系统源码 体验地址请查看...

spdlog一个非常好用的C++日志库(七): 源码分析之异常类spdlog_ex

目录 1.自定义异常类spdlog_ex 1.1.通用异常 1.2.系统调用异常 1.3.what()函数 2.异常的使用 2.1.抛出异常 2.2.控制异常使用 1.自定义异常类spdlog_ex 标准库异常类&#xff08;std::exception&#xff09;系列&#xff0c;能满足大多数使用异常的场景&#xff0c;但对…...

从一次 SQL 查询的全过程了解 DolphinDB 线程模型

1. 前言 DolphinDB 的线程模型较为复杂&#xff0c;写入与查询分布式表都可能需要多个类型的线程。通过了解 SQL 查询的全过程&#xff0c;可以帮助我们了解 DolphinDB 的线程模型&#xff0c;掌握 DolpinDB 的配置&#xff0c;以及优化系统性能的方法。 本教程以一个分布式 …...

Vue3.js“非原始值”响应式实现基本原理笔记(二)

如果您觉得这篇文章有帮助的话&#xff01;给个点赞和评论支持下吧&#xff0c;感谢~ 作者&#xff1a;前端小王hs 阿里云社区博客专家/清华大学出版社签约作者/csdn百万访问前端博主/B站千粉前端up主 此篇文章是博主于2022年学习《Vue.js设计与实现》时的笔记整理而来 书籍&a…...

论文 | PRCA: 通过可插拔奖励驱动的上下文适配器拟合用于检索问答的黑盒大语言模型

论文全称&#xff1a;PRCA: Fitting Black-Box Large Language Models for Retrieval Question Answering via Pluggable Reward-Driven Contextual Adapter 核心问题&#xff1a;如何在检索增强式问答&#xff08;ReQA&#xff09;任务中&#xff0c;利用大型语言模型&#xf…...

网络状态的智能感知:WebKit 支持 Network Information API 深度解析

网络状态的智能感知&#xff1a;WebKit 支持 Network Information API 深度解析 在现代 Web 应用中&#xff0c;理解用户的网络连接状态对于提供适应性体验至关重要。Network Information API&#xff0c;一个新兴的 Web API&#xff0c;允许 Web 应用访问设备的网络信息&…...

Vue3基础知识:组合式API中的provide和inject,他们作用是什么?如何使用?以及案例演示

1.provide和inject相较于父子传递的不同在于provide,inject可以用于跨层级通信&#xff08;通俗易懂的讲就是可以实现爷孙之间的直接信息传递&#xff09;。 1.跨层级传递数据 1.在顶层组件通过provide函数提供数据 2.底层组件通过inject函数获取数据 演示一&#xff1a;跨…...

Transformer自注意力机制(Self-Attention)模型

​ 上一篇我们介绍了transform专题一&#xff1a;Seq2seq model&#xff0c;也知道了transfrom属于seq2seq模型&#xff0c;这一排篇咱们接着介绍另外几种seq2seq架构的模型。&#xff09;RNN&#xff08;循环神经网络&#xff09;CNN&#xff08;卷积神经网络&#xff09;&…...

【计算机体系结构】缓存的false sharing

在介绍缓存的false sharing之前&#xff0c;本文先介绍一下多核系统中缓存一致性是如何维护的。 目前主流的多核系统中的缓存一致性协议是MESI协议及其衍生协议。 MESI协议 MESI协议的4种状态 MESI协议有4种状态。MESI是4种状态的首字母缩写&#xff0c;缓存行的4种状态分别…...

Ubuntu24.04 Isaacgym的安装

官方论坛 rl-接口 教程1 教程2 教程3 1.下载压缩包 link 2. 解压 tar -xvf IsaacGym_Preview_4_Package.tar.gz核心教程在 isaacgym/docs/install.html下 3. 从源码安装 Ubuntu24.04还需首先进入虚拟环境 python -m venv myenv # 创建虚拟环境&#xff0c;已有可跳过…...

docker 设置代理,通过代理服务器拉取镜像

docker 拉取目标镜像需要通过代理服务器进行时&#xff0c;可以通过为 docker 配置全局代理来实现。 注&#xff1a;Linux 上通过临时命令 export HTTP_PROXY 设置的代理&#xff0c;对 curl 这些有用&#xff0c;但是对 docker pull 不起作用。 示例 假设您的代理服务器地址是…...

OpenCV教程02:图像处理系统1.0(翻转+形态学+滤波+缩放+旋转)

-------------OpenCV教程集合------------- Python教程99&#xff1a;一起来初识OpenCV&#xff08;一个跨平台的计算机视觉库&#xff09; OpenCV教程01&#xff1a;图像的操作&#xff08;读取显示保存属性获取和修改像素值&#xff09; OpenCV教程02&#xff1a;图像处理…...

人工智能在招投标领域的运用---监控视频连续性检测

作者&#xff1a;舒城县公共交易中心 zhu_min726126.com 原创&#xff0c;转载请注明出处。 摘要 随着人工智能&#xff08;AI&#xff09;技术的飞速发展&#xff0c;其在各个领域的应用日益广泛。本文旨在探讨人工智能在招投标领域的运营&#xff0c;重点介绍AI对视频完整…...

加装德国进口高精度主轴 智能手机壳「高质量高效率」钻孔铣槽

在当前高度智能化的社会背景下&#xff0c;智能手机早已成为人们生活、工作的必备品&#xff0c;智能手机壳作市场需求量巨大。智能手机壳的加工过程涉及多个环节&#xff0c;包括钻孔和铣槽等。钻孔要求精度高、孔位准确&#xff0c;而铣槽则需要保证槽位规整、深度适宜。这些…...

Java Stream API 常用操作技巧

Java 8 引入的 Stream API 为集合操作提供了一种声明式编程模型&#xff0c;极大地简化了数据处理的复杂性。本文将介绍 Java Stream API 的几种常用操作方式&#xff0c;帮助开发者更高效地处理集合数据。 1. 过滤&#xff08;Filtering&#xff09; 过滤是选择集合中满足特…...

SwiftData 模型对象的多个实例在 SwiftUI 中不能及时同步的解决

概览 我们已经知道,用 CoreData 在背后默默支持的 SwiftUI 视图在使用 @FetchRequest 来查询托管对象集合时,若查询结果中的托管对象在别处被改变将不会在 FetchedResults 中得到及时的刷新。 那么这一“囧境”在 SwiftData 里是否也会“卷土重来”呢?空说无益,就让我们在…...

Android 系统网络、时间服务器配置修改

1.修改wifi 是否可用的检测地址&#xff1a; 由于编译的源码用的是谷歌的检测url,国内访问不了&#xff0c;系统会认为wifi网络受限&#xff0c;所以改成国内的地址 adb shell settings delete global captive_portal_https_urladb shell settings delete global captive_por…...

类和对象深入理解

目录 static成员概念静态成员变量面试题补充代码1代码2代码3如何访问private中的成员变量 静态成员函数静态成员函数没有this指针 特性 友元友元函数友元类 内部类特性1特性2 匿名对象拷贝对象时的一些编译器优化 感谢各位大佬对我的支持,如果我的文章对你有用,欢迎点击以下链接…...

在postgres数据库中的几个简单用法

1、例如表中coord_str的字段数据是121.12334 31.3435这样的字符串&#xff0c;如何将对应的数据转换成geometry数据&#xff0c;实现如下 UPDATE coordinates SET geom ST_GeomFromText(POINT( || split_part(coord_str, , 1) || || split_part(coord_str, , 2) || ), 43…...

SQLServer Manager Studio扩展开发从入门到弃坑

Visualstudio的已经开发好了&#xff0c;可这个就是不行&#xff0c;直接运行点这些按钮加载失败&#xff0c;而我直接不调试模式&#xff0c;则直接什么都没有&#xff0c;调试 发现是根本没触发逻辑的。 文档资料太少&#xff0c; 我换了几个ssms.exe都不行&#xff0c;18-20…...

ComfyUI预处理器ControlNet简单介绍与使用(附件工作流)

简介 ControlNet 是一个很强的插件&#xff0c;提供了很多种图片的控制方式&#xff0c;有的可以控制画面的结构&#xff0c;有的可以控制人物的姿势&#xff0c;还有的可以控制图片的画风&#xff0c;这对于提高AI绘画的质量特别有用。接下来就演示几种热门常用的控制方式 1…...

【篇三】在vue3上实现阿里云oss文件直传

之前写了两篇关于文件上传的文章 【篇一】使用springbootvue实现阿里云oss上传 【篇二】使用springbootvue实现阿里云oss文件直传&#xff0c;解决大文件分片上传问题 今天介绍一下在vue3中实现阿里云oss文件直传&#xff0c;主要是基于篇二中的源码进行修改&#xff0c;看具体…...

OceanBase v4.2 特性解析:对Json与Xml的扩展支持

1. 背景 OceanBase的Oracle模式当前已实现对XMLType类型的支持&#xff0c;不仅包含了基本的构造、查询、更新以及格式转换功能&#xff0c;还支持使用Xpath查询从XML数据中提取特定值。在V 4.2.2 版本中&#xff0c;我们进一步扩展了Oracle模式下对XMLType的支持&#xff0c;…...

《框架封装 · 统一异常处理和返回值包装》

&#x1f4e2; 大家好&#xff0c;我是 【战神刘玉栋】&#xff0c;有10多年的研发经验&#xff0c;致力于前后端技术栈的知识沉淀和传播。 &#x1f497; &#x1f33b; CSDN入驻不久&#xff0c;希望大家多多支持&#xff0c;后续会继续提升文章质量&#xff0c;绝不滥竽充数…...

深入WebKit:揭秘复杂文档的高效渲染之道

深入WebKit&#xff1a;揭秘复杂文档的高效渲染之道 在当今信息爆炸的时代&#xff0c;网页不再仅仅是简单的文本和图片的集合&#xff0c;而是充满了复杂布局和丰富媒体内容的交互式平台。WebKit 作为众多流行浏览器的心脏&#xff0c;其布局引擎承担着将 HTML、CSS 代码转换…...

进程的控制-孤儿进程和僵尸进程

孤儿进程 &#xff1a; 一个父进程退出&#xff0c;而它的一个或多个子进程还在运行&#xff0c;那么那些子进程将成为孤儿进程。孤儿进程将被 init 进程( 进程号为 1) 所收养&#xff0c;并由 init 进程对它们完成状态收集工作 为了释放子进程的占用的系统资源&#xff1a; …...

【Unity navigation面板】

【Unity navigation面板】 Unity的Navigation面板是一个集成在Unity编辑器中的界面&#xff0c;它允许开发者对导航网格&#xff08;NavMesh&#xff09;进行配置和管理。 Unity Navigation面板的一些关键特性和功能&#xff1a; 导航网格代理&#xff08;NavMesh Agent&…...

二刷算法训练营Day53 | 动态规划(14/17)

目录 详细布置&#xff1a; 1. 392. 判断子序列 2. 115. 不同的子序列 详细布置&#xff1a; 1. 392. 判断子序列 给定字符串 s 和 t &#xff0c;判断 s 是否为 t 的子序列。 字符串的一个子序列是原始字符串删除一些&#xff08;也可以不删除&#xff09;字符而不改变剩余…...

将缓冲文件写到磁盘中的命令sync

将缓冲文件写到磁盘中的命令sync There is no nutrition in the blog content. After reading it, you will not only suffer from malnutrition, but also impotence. The blog content is all parallel goods. Those who are worried about being cheated should leave quick…...

灵活视图变换器:为扩散模型设计的革新图像生成架构

在自然界中&#xff0c;图像的分辨率是无限的&#xff0c;而现有的图像生成模型在跨任意分辨率泛化方面存在困难。虽然扩散变换器&#xff08;DiT&#xff09;在特定分辨率范围内表现出色&#xff0c;但在处理不同分辨率的图像时却力不从心。为了克服这一限制&#xff0c;来自上…...

[终端安全]-1 总体介绍

有朋友一直在和笔者研讨智驾安全这个热门话题&#xff0c;笔者十多年工作从不离终端安全这个核心话题&#xff08;芯片安全、操作系统安全、应用安全&#xff09;&#xff0c;近来也一直在梳理终端安全体系&#xff1b;手机、汽车皆是我们生活中应用最普遍的智能终端&#xff0…...

Mysql5.7并发插入死锁问题

死锁的产生条件 互斥、请求和保持、不可剥夺、循环等待 MySQL锁类型 死锁复现 环境&#xff1a;Mysql 5.7版本&#xff0c;Innodb引擎&#xff0c;可重复度隔离级别 并发场景下使用duplicate key update插入或更新数据可能会造成死锁&#xff0c;下面就产生死锁的条件进行模…...

网络“ping不通”,如何排查和解决呢?

网络问题往往复杂且难以预测&#xff0c;其中“ping不通”是常见的网络故障之一。 1. 确认问题现象 首先&#xff0c;明确问题是完全无法ping通(无响应)还是ping通但有高延迟或丢包。这有助于缩小问题范围。 2. 本地检查 网络接口状态&#xff1a;使用ifconfig(Linux)或ipc…...

日常学习--20240706

1、udp协议的特点有哪些&#xff1f; a、无连接&#xff0c;发送和接收数据不需要建立连接&#xff0c;开销小&#xff0c;实时性好 b、不可靠传输&#xff0c;不保证数据包能够到达目的地&#xff0c;也不保证数据包的顺序 c、面向数据报的&#xff0c;以数据报形式发送数据…...

入门PHP就来我这(高级)12 ~ 获取数据

有胆量你就来跟着路老师卷起来&#xff01; -- 纯干货&#xff0c;技术知识分享 路老师给大家分享PHP语言的知识了&#xff0c;旨在想让大家入门PHP&#xff0c;并深入了解PHP语言。 1 从结果集中获取一行作为对象 表中数据行如下&#xff1a; 利用mysqli_fetch_array()函数获…...

AIGC专栏12——EasyAnimateV3发布详解 支持图文生视频 最大支持960x960x144帧视频生成

AIGC专栏12——EasyAnimateV3发布详解 支持图&文生视频 最大支持960x960x144帧视频生成 学习前言项目特点生成效果相关地址汇总项目主页Huggingface体验地址Modelscope体验地址源码下载地址 EasyAnimate V3详解技术储备Diffusion Transformer (DiT)Hybrid Motion ModuleU-V…...

【python】python猫眼电影数据抓取分析可视化(源码+数据集+论文)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;公众号&#x1f448;&#xff1a;测试开发自动化【获取源码商业合作】 &#x1f449;荣__誉&#x1f448;&#xff1a;阿里云博客专家博主、5…...

Android 四大组件

1. Activity 应用程序中&#xff0c;一个Activity通常是一个单独的屏幕&#xff0c;它上面可以显示一些控件&#xff0c;也可以监听并对用户的事件做出响应。 Activity之间通过Intent进行通信&#xff0c;在Intent 的描述结构中&#xff0c;有两个最重要的部分&#xff1a;动…...

【Python】已解决:ModuleNotFoundError: No module named ‘nltk’

文章目录 一、分析问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项 已解决&#xff1a;ModuleNotFoundError: No module named ‘nltk’ 一、分析问题背景 在使用Python进行自然语言处理或文本分析时&#xff0c;我们经常会用到各种库来辅助我们的工…...