放置颜色条#

颜色条指示图像数据的定量范围。在图形中放置颜色条并非易事,因为需要为它们腾出空间。

自动放置颜色条#

最简单的情况是将颜色条附加到每个坐标轴。请注意,在此示例中,颜色条会从父坐标轴中占用一些空间。

import matplotlib.pyplot as plt
import numpy as np

# Fixing random state for reproducibility
np.random.seed(19680801)

fig, axs = plt.subplots(2, 2)
cmaps = ['RdBu_r', 'viridis']
for col in range(2):
    for row in range(2):
        ax = axs[row, col]
        pcm = ax.pcolormesh(np.random.random((20, 20)) * (col + 1),
                            cmap=cmaps[col])
        fig.colorbar(pcm, ax=ax)
colorbar placement

第一列在两行中具有相同类型的数据,因此可能需要仅使用一个颜色条。我们通过将带有 *ax* kwarg 的坐标轴列表传递给 Figure.colorbar 来实现此目的。

fig, axs = plt.subplots(2, 2)
cmaps = ['RdBu_r', 'viridis']
for col in range(2):
    for row in range(2):
        ax = axs[row, col]
        pcm = ax.pcolormesh(np.random.random((20, 20)) * (col + 1),
                            cmap=cmaps[col])
    fig.colorbar(pcm, ax=axs[:, col], shrink=0.6)
colorbar placement

被占用的空间会导致同一子图布局中的坐标轴大小不同,如果每个图上的 x 轴都应该像下面那样具有可比性,这通常是不希望的

fig, axs = plt.subplots(2, 1, figsize=(4, 5), sharex=True)
X = np.random.randn(20, 20)
axs[0].plot(np.sum(X, axis=0))
pcm = axs[1].pcolormesh(X)
fig.colorbar(pcm, ax=axs[1], shrink=0.6)
colorbar placement

这通常是不希望的,并且可以通过各种方式解决,例如,向其他坐标轴添加颜色条,然后将其删除。但是,最直接的方法是使用 约束布局

fig, axs = plt.subplots(2, 1, figsize=(4, 5), sharex=True, layout='constrained')
axs[0].plot(np.sum(X, axis=0))
pcm = axs[1].pcolormesh(X)
fig.colorbar(pcm, ax=axs[1], shrink=0.6)
colorbar placement

使用此范例可以实现相对复杂的颜色条布局。请注意,此示例在 layout='constrained' 的情况下效果更好

fig, axs = plt.subplots(3, 3, layout='constrained')
for ax in axs.flat:
    pcm = ax.pcolormesh(np.random.random((20, 20)))

fig.colorbar(pcm, ax=axs[0, :2], shrink=0.6, location='bottom')
fig.colorbar(pcm, ax=[axs[0, 2]], location='bottom')
fig.colorbar(pcm, ax=axs[1:, :], location='right', shrink=0.6)
fig.colorbar(pcm, ax=[axs[2, 1]], location='left')
colorbar placement

调整颜色条和父坐标轴之间的间距#

可以使用 *pad* 关键字参数调整颜色条与父坐标轴的距离。这是以父坐标轴宽度的分数单位表示的,垂直坐标轴的默认值为 0.05(水平坐标轴为 0.15)。

fig, axs = plt.subplots(3, 1, layout='constrained', figsize=(5, 5))
for ax, pad in zip(axs, [0.025, 0.05, 0.1]):
    pcm = ax.pcolormesh(np.random.randn(20, 20), cmap='viridis')
    fig.colorbar(pcm, ax=ax, pad=pad, label=f'pad: {pad}')
fig.suptitle("layout='constrained'")
layout='constrained'

请注意,如果不使用约束布局,则 pad 命令会使父坐标轴缩小

fig, axs = plt.subplots(3, 1, figsize=(5, 5))
for ax, pad in zip(axs, [0.025, 0.05, 0.1]):
    pcm = ax.pcolormesh(np.random.randn(20, 20), cmap='viridis')
    fig.colorbar(pcm, ax=ax, pad=pad, label=f'pad: {pad}')
fig.suptitle("No layout manager")
No layout manager

手动放置颜色条#

有时,colorbar 提供的自动放置效果不理想。我们可以手动创建一个坐标轴,并通过将该坐标轴传递给 *cax* 关键字参数来告知 colorbar 使用该坐标轴。

使用 inset_axes#

我们可以手动创建任何类型的坐标轴供颜色条使用,但是 Axes.inset_axes 很有用,因为它是父坐标轴的子级,并且可以相对于父坐标轴进行定位。在这里,我们在父坐标轴底部附近居中添加一个颜色条。

fig, ax = plt.subplots(layout='constrained', figsize=(4, 4))
pcm = ax.pcolormesh(np.random.randn(20, 20), cmap='viridis')
ax.set_ylim([-4, 20])
cax = ax.inset_axes([0.3, 0.07, 0.4, 0.04])
fig.colorbar(pcm, cax=cax, orientation='horizontal')
colorbar placement

如果希望坐标轴位于图形上的某个数据位置,则 Axes.inset_axes 还可以使用 *transform* 关键字参数以数据坐标指定其位置

fig, ax = plt.subplots(layout='constrained', figsize=(4, 4))
pcm = ax.pcolormesh(np.random.randn(20, 20), cmap='viridis')
ax.set_ylim([-4, 20])
cax = ax.inset_axes([7.5, -1.7, 5, 1.2], transform=ax.transData)
fig.colorbar(pcm, cax=cax, orientation='horizontal')
colorbar placement

附加到固定纵横比坐标轴的颜色条#

为具有固定纵横比的坐标轴放置颜色条是一个特殊的挑战,因为父坐标轴的大小会根据数据视图而变化。

fig, axs = plt.subplots(2, 2,  layout='constrained')
cmaps = ['RdBu_r', 'viridis']
for col in range(2):
    for row in range(2):
        ax = axs[row, col]
        pcm = ax.pcolormesh(np.random.random((20, 20)) * (col + 1),
                            cmap=cmaps[col])
        if col == 0:
            ax.set_aspect(2)
        else:
            ax.set_aspect(1/2)
        if row == 1:
            fig.colorbar(pcm, ax=ax, shrink=0.6)
colorbar placement

我们使用 Axes.inset_axes 在“坐标轴坐标”中定位坐标轴来解决此问题(请参阅 变换教程)。请注意,如果放大父坐标轴,从而更改其形状,则颜色条的位置也会更改。

fig, axs = plt.subplots(2, 2, layout='constrained')
cmaps = ['RdBu_r', 'viridis']
for col in range(2):
    for row in range(2):
        ax = axs[row, col]
        pcm = ax.pcolormesh(np.random.random((20, 20)) * (col + 1),
                            cmap=cmaps[col])
        if col == 0:
            ax.set_aspect(2)
        else:
            ax.set_aspect(1/2)
        if row == 1:
            cax = ax.inset_axes([1.04, 0.2, 0.05, 0.6])
            fig.colorbar(pcm, cax=cax)
colorbar placement

脚本的总运行时间: (0 分钟 10.655 秒)

由 Sphinx-Gallery 生成的图库