От растрового Matplotlib к Geoviews / Holoviews / hvplot: как преобразовать x, y и z - PullRequest
0 голосов
/ 01 марта 2019

Я понимаю, что Geoviews и Holoviews имеют общие атрибуты, и Hvplot должен быть высокоуровневым API для построения графиков для всех трех.

Теперь, исходя из Matplotlib, мне все еще трудно адаптироваться к параметрам, необходимым дляотображать растровые изображения в Geoviews или Holoviews.

Вот пример, где я делаю оценку плотности ядра для пространственных данных:

# add coordinates of observations
xy_train = np.vstack([y, x]).T
print(xy_train)
# [[5654810.66920637  413645.79802685]
# [5654712.51814666  412629.87266155]
# [5656120.03682466  411642.74943511]
# ...
# [5656316.96943554  411795.80163676]
# [5656299.73356505  411795.50717494]
# [5655756.85624901  411734.34680852]]

# create mesh
xbins=100j
ybins=100j
xx, yy = np.mgrid[left_bound:right_bound:xbins, 
                bottom_bound:top_bound:ybins]
xy_sample = np.vstack([yy.ravel(), xx.ravel()]).T
# compute Kernel Density here
# ..
kde = KernelDensity(kernel='gaussian', bandwidth=100, algorithm='ball_tree')
kde.fit(xy_train)
# get results
z = np.exp(kde.score_samples(xy_sample))
# reshape results to mesh
zz = z.reshape(xx.shape)
# plot in matplotlib
fig, ax_lst = plt.subplots(111)
levels = np.linspace(zz.min(), zz.max(), 25)
axis.contourf(xx, yy, zz, levels=levels, cmap='viridis')
axis.plot()
plt.show()

Показывает мое изображение: enter image description here

Теперь я хочу использовать среду pyviz для интерактивного отображения и наложения карты, например, используя Geoviews.

Это как-то работает, но выдает ошибку:

xr_dataset = gv.Dataset(hv.Image((xx, yy, zz.T), datatype=['grid']), crs=ccrs.UTM(zone='33N'))

Image02195: Размеры изображения x и y неравномерно выбраны с относительным допуском 0,001.Пожалуйста, используйте элемент QuadMesh для нерегулярных выборок данных или установите более высокий допуск для hv.config.image_rtol или параметра rtol в конструкторе Image.

Я все еще могу отображать изображение (каким-то образом с низким разрешением).

gv.tile_sources.CartoLight.opts(width=800, height=480) * xr_dataset.to.image(['x', 'y']).opts(cmap='viridis', alpha=0.5)

enter image description here

.. но когда я пытаюсь создать FilledContours в Geoviews, кажется, что он не работает, как в matplotlib:

gv.FilledContours(xx, yy, zz, levels=levels, cmap='viridis')

ValueError: аргумент kdims ожидает измерение или список измерений, заданных как кортежи, строки, словари или экземпляры измерений, а не тип ndarray.Убедитесь, что вы передали данные в качестве первого аргумента.

Документация не содержит много информации о том, как я должен форматировать измерения (hv.help(gv.FilledContours)).Я думаю, что где-то теряюсь, когда мне нужно создать растр из координатной сетки xx / yy (hv.Image((xx, yy, zz.T), datatype=['grid'])).

Может кто-нибудь объяснить разницу в синтаксисе, которая требуется для matplotlib Contourf и Holoviews / Geoviews / Hvplot FilledContours?

[править]

Я нашел способ создания контуров,но проблема с размерами сохраняется:

# get xarray dataset, suited for handling raster data in pyviz
xr_dataset = gv.Dataset(hv.Image((xx.T, yy.T, zz.T), bounds=(left_bound,bottom_bound,right_bound,top_bound), 
        kdims=[hv.Dimension('x'),  hv.Dimension('y')], datatype=['grid']), crs=ccrs.UTM(zone='33N'))
# Error: Image06593: Image dimension(s) x and y are not evenly sampled to relative tolerance of 0.001

# create contours from image
gv.FilledContours(xr_dataset)

# plot
gv.tile_sources.EsriImagery.opts(width=800, height=480) * gv.FilledContours(xr_dataset).opts(cmap='viridis', alpha=0.5)

enter image description here

1 Ответ

0 голосов
/ 01 марта 2019

Главное, что нужно знать об элементах HoloViews / GeoViews, это то, что данные почти всегда указываются в качестве первого аргумента, в отличие от matplotlib, где данные часто указываются с использованием нескольких аргументов.В вашем случае у вас уже был правильный синтаксис для Image, но вы не перенесли его на другие элементы.Таким образом, чтобы сделать это конкретнее, чтобы построить изображение, вы должны сделать:

img = gv.Image((xx, yy, zz.T), crs=ccrs.UTM(zone='33N'))

Однако, поскольку у вас есть двухмерные координатные массивы, а не одномерные координаты, которых ожидает изображение (в следующем выпуске это приведет к ошибке), вына самом деле есть QuadMesh, который получается таким же образом:

qmesh = gv.QuadMesh((xx, yy, zz.T), crs=ccrs.UTM(zone='33N'))

И то же самое верно и для geoviews FilledContours:

contours = gv.FilledContours((xx, yy, zz.T), crs=ccrs.UTM(zone='33N'))

Итак, подведем итог,Разница между элементами HoloViews и вызовами matplotlib заключается в том, что HoloViews представляет собой облегченную оболочку вокруг ваших данных, которая позволяет придать каждому координатному и массиву значений семантическое значение, присваивая их измерению ключа или значения, в то время как matplotlib делает это отображение более явным.

HoloViews понимает ряд форматов для определения данных в сетке, таких как ваш, самый простой - это набор массивов координат x / y и массив значений, он также понимает объекты xarray и словари различных массивов, которые выглядятнапример:

contours = gv.FilledContours({'x': xx, 'y': yy, 'z': zz.T}, ['x', 'y'], 'z', crs=ccrs.UTM(zone='33N'))

В этом формате мы можем явно видеть, как координатные массивы 'x' и 'y' отображаются в ключевые измерения, а массив значений 'z' - в измерения значений.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...