绘制 MetPy Q 向量

Plotting MetPy Q-vectors

我在根据 GFS 数据绘制来自 metpy.calc 的 Q 向量时遇到问题;应用 ax.set_extentax.quiver

时,我无法正确绘制矢量

计算代码:

import metpy.calc as mpcalc

query.variables('Temperature_isobaric', 'Geopotential_height_isobaric',
                'u-component_of_wind_isobaric', 'v-component_of_wind_isobaric')

data = subset_access.get_data(query)

lat = data.variables['lat'][:]
lon = data.variables['lon'][:]

press = data.variables['isobaric'][:] * units.Pa

# Make the pressure same dimensions as the temperature and winds
pressure_for_calc = press[:, None, None]

temperature = data.variables['Temperature_isobaric'][0] * units.kelvin

u = data.variables['u-component_of_wind_isobaric'][0] * units.meter / 
    units.second

v = data.variables['v-component_of_wind_isobaric'][0] * units.meter / 
    units.second

dx, dy = mpcalc.lat_lon_grid_deltas(lon, lat)

现在我试图通过 q 向量函数 运行 dx 和 dy:

Q = mpcalc.q_vector(u,v,temperature,pressure_for_calc,dx,dy)

但我收到一个错误,我认为与 dx 和 dy 维度有关:

IndexError: too many indices for array

dx.shape, dy.shape
>>> ((101, 160), (100, 161))

好的,这显然是个问题;我每个都需要一个一维数组,所以我探测了温度数组的形状:

print(temperature.shape)
>>> (31, 101, 161)

所以我尝试了 dx 和 dy 的一个子集:

print(dx[:,0].shape, dy[0,:].shape)
>>> (101,) (161,)

然后我认为这应该与温度和压力数组对齐,我根据这些子集再次尝试计算:

Q = mpcalc.q_vector(u,v,temperature,pressure_for_calc,dx[0,:],dy[:,0])

没有错误,现在感觉很好。检查我假设为 x 和 y 分量的 Q 的维度:

 print(Q[0].shape, Q[1].shape)
 >>> (31, 101, 161)
 >>> (31, 101, 161)

好像要排队...

但是,当我查看纬度和经度的尺寸时:

lat.shape, lon.shape
>>> ((101,), (161,))

dx和dy的形状好像倒过来了?

我是不是遗漏了什么,或者我只是在计算 Q 向量时完全错了?这是我的第一次尝试,我不确定我所做的事情是否正确。

当我尝试用 ax.quiver

的任何投影绘制它们时,真正的问题就来了

绘图代码:

import matplotlib.pyplot as plt
import cartopy.crs as ccrs
import cartopy.feature as cfeature

# Set Projection of Data
datacrs = ccrs.PlateCarree()

# Set Projection of Plot
plotcrs = ccrs.LambertConformal()

# Add Map Features
states_provinces = cfeature.NaturalEarthFeature(category='cultural',
    name='admin_1_states_provinces_lakes',scale='50m', facecolor='none')

country_borders = cfeature.NaturalEarthFeature(category='cultural',
    name='admin_0_countries',scale='50m', facecolor='none')

# Lat/Lon Extents [lon0,lon1,lat0,lat1]
extent = [-130., -70, 20., 60.]
# Create a plot
fig = plt.figure(figsize=(17., 11.))

# Add the map
ax = plt.subplot(111, projection=plotcrs)

# Add state boundaries to plot
ax.add_feature(states_provinces, edgecolor='k', linewidth=1)

# Add country borders to plot
ax.add_feature(country_borders, edgecolor='black', linewidth=1)
lon_slice = slice(None, None, 8)
lat_slice = slice(None, None, 8)

ax.quiver(lon[lon_slice],lat[lat_slice],Q[0][0,lon_slice,lat_slice], Q[1][0,lon_slice,lat_slice],
color='k',transform=plotcrs)

ax.set_extent(extent, datacrs)

plt.show()

生成的地图:

当我省略 ax.set_extent 时,它似乎绘制了 Q 向量,只是现在没有地图背景...

所以我想我的两个问题是:

1) 我是否根据 GFS 数据适当地计算了 Q 向量?

2) 我错过了什么绘图?

所以我认为您正确计算了 Q 向量,但有一个更简单的解决方案。发生错误是因为您传递的是 dxdy 的二维数组,但您的字段 temperaturepressure_for_calc 是 3D 的。 NumPy 不知道它应该为每个高度级别重复 dx 和 dy。您可以在不进行子集化的情况下完成此操作:

Q = mpcalc.q_vector(u, v, temperature,pressure_for_calc, dx[None, :], dy[None, :])

这样做是插入一个大小为 1 的维度作为 dxdy 的第一个维度,其余维度不受影响。这允许一切与其他阵列正确对齐。

就绘图而言,这是一个经典的 CartoPy 问题。您对 quiver 的调用应如下所示:

ax.quiver(lon[lon_slice], lat[lat_slice],
          Q[0][0,lon_slice,lat_slice].m, Q[1][0,lon_slice,lat_slice].m,
          color='k', transform=ccrs.PlateCarree())

注意要通过 transform=ccrs.PlateCarree() 的更改。这是告诉 CartoPy 您传递给 quiver 的数据在 longitude/latitude 坐标系中的方法。这还假设您正在绘制的矢量在此坐标系中被正确引用——它们应该是因为您通过了 dxdympcalc.lat_lon_grid_deltas() 计算得出。请注意,在这种情况下,由于 CartoPy 将对向量进行一些重新投影,因此您需要使用 .m 来删除单位。