Основы конфигурирования тайловых сеток

Материал из GIS-Lab
Перейти к навигации Перейти к поиску
Эта страница является черновиком статьи.


Введение

Предположим, что вы ознакомились со спецификацией Tile Map Service и решили создать собственный TMS-сервис. Спецификация предполагает, что у вас уже есть готовый набор тайлов или перечень необходимых настроек, согласно которому эти тайлы будут создаваться (см. раздел URL, являющиеся скриптами). Наиболее распространенной является ситуация, когда автор TMS-сервиса не только создаёт прослойку между готовым набором тайлов и клиентским приложением, но и отвечает за подготовку самих тайлов. В этом случае разработчик должен хорошо разбираться в вопросах, связанных с конфигурированием тайловых сеток. Даже если в его приложении тайлы создаются по запросу, то есть "на лету", эти знания все равно крайне необходимы.

Параметры тайловой сетки

Проекция

Прежде всего важно определиться с проекцией, которая будет использоваться при отрисовке данных. Для этого необходимо ответить на вопрос для каких целей будут предназначены ваши тайлы. Если вы хотите просто отображать свою карту поверх подложек, предоставляемых такими сервисами как OpenStreetMap или Google Maps, то вы должны использовать такую же проекцию, что и указанные сервисы, а именно EPSG:3857. Если же вы преследуете какие-то иные цели, например, желаете вычислять расстояния, измерять площади объектов на карте или планируете создать карту Антарктиды, то в этом случае вам необходимо выбрать наиболее подходящую для этих задач проекцию.

В качестве примера в данной статье мы будем конфигурировать тайловую сетку на территорию Алтайского края, поэтому для отрисовки данных будем использовать проекцию UTM зона 44 EPSG:32644, что в дальнейшем позволит нам осуществлять корректное измерение расстояний на полученной карте.

Охват

Следующий важный этап - это выбрать охват (ограничивающий прямоугольник), то есть ту область на которую будут генерироваться тайлы, вне ее тайлы не создаются. Параметры охвата Алтайского края получим на основе данных слоя административных границ набора Geosample. Для этого воспользуемся утилитой ogrinfo:

ogrinfo -sql "SELECT ST_Transform(the_geom, 32644) FROM admin WHERE name='Алтайский край'" PG:"host=gis-lab.info dbname=geosample user=guest password=guest" | grep Extent

В результате чего мы получим охват интересующей нас территории в единицах измерения системы координат проекции EPSG:32644 (minx, miny, maxx, maxy):

Extent: (287157.161574, 5613155.489664) - (920220.378205, 6045880.725611)

Масштабные уровни (разрешения)

После того, как мы определились с проекцией и охватом, необходимо разобраться с масштабными уровнями. Каждый масштабный уровень определяет разрешение с которым отрендерена карта на данном уровне, при этом количество таких уровней может быть произвольным. Величина разрешения, используемого на том или ином уровне также ничем не лимитирована. Разрешение показывает сколько единиц измерения координат исходных данных укладывается в одном пикселе отрендеренной карты. Таким образом в нашем случае (проекция UTM) единицы измерения разрешения - м/пиксел, в случае же работы с географическими системами координат это будут градус/пиксел.

Предположим, мы хотим предоставить доступ к нашим данным, отрендеренным с разрешениями R, равными 3000, 1500 и 1000 метров на пиксел. Посчитаем для этих разрешений размеры результирующей карты (w - ширина, h - высота):

3000

w = (maxx - minx)/R = (920220-287157)/3000 = 211 (пикселов)
h = (maxy - miny)/R = (6045880-5613155)/3000 = 144 (пикселов)

Zoom-level-0.png

1500

w = (maxx - minx)/R = (920220-287157)/1500 = 422 (пикселов)
h = (maxy - miny)/R = (6045880-5613155)/1500 = 288 (пикселов)

Zoom-level-1.png

1000

w = (maxx - minx)/R = (920220-287157)/1000 = 633 (пикселов)
h = (maxy - miny)/R = (6045880-5613155)/1000 = 432 (пикселов)

