Matplotlibight_layout()未考虑人物字幕


207

如果我在matplotlib图上添加了字幕,则该字幕会被该字幕图的标题覆盖。有人知道如何轻松地解决这个问题吗?我尝试过该tight_layout()功能,但只会使情况变得更糟。

例:

import numpy as np
import matplotlib.pyplot as plt

f = np.random.random(100)
g = np.random.random(100)
fig = plt.figure()
fig.suptitle('Long Suptitle', fontsize=24)
plt.subplot(121)
plt.plot(f)
plt.title('Very Long Title 1', fontsize=20)
plt.subplot(122)
plt.plot(g)
plt.title('Very Long Title 2', fontsize=20)
plt.tight_layout()
plt.show()

Answers:


178

您可以在tight_layout调用中调整子图的几何形状,如下所示:

fig.tight_layout(rect=[0, 0.03, 1, 0.95])

如文档(https://matplotlib.org/users/tight_layout_guide.html)中所述:

tight_layout()仅考虑刻度标签,轴标签和标题。因此,其他艺术家可能会被裁剪,也可能会重叠。


1
它像魅力一样工作,但是在指定边界框时如何发生变化?我读了文档,但对我来说并不清楚。如果您能向我解释,那就太好了!谢谢。
thepunitsingh

2
你能用rect解释每个数字的含义吗?我在文档中没有看到它们。
史蒂芬

6
@steven查看tight_layout文档:[left, bottom, right, top] in normalized (0, 1) figure coordinates
汤普劳

1
您可以通过降低最合适的值来进一步增加字幕和坐标轴之间的距离:tight_layout(rect=[0, 0.03, 1, 0.9])而不是0.95像答案中那样。
ComFreek

1
在jupyter内部不起作用。尝试对数字使用不同的值没有效果
Aleksejs Fomins

114

您可以使用plt.subplots_adjust(top=0.85)以下方法手动调整间距:

import numpy as np
import matplotlib.pyplot as plt

f = np.random.random(100)
g = np.random.random(100)
fig = plt.figure()
fig.suptitle('Long Suptitle', fontsize=24)
plt.subplot(121)
plt.plot(f)
plt.title('Very Long Title 1', fontsize=20)
plt.subplot(122)
plt.plot(g)
plt.title('Very Long Title 2', fontsize=20)
plt.subplots_adjust(top=0.85)
plt.show()

5
请注意,致电时top的默认值为0.9subplots_adjust
Brian Burns

72
我不敢相信在我们的领主2017年,这仍然是matplotlib的错误。
wordforthewise

29
请注意,subplots_adjust必须在调用之后再调用tight_layout,否则调用不会生效subplots_adjust
Achal Dave

7
@wordsforthewise现在使2018年
vlsd

6
@vlsd您好,2019年
陆小雨

55

您可以很容易地在代码中更改的一件事就是fontsize您正在使用标题。但是,我将假设您不只是想这样做!

一些替代使用fig.subplots_adjust(top=0.85)

通常tight_layout(),在将所有内容放置在合适的位置方面做得很好,以免它们重叠。tight_layout()在这种情况下无济于事的原因是因为tight_layout()没有考虑到fig.suptitle()。GitHub上有一个未解决的问题:https : //github.com/matplotlib/matplotlib/issues/829 [由于需要完整的几何管理器,于2014年关闭-移至https://github.com/matplotlib/matplotlib / issues / 1109 ]。

如果您阅读该线程,则可以解决涉及的问题GridSpec。关键是在tight_layout使用rectkwarg 调用时,在图的顶部保留一些空间。对于您的问题,代码变为:

使用GridSpec

import numpy as np
import matplotlib.pyplot as plt
import matplotlib.gridspec as gridspec

f = np.random.random(100)
g = np.random.random(100)

fig = plt.figure(1)
gs1 = gridspec.GridSpec(1, 2)
ax_list = [fig.add_subplot(ss) for ss in gs1]

ax_list[0].plot(f)
ax_list[0].set_title('Very Long Title 1', fontsize=20)

ax_list[1].plot(g)
ax_list[1].set_title('Very Long Title 2', fontsize=20)

fig.suptitle('Long Suptitle', fontsize=24)    

gs1.tight_layout(fig, rect=[0, 0.03, 1, 0.95])  

plt.show()

结果:

使用gridspec

可能GridSpec对您来说有点过大,否则您的实际问题将涉及在更大的画布上进行更多的子图绘制或其他复杂情况。一个简单的技巧是仅使用annotate()并将其锁定'figure fraction'到来模仿suptitle。不过,一旦查看输出,您可能需要进行一些更精细的调整。请注意,这第二个解决方案并不能使用tight_layout()

更简单的解决方案(尽管可能需要微调)

fig = plt.figure(2)

ax1 = plt.subplot(121)
ax1.plot(f)
ax1.set_title('Very Long Title 1', fontsize=20)

ax2 = plt.subplot(122)
ax2.plot(g)
ax2.set_title('Very Long Title 2', fontsize=20)

# fig.suptitle('Long Suptitle', fontsize=24)
# Instead, do a hack by annotating the first axes with the desired 
# string and set the positioning to 'figure fraction'.
fig.get_axes()[0].annotate('Long Suptitle', (0.5, 0.95), 
                            xycoords='figure fraction', ha='center', 
                            fontsize=24
                            )
plt.show()

结果:

简单

[使用Python2.7.3(64位)和matplotlib1.2.0]


1
谢谢,如果我使用您建议的第一个解决方案(使用GridSpec),如何创建共享轴的子图?我通常plt.subplots(N,1, sharex=<>, sharey=<>)在创建子图时使用,但是您发布的代码add_subplot改为使用
Amelio Vazquez-Reina 2014年

10
投射fig.tight_layout(rect=[0, 0.03, 1, 0.95])也可以。
soupault 2015年

1
第二种解决方案是唯一对我有用的解决方案。谢谢!
哈格巴尔

19

一种替代且易于使用的解决方案是在调用suptitle时使用y参数来调整图中字幕文本的坐标(请参阅docs):

import numpy as np
import matplotlib.pyplot as plt

f = np.random.random(100)
g = np.random.random(100)
fig = plt.figure()
fig.suptitle('Long Suptitle', y=1.05, fontsize=24)
plt.subplot(121)
plt.plot(f)
plt.title('Very Long Title 1', fontsize=20)
plt.subplot(122)
plt.plot(g)
plt.title('Very Long Title 2', fontsize=20)
plt.show()

8
这是在笔记本电脑中的好方法,但是plt.savefig()命令不服从。该解决方案仍然在savefig命令中切断标题。
超级英雄

11

紧密的布局不适用于字幕,但constrained_layout可以。查看此问题使用matplotlib中的许多子图来提高子图大小/间距

我发现立即添加子图看起来更好,即

fig, axs = plt.subplots(rows, cols, constrained_layout=True)

# then iterating over the axes to fill in the plots

但是也可以在创建图形时添加它:

fig = plt.figure(constrained_layout=True)

ax1 = fig.add_subplot(cols, rows, 1)
# etc

注意:为了使子图更加紧密,我还使用了

fig.subplots_adjust(wspace=0.05)

和constrained_layout不适用于此:(


4

我一直在努力与matplotlib修剪方法,所以我现在只是做了功能通过做这个bash调用ImageMagickmogrify命令,它运作良好,并得到所有多余的空白关图的边缘。这要求您正在使用UNIX / Linux,正在使用bashShell,并且已经ImageMagick安装。

拨打电话后,只需对此savefig()拨打电话即可。

def autocrop_img(filename):
    '''Call ImageMagick mogrify from bash to autocrop image'''
    import subprocess
    import os

    cwd, img_name = os.path.split(filename)

    bashcmd = 'mogrify -trim %s' % img_name
    process = subprocess.Popen(bashcmd.split(), stdout=subprocess.PIPE, cwd=cwd)

3

正如其他人所提到的,默认情况下,紧凑的布局不考虑字幕。但是,我发现可以使用该bbox_extra_artists参数来传递字幕作为应该考虑的边界框:

st = fig.suptitle("My Super Title")
plt.savefig("figure.png", bbox_extra_artists=[st], bbox_inches='tight')

这迫使紧密的布局计算要考虑suptitle在内,并且看起来就像您期望的那样。


这是对我有用的唯一选择。我使用Jupyter笔记本制作图形并保存它们。我在Linux机器上使用python 3.6工作。
GRquanti

1

当我使用tight_layout非常大的网格图(超过200个子图)并在jupyter笔记本中进行渲染时,出现了类似的问题。我制定了一个快速解决方案,始终将您suptitle放在顶部子图上方一定距离处:

import matplotlib.pyplot as plt

n_rows = 50
n_col = 4
fig, axs = plt.subplots(n_rows, n_cols)

#make plots ...

# define y position of suptitle to be ~20% of a row above the top row
y_title_pos = axs[0][0].get_position().get_points()[1][1]+(1/n_rows)*0.2
fig.suptitle('My Sup Title', y=y_title_pos)

对于可变大小的子图,您仍然可以使用此方法获得最顶层子图的顶部,然后手动定义一个附加量以添加到字幕中。


1

唯一对我有用的是修改对suptitle的调用:

fig.suptitle("title", y=.995)
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.