离线地图下载方法解析

jerry 地图开发 2015年11月26日 收藏

在使用Google Map, Microsoft Map 等在线地图时,常常想知道这些地图在服务器上是如何存放的,目前大部分地图服务器都是将地图以图片存储以提高响应速度。 一般大小为256X256个像素。
如果知道每个图片的URL,自己写个地图下载软件并不困难。从而做成离线地图。
下面程序片段给出了常用地图服务器上地图图片存取的URL计算方法。

  1. using System.Collections.Generic;
  2. using System.Text;
  3. namespace MapTileURL
  4. {
  5.     class Program
  6.     {
  7.         private static Dictionary<string, string> MapTypeURLs = new Dictionary<string, string>
  8.         {
  9.             { "GoogleSatURL", "<a href="http://khm{GOOG_DIGIT}.google.com/kh/v=58&x={X}&y={Y}&z={ZOOM}&s={GALILEO">http://khm{GOOG_DIGIT}.google.com/kh/v=58&x={X}&y={Y}&z={ZOOM}&s={GALILEO</a>}" },
  10.             { "GoogleMapURL", "<a href="http://mt{GOOG_DIGIT}.google.com/vt/lyrs=m@121&hl=en&x={X}&y={Y}&z={ZOOM}&s={GALILEO">http://mt{GOOG_DIGIT}.google.com/vt/lyrs=m@121&hl=en&x={X}&y={Y}&z={ZOOM}&s={GALILEO</a>}" },
  11.             { "GoogleHybURL", "<a href="http://mt{GOOG_DIGIT}.google.com/vt/lyrs=h@121&hl=en&x={X}&y={Y}&z={ZOOM}&s={GALILEO">http://mt{GOOG_DIGIT}.google.com/vt/lyrs=h@121&hl=en&x={X}&y={Y}&z={ZOOM}&s={GALILEO</a>}" },
  12.             { "GoogleTerURL", "<a href="http://mt{GOOG_DIGIT}.google.com/vt/lyrs=t@108,r@121&hl=en&x={X}&y={Y}&z={ZOOM}&s={GALILEO">http://mt{GOOG_DIGIT}.google.com/vt/lyrs=t@108,r@121&hl=en&x={X}&y={Y}&z={ZOOM}&s={GALILEO</a>}" },
  13.             { "GoogleChinaURL", "<a href="http://mt{GOOG_DIGIT}.google.cn/vt/lyrs=m@121&hl=en&gl=cn&x={X}&y={Y}&z={ZOOM}&s={GALILEO">http://mt{GOOG_DIGIT}.google.cn/vt/lyrs=m@121&hl=en&gl=cn&x={X}&y={Y}&z={ZOOM}&s={GALILEO</a>}" },
  14.             { "YahooMapURL", "<a href="http://maps{Y_DIGIT}.yimg.com/hx/tl?v=4.3&.intl=en&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM}&r=1">http://maps{Y_DIGIT}.yimg.com/hx/tl?v=4.3&.intl=en&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM}&r=1</a>" },
  15.             { "YahooSatURL", "<a href="http://maps{Y_DIGIT}.yimg.com/ae/ximg?v=1.9&t=a&s=256&.intl=en&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM}&r=1">http://maps{Y_DIGIT}.yimg.com/ae/ximg?v=1.9&t=a&s=256&.intl=en&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM}&r=1</a>" },
  16.             { "YahooInMapURL", "<a href="http://maps.yimg.com/hw/tile?locale=en&imgtype=png&yimgv=1.2&v=4.1&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM_2">http://maps.yimg.com/hw/tile?locale=en&imgtype=png&yimgv=1.2&v=4.1&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM_2</a>}" },
  17.             { "YahooInHybURL", "<a href="http://maps.yimg.com/hw/tile?imgtype=png&yimgv=0.95&t=h&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM_2">http://maps.yimg.com/hw/tile?imgtype=png&yimgv=0.95&t=h&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM_2</a>}" },
  18.             { "YahooHybURL", "<a href="http://maps{Y_DIGIT}.yimg.com/hx/tl?v=4.3&t=h&.intl=en&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM}&r=1">http://maps{Y_DIGIT}.yimg.com/hx/tl?v=4.3&t=h&.intl=en&x={X}&y={YAHOO_Y}&z={YAHOO_ZOOM}&r=1</a>" },
  19.             { "MicrosoftBrMapURL", "<a href="http://imakm{MS_DIGITBR}.maplink3.com.br/maps.ashx?v={QUAD}|t&call=2.2.4">http://imakm{MS_DIGITBR}.maplink3.com.br/maps.ashx?v={QUAD}|t&call=2.2.4</a>" },
  20.             { "MicrosoftHybURL", "<a href="http://ecn.t{MS_DIGIT}.tiles.virtualearth.net/tiles/h{QUAD}.png?g=441&mkt=en-us&n=z">http://ecn.t{MS_DIGIT}.tiles.virtualearth.net/tiles/h{QUAD}.png?g=441&mkt=en-us&n=z</a>" },
  21.             { "MicrosoftSatURL", "<a href="http://ecn.t{MS_DIGIT}.tiles.virtualearth.net/tiles/a{QUAD}.png?g=441&mkt=en-us&n=z">http://ecn.t{MS_DIGIT}.tiles.virtualearth.net/tiles/a{QUAD}.png?g=441&mkt=en-us&n=z</a>" },
  22.             { "MicrosoftMapURL", "<a href="http://ecn.t{MS_DIGIT}.tiles.virtualearth.net/tiles/r{QUAD}.png?g=441&mkt=en-us&n=z">http://ecn.t{MS_DIGIT}.tiles.virtualearth.net/tiles/r{QUAD}.png?g=441&mkt=en-us&n=z</a>" },
  23.             { "MicrosoftTerURL", "<a href="http://ecn.t{MS_DIGIT}.tiles.virtualearth.net/tiles/r{QUAD}.png?g=441&mkt=en-us&shading=hill&n=z">http://ecn.t{MS_DIGIT}.tiles.virtualearth.net/tiles/r{QUAD}.png?g=441&mkt=en-us&shading=hill&n=z</a>" },
  24.             { "OpenStreetMapURL", "<a href="http://tile.openstreetmap.org/{ZOOM}/{X}/{Y}.png">http://tile.openstreetmap.org/{ZOOM}/{X}/{Y}.png</a>" },
  25.             { "OSMARenderURL", "<a href="http://tah.openstreetmap.org/Tiles/tile/{ZOOM}/{X}/{Y}.png">http://tah.openstreetmap.org/Tiles/tile/{ZOOM}/{X}/{Y}.png</a>" },
  26.             { "OpenAerialMapURL", "<a href="http://tile.openaerialmap.org/tiles/?v=mgm&layer=openaerialmap-900913&x={X}&y={Y}&zoom={OAM_ZOOM">http://tile.openaerialmap.org/tiles/?v=mgm&layer=openaerialmap-900913&x={X}&y={Y}&zoom={OAM_ZOOM</a>}" },
  27.             { "OpenCycleMapURL", "<a href="http://andy.sandbox.cloudmade.com/tiles/cycle/{ZOOM}/{X}/{Y}.png">http://andy.sandbox.cloudmade.com/tiles/cycle/{ZOOM}/{X}/{Y}.png</a>" }
  28.         };
  29.         readonly static char[] Lookup = new[] { 'q', 't', 'r', 's' };
  30.         public static string GetQuadtreeFromXYZoom(int x, int y, int zoom)
  31.         {
  32.             var quad = new StringBuilder();
  33.             for (int i = 0; i < zoom; i++)
  34.             {
  35.                 int rx = x % 2;
  36.                 int ry = y % 2;
  37.                 x /= 2;
  38.                 y /= 2;
  39.                 quad.Insert(0, Lookup[rx * 2 + ry]);
  40.             }
  41.             quad.Insert(0, 't');
  42.             return quad.ToString();
  43.         }
  44.         public static string BuildUrl(string tileType, int x,int y,int zoom)
  45.         {
  46.     
  47.             string url = "";
  48.             // Edited by Shustrik - compacted the switch statement
  49.             switch (tileType)
  50.             {
  51.                 case "GoogleSat":
  52.                 case "GoogleSatH":
  53.                     url = MapTypeURLs["GoogleSatURL"];
  54.                     break;
  55.                 case "GoogleMap":
  56.                     url = MapTypeURLs["GoogleMapURL"];
  57.                     break;
  58.                 case "GoogleHyb":
  59.                     url = MapTypeURLs["GoogleHybURL"];
  60.                     break;
  61.                 case "GoogleTer":
  62.                     url = MapTypeURLs["GoogleTerURL"];
  63.                     break;
  64.                 case "GoogleChina":
  65.                     url = MapTypeURLs["GoogleChinaURL"];
  66.                     break;
  67.                 case "MicrosoftMap":
  68.                     url = MapTypeURLs["MicrosoftMapURL"];
  69.                     break;
  70.                 case "MicrosoftSat":
  71.                     url = MapTypeURLs["MicrosoftSatURL"];
  72.                     break;
  73.                 case "MicrosoftHyb":
  74.                     url = MapTypeURLs["MicrosoftHybURL"];
  75.                     break;
  76.                 case "MicrosoftTer":
  77.                     url = MapTypeURLs["MicrosoftTerURL"];
  78.                     break;
  79.                 case "MicrosoftBrMap":
  80.                     if (zoom <= 10)
  81.                         url =  MapTypeURLs["MicrosoftMapURL"];
  82.                     else
  83.                         url = MapTypeURLs["MicrosoftBrMapURL"];
  84.                     break;
  85.                 case "YahooMap":
  86.                     url = MapTypeURLs["YahooMapURL"];
  87.                     break;
  88.                 case "YahooSat":
  89.                 case "YahooSatH":
  90.                 case "YahooSatH2":
  91.                     url = MapTypeURLs["YahooSatURL"];
  92.                     break;
  93.                 case "YahooHyb":
  94.                     url = MapTypeURLs["YahooHybURL"];
  95.                     break;
  96.                 case "YahooInMap":
  97.                     url = MapTypeURLs["YahooInMapURL"];
  98.                     break;
  99.                 case "YahooInHyb":
  100.                     url = MapTypeURLs["YahooInHybURL"];
  101.                     break;
  102.                 case "OpenStreetMap":
  103.                     url = MapTypeURLs["OpenStreetMapURL"];
  104.                     break;
  105.                 case "OSMARender":
  106.                     url = MapTypeURLs["OSMARenderURL"];
  107.                     break;
  108.                 case "OpenAerialMap":
  109.                     url = MapTypeURLs["OpenAerialMapURL"];
  110.                     break;
  111.                 case "OpenCycleMap":
  112.                     url = MapTypeURLs["OpenCycleMapURL"];
  113.                     break;
  114.             }
  115.           
  116.             // Edited by Shustrik - added variables for configuration settings other than Google
  117.             string quadcode = "";
  118.             for (int i = (int)zoom - 1; i >= 0; i--)
  119.                 quadcode = quadcode + (((((y >> i) & 1) << 1) + ((x >> i) & 1)));
  120.             url = url.Replace("{X}", x.ToString());
  121.             url = url.Replace("{Y}", y.ToString());
  122.             url = url.Replace("{Z}", ((int)zoom).ToString());
  123.             url = url.Replace("{ZOOM}", ((int)zoom).ToString());
  124.             url = url.Replace("{QUAD}", quadcode);
  125.             url = url.Replace("{YAHOO_Y}", (((1 << ((int)zoom)) >> 1) - 1 - y).ToString());
  126.             url = url.Replace("{YAHOO_ZOOM}", ((int)zoom + 1).ToString());
  127.             url = url.Replace("{YAHOO_ZOOM_2}", (17 - (int)zoom + 1).ToString());
  128.             url = url.Replace("{OAM_ZOOM}", (17 - (int)zoom).ToString());
  129.             url = url.Replace("{GOOG_DIGIT}", ((x + y) & 3).ToString());
  130.             url = url.Replace("{GOOG_QUAD}", GetQuadtreeFromXYZoom(x, y, zoom));
  131.             url = url.Replace("{MS_DIGITBR}", ((((y & 1) << 1) + (x & 1)) + 1).ToString());
  132.             url = url.Replace("{MS_DIGIT}", ((((y & 3) << 1) + (x & 1))).ToString());
  133.             url = url.Replace("{Y_DIGIT}", ((x + y + zoom) % 3 +1).ToString());
  134.             url = url.Replace("{GALILEO}", "Galileo".Substring(0, ((3 * x + y) & 7)));
  135.             // support old style {} vars
  136.             url = url.Replace("QQQQ", GetQuadtreeFromXYZoom(x,y,zoom));
  137.             url = url.Replace("XXXX", x.ToString());
  138.             url = url.Replace("YYYY", y.ToString());
  139.             url = url.Replace("ZZZZ", (17 - (int)zoom).ToString());
  140.             url = url.Replace("{OSM_ZOOM}", ((int)zoom).ToString());
  141.             url = url.Replace("{MS_QUADCODE}", quadcode);
  142.             url = url.Replace("*", ((x+y+zoom) %4).ToString());
  143.             return url;
  144.         }
  145.         static void Main(string[] args)
  146.         {
  147.             string url = BuildUrl("GoogleMap", 1, 1, 1);
  148.             //return <a href="http://mt2.google.com/vt/lyrs=m@121&hl=en&x=1&y=1&z=1&s=Gali">http://mt2.google.com/vt/lyrs=m@121&hl=en&x=1&y=1&z=1&s=Gali</a>
  149.         }
  150.     }
  151. }

例子中取出 google map ,在 zoom=1 及 x=1,y=1是的图片。
http://mt2.google.com/vt/lyrs=m@121&hl=en&x=1&y=1&z=1&s=Gali

下载工具  http://www.mapdigit.com/forum/viewthread.php?tid=43&extra=page%3D1 中文说明参见离线地图生成工具结构图离线地图生成工具使用方法