Zoom-level-2.png

Как уже было отмечено выше, разрешение i-го (i - целое больше 0) масштабного уровня может быть произвольным, однако на практике наиболее распространённым является случай, когда разрешение i-го уровня Ri в 2 раза выше разрешения предыдущего Ri-1:

Ri = Ri-1/2

Разрешение 0-го (i=0) масштабного уровня может устанавливаться произвольно, но опять же на практике широкое распространение получил следующий подход. Считается, что на нулевом уровне должно быть такое разрешение, которое бы позволило уместить карту целиком в один тайл. Чтобы его рассчитать, нам необходимо определиться с размерами тайла.

Размер тайла

Размер тайла (ширина и высота) определяет объём информации, передаваемой одним тайлом. Очевидно, что чем больше геометрические размеры тайла, тем больший объём дискового пространства он занимает и тем дольше он будет передаваться клиенту. Чем меньше размер тайла, тем он быстрее может быть доставлен клиенту, но при этом возрастает количество запросов с серверу, что в конечном итоге приводит к еще более длительному процессу получения тайлов. Экспериментальным путём было установлено, что наиболее эффективный размер тайла - 256x256 пикселов. Пример тайла сервиса OpenStreetMap:

Osm-tile-example.png

Вернёмся к нашей задаче определения разрешения 0-го уровня. Определим ширину (W) и высоту (H) карты в единицах измерения данных:

W = maxx-minx = 920220-287157 = 633063 (м)
H = maxy-miny = 6045880-5613155 = 432725 (м)

Тогда разрешение 0-го уровня можно вычислить так:

R0 = max(W,H)/256 = 633063/256 = 2472 (м/пиксел)

Можно записать общую формулу расчета разрешения i-го уровня карты размером WxH с размером тайла 256x256 с учётом соотношения Ri = Ri-1/2:

Ri = max(W,H)/(256*2^i)

Положение начала отсчёта тайловой сетки

Начало отсчёта тайловой сетки, задаваемое в единицах измерения исходных данных, определяет индексы тайлов, по которым они будут доступны. Согласно спецификации TMS начало отсчёта тайловой сетки может располагаться вне охвата карты. В этом случае тайл (0,0) может и не существовать, однако это достаточно экзотический случай. Наиболее распространённым является случай, когда начало отсчёта тайловой сетки совпадает с координатами левого нижнего угла охвата карты (minx, miny):

Zoom-level-2-ll.png

Другой часто встречающийся случай (не согласующийся со спецификацией TMS) - это когда положение начала отсчета тайловой сетки помещается в левую верхнюю точку охвата карты (minx, maxy):

Zoom-level-2-ul.png

Данный подход используется, например, в OpenStreetMap. Пример запроса тайла по адресу http://b.tile.openstreetmap.org/01/0/1.png:

Osm-tile-example-0-1.png

Как можно заметить на данном тайле представлена нижняя левая часть карты мира, но номер тайла при этом не (0,0), а (0,1).

Количество тайлов на различных масштабных уровнях

Чтобы оценить количество тайлов на каждом масштабном уровне воспользуемся консольной утилитой, идущей в составе кэширующего прокси-сервера MapProxy. Для этого вначале составим описание тайловой сетки в специальном конфигурационном файле MapProxy mapproxy.yaml. После прочтения вышеизложенного материала данной статьи у вас должно появиться четкое понимание описываемых в следующей конфигурации параметров:

grids:
  utmgrid:
    srs: EPSG:32644
    bbox_srs: EPSG:32644
    tile_size: [256, 256]
    origin: ll # low left
    bbox: [287157.161574,5613155.489664,920220.378205,6045880.725611]

Запускаем утилиту:

mapproxy-util grids --grid utmgrid --mapproxy-conf mapproxy.yaml

Получаем результат:

