在 Pandas 条形图上用值注释条形

Posted

技术标签:

【中文标题】在 Pandas 条形图上用值注释条形【英文标题】:Annotate bars with values on Pandas bar plots 【发布时间】:2014-10-16 08:21:18 【问题描述】:

我正在寻找一种方法来使用我的 DataFrame 中的舍入数值在 Pandas 条形图中注释我的条形。

>>> df=pd.DataFrame('A':np.random.rand(2),'B':np.random.rand(2),index=['value1','value2'] )         
>>> df
                 A         B
  value1  0.440922  0.911800
  value2  0.588242  0.797366

我想得到这样的东西:

我尝试使用此代码示例,但注释都以 x 刻度为中心:

>>> ax = df.plot(kind='bar') 
>>> for idx, label in enumerate(list(df.index)): 
        for acc in df.columns:
            value = np.round(df.ix[idx][acc],decimals=2)
            ax.annotate(value,
                        (idx, value),
                         xytext=(0, 15), 
                         textcoords='offset points')

【问题讨论】:

Tom 搞定了,但我有一个更复杂的解决方案:***.com/questions/19917587/… 【参考方案1】:

As of matplotlib 3.4.0:

为自动标记条形图添加了新的 Axes.bar_label 辅助方法。

对于单组条形图,提供ax.containers[0]:

df = pd.DataFrame('A': np.random.rand(2), index=['value1', 'value2'])
ax = df.plot.barh()

ax.bar_label(ax.containers[0])

对于多组条形图,迭代ax.containers

df = pd.DataFrame('A': np.random.rand(2), 'B': np.random.rand(2), index=['value1', 'value2'])
ax = df.plot.bar()

for container in ax.containers:
    ax.bar_label(container)

有关使用可选样式参数的综合示例,请参阅 matplotlib's bar label demos:

Axes.bar_label(self, container, labels=None, *, fmt='%g', label_type='edge', padding=0, **kwargs)

【讨论】:

很好的答案,与熊猫配合得很好【参考方案2】:

斧头告诉我们盒子的大小。

x_position=##define a value
y_position=##define a value
for patch in ax.patches:
    b= patch.get_bbox()
    y_value=b.y1-b.y0
    ax.annotate(y_value, "x_position" , "y_position"))
plt.show()

为了更清楚:: Bbox(x0=3.75, y0=0.0, x1=4.25, y1=868.0) Bbox(x0=4.75, y0=0.0, x1=5.25, y1=868.0) Bbox(x0=5.75, y0=0.0, x1=6.25, y1=1092.0) Bbox(x0=6.75, y0=0.0, x1=7.25, y1=756.0) Bbox(x0=7.75, y0=0.0, x1=8.25, y1=756.0) Bbox(x0=8.75, y0=0.0, x1=9.25, y1=588.0) Bbox(x0=3.75, y0=868.0, x1=4.25, y1=3724.0) Bbox(x0=4.75, y0=868.0, x1=5.25, y1=3528.0) Bbox(x0=5.75, y0=1092.0, x1=6.25, y1=3948.0) Bbox(x0=6.75, y0=756.0, x1=7.25, y1=2884.0) Bbox(x0=7.75, y0=756.0, x1=8.25, y1=3024.0) Bbox(x0=0.75, y0=4004.0, x1=1.25, y1=4396.0) Bbox(x0=1.75, y0=3668.0, x1=2.25, y1=4060.0) Bbox(x0=2.75, y0=3864.0, x1=3.25, y1=4060.0)

这是我的程序中 patch.get_bbox() 的输出。 我们可以从这里提取边界框细节并根据我们的要求进行操作

【讨论】:

【参考方案3】:

您可以直接从轴的补丁中获得它:

for p in ax.patches:
    ax.annotate(str(p.get_height()), (p.get_x() * 1.005, p.get_height() * 1.005))

您需要调整字符串格式和偏移量以使事物居中,也许使用p.get_width() 的宽度,但这应该可以帮助您入门。除非您在某处跟踪偏移量,否则它可能不适用于堆积条形图。

【讨论】:

感谢@TomAugsPurger,以下方法有效:for p in ax.patches: ax.annotate(np.round(p.get_height(),decimals=2), (p.get_x()+p.get_width()/2., p.get_height()), ha='center', va='center', xytext=(0, 10), textcoords='offset points') 另一个问题:您将如何处理带有负值的条形?使用上面的代码,标签总是带有正坐标,即使条形值为负。 单杠有没有类似的解决方案:kind=barh? 我发现这适用于 barh:ax.annotate(str(p.get_width()), (p.get_x() + p.get_width(), p.get_y()), xytext=(5, 10), textcoords='offset points') 谢谢@capitalistpug。我发现添加水平对齐会让你的效果更好。 ax.annotate(str(int(p.get_width())), (p.get_x() + p.get_width(), p.get_y()), xytext=(-2, 4), textcoords='offset points', horizontalalignment='right') +1【参考方案4】:

解决方案也可以处理带有样本浮点格式的负值。

仍然需要调整偏移量。

df=pd.DataFrame('A':np.random.rand(2)-1,'B':np.random.rand(2),index=['val1','val2'] )
ax = df.plot(kind='bar', color=['r','b']) 
x_offset = -0.03
y_offset = 0.02
for p in ax.patches:
    b = p.get_bbox()
    val = ":+.2f".format(b.y1 + b.y0)        
    ax.annotate(val, ((b.x0 + b.x1)/2 + x_offset, b.y1 + y_offset))

【讨论】:

以上是关于在 Pandas 条形图上用值注释条形的主要内容,如果未能解决你的问题,请参考以下文章

Python pandas / matplotlib在条形图列上方注释标签[重复]

Pandas,条形图注释

在 pandas 中创建一个条形图,x 轴为日期,另一列中的每个值一个条形图

Pandas - 绘制堆积条形图

如何向条形图添加多个注释

Pandas:带有两个条形图和两个 y 轴的条形图