gpt4 book ai didi

python - 在 matplotlib 中注释子图可将图形缩放到最大轴

转载 作者:行者123 更新时间:2023-12-02 07:21:50 24 4
gpt4 key购买 nike

当我用 5 个子图制作图形并注释每个子图中的条形时,matplotlib 似乎会缩放图形,以便从最大 y 轴缩放到最小 y 轴的最大值。

我无法很好地描述问题,但请看这张图片:

在图形应该开始的地方上方有大量的空白。

但是,理想情况下该图应如下所示

this

当我将 4 个最小轴设置为与最大轴具有相同的 y 上限时,图形会正确缩放,但出于可视化的目的,我宁愿不这样做。

为什么会发生这种情况?是否有办法控制图形,使其不会像第一张图片中那样自动缩放?或者,有一种更合适的方式来绘制我希望实现的目标?

我用来生成图形的代码:

import numpy as np
from matplotlib import pyplot as plt
from matplotlib.patches import Patch
from matplotlib import rcParams
rcParams['font.family'] = 'sans-serif'
rcParams['font.sans-serif'] = ['Arial']
department = ["100", "1,000", "10,000", \
"100,000", "1,000,000"]
quarter = ["Serial", "MPI", "CUDA", "Hybrid"]
budgets = np.array([[0.049979, 0.43584, 2.787366, 19.75062, 201.6935],\
[2.184624, 0.175213, 0.677837, 5.265575, 46.33678],\
[0.050294, 0.068537, 0.23739, 1.93778, 18.55734],\
[3.714284, 3.9917, 4.977599, 6.174967, 37.732232]])

budgets = np.transpose(budgets)
em = np.zeros((len(department), len(quarter)))

# set up barchart
x = np.arange(len(department)) # label locations
width = 0.8 # width of all the bars

# set up figure
fig, (ax1, ax2, ax3, ax4, ax5) = plt.subplots(1, 5)
axes = [ax1, ax2, ax3, ax4, ax5]

# generate bars
rects = []
color = ["tomato", "royalblue", "limegreen", "orange"]
n = len(quarter)
for i in range(n):
bar_x = x - width/2.0 + i/float(n)*width + width/(n*2)

m = len(budgets[:,i])
for j in range(m):
bar_x = x[j] - width/2.0 + i/float(n)*width + width/(n*2)
e = budgets[j,i]
#bar_x = x - width/2.0 + i/float(n)*width + width/(n*2)
rects.append(axes[j].bar(bar_x, e, width=width/float(n), \
label=quarter[i], color=color[i]))

# set figure properties
fig.set_size_inches(12, 2.5)
fig.tight_layout(rect=[0, 0.03, 1, 0.95])
nAx = len(axes)
for i in range(nAx):
#axes[i].set_aspect("auto")
axes[i].tick_params(axis='x', which='both', bottom=False, top=False,
labelbottom=False)

ax1.set_ylabel("Time (ms)")
for i in range(nAx):
axes[i].yaxis.grid(which="major", color="white", lw=0.75)
ax1.set_ylim([0, 4])

fig.suptitle("Time per iteration for differing dataset sizes") # title

for i in range(nAx):
axes[i].set_xlabel(department[i])

# annotate bars
for i in range(nAx):
for rect in rects:
j = 0;
for bar in rect:
y_bottom, y_top = axes[i].get_ylim() # axis limits

height = bar.get_height() # bar's height

va = 'bottom'
offset = 3
color = 'k'
fg = 'w'

# keep label within plot
if (y_top < 1.1 * height):
offset = -3
va = 'top'
color='w'
fg = 'k'

# annotate the bar
axes[i].annotate('{:.2f}'.format(height),
xy=(bar.get_x() + bar.get_width()/2, height),
xytext=(0,offset),
textcoords="offset points",
ha='center', va=va, color=color)


# set custom legend
legend_elements = [Patch(facecolor='tomato', label='Serial'),
Patch(facecolor='royalblue', label='MPI'),
Patch(facecolor='limegreen', label='CUDA'),
Patch(facecolor='orange', label='Hybrid')]
plt.legend(handles=legend_elements, loc="upper center", fancybox=False,
edgecolor='k', ncol=4, bbox_to_anchor=(-2, -0.1))

plt.show()

最佳答案

这是部分答案。

这可能是一个错误,因为我无法重现该问题,直到我切换到 Debian 系统中的 Jupyter 笔记本(硬件也不同)。当从 .py 脚本显示时,您的图形在我的 macOS Jupyter 笔记本中以及在 Debian 中正确绘制。

问题似乎出在您的注释上。如果您在注释后调用 tight_layout ,您可能会收到如下警告:

<ipython-input-80-f9f592f5efc5>:88: UserWarning: Tight layout not applied. The bottom and top margins cannot be made large enough to accommodate all axes decorations. 
fig.tight_layout(rect=[0, 0.03, 1, 0.95])

annotate 函数似乎正在为您的注释计算一些完全古怪的坐标,尽管文本最终位于正确的位置。如果删除它们,空白就会消失。您可以尝试计算注释 a different wayxy 坐标 a 。这可能会帮助您开始:

        axes[i].annotate('{:.2f}'.format(height),
xy=(bar.get_x() + bar.get_width()/2, height),
xytext=(0,offset),
textcoords="offset points",
xycoords="axes points", # change
ha='center', va=va, color=color)

输出:

enter image description here

要正确计算点,您可以尝试使用适当的轴 transformation ,不过,我还是无法让它工作,这可能与错误有关。

关于python - 在 matplotlib 中注释子图可将图形缩放到最大轴,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61071456/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com