Configuration:
    bbox: [287157.161574, 5613155.489664, 920220.378205, 6045880.725611]
    bbox_srs: 'EPSG:32644'
    origin: 'll'
    srs: 'EPSG:32644'
    tile_size: [256, 256]
Levels: Resolutions, # x * y = total tiles
    00:  2472.9031899648435,   #      1 * 1      =        1
    01:  1236.4515949824217,   #      2 * 2      =        4
    02:  618.2257974912109,    #      4 * 3      =       12
    03:  309.11289874560543,   #      8 * 6      =       48
    04:  154.55644937280272,   #     16 * 11     =      176
    05:  77.27822468640136,    #     32 * 22     =      704
    06:  38.63911234320068,    #     64 * 44     =   2.816K
    07:  19.31955617160034,    #    128 * 88     =  11.264K
    08:  9.65977808580017,     #    256 * 175    =  44.800K
    09:  4.829889042900085,    #    512 * 350    = 179.200K
    10:  2.4149445214500425,   #   1024 * 700    = 716.800K
    11:  1.2074722607250212,   #   2048 * 1400   =   2.867M
    12:  0.6037361303625106,   #   4096 * 2800   =  11.469M
    13:  0.3018680651812553,   #   8192 * 5600   =  45.875M
    14:  0.15093403259062765,  #  16384 * 11200  = 183.501M
    15:  0.07546701629531383,  #  32768 * 22399  = 733.970M
    16:  0.03773350814765691,  #  65536 * 44797  =   2.936G
    17:  0.018866754073828457, # 131072 * 89594  =  11.743G
    18:  0.009433377036914228, # 262144 * 179187 =  46.973G
    19:  0.004716688518457114, # 524288 * 358373 = 187.891G

Как можно заметить, уже на 11 масштабном уровне количество тайлов превысило отметку в 1 миллион, а на 16 в 1 миллиард.

Проведём оценку количества тайлов на различных масштабных уровнях сервиса OpenStreetMap. Для этого вновь воспользуемся утилитой из состава MapProxy и указав в качестве имени сетки специальное служебное значение GLOBAL_MERCATOR:

mapproxy-util grids --grid GLOBAL_MERCATOR --mapproxy-conf mapproxy.yaml

Результат:

Configuration:
    bbox*: [-20037508.342789244, -20037508.342789244, 20037508.342789244, 20037508.342789244]
    origin*: 'sw'
    srs: 'EPSG:900913'
    tile_size*: [256, 256]
Levels: Resolutions, # x * y = total tiles
    00:  156543.03392804097,  #      1 * 1      =        1
    01:  78271.51696402048,   #      2 * 2      =        4
    02:  39135.75848201024,   #      4 * 4      =       16
    03:  19567.87924100512,   #      8 * 8      =       64
    04:  9783.93962050256,    #     16 * 16     =      256
    05:  4891.96981025128,    #     32 * 32     =   1.024K
    06:  2445.98490512564,    #     64 * 64     =   4.096K
    07:  1222.99245256282,    #    128 * 128    =  16.384K
    08:  611.49622628141,     #    256 * 256    =  65.536K
    09:  305.748113140705,    #    512 * 512    = 262.144K
    10:  152.8740565703525,   #   1024 * 1024   =   1.049M
    11:  76.43702828517625,   #   2048 * 2048   =   4.194M
    12:  38.21851414258813,   #   4096 * 4096   =  16.777M
    13:  19.109257071294063,  #   8192 * 8192   =  67.109M
    14:  9.554628535647032,   #  16384 * 16384  = 268.435M
    15:  4.777314267823516,   #  32768 * 32768  =   1.074G
    16:  2.388657133911758,   #  65536 * 65536  =   4.295G
    17:  1.194328566955879,   # 131072 * 131072 =  17.180G
    18:  0.5971642834779395,  # 262144 * 262144 =  68.719G
    19:  0.29858214173896974, # 524288 * 524288 = 274.878G

Так как в OpenStreetMap тайлы рендерятся только до 18 уровня, то получаем, что на последнем уровне количество тайлов достигает отметки 68 миллиардов.