572

I need to generate a whole bunch of vertically-stacked plots in matplotlib. The result will be saved using savefig and viewed on a webpage, so I don't care how tall the final image is, as long as the subplots are spaced so they don't overlap.

No matter how big I allow the figure to be, the subplots always seem to overlap.

My code currently looks like

import matplotlib.pyplot as plt
import my_other_module

titles, x_lists, y_lists = my_other_module.get_data()

fig = plt.figure(figsize=(10,60))
for i, y_list in enumerate(y_lists):
    plt.subplot(len(titles), 1, i)
    plt.xlabel("Some X label")
    plt.ylabel("Some Y label")
    plt.title(titles[i])
    plt.plot(x_lists[i],y_list)
fig.savefig('out.png', dpi=100)
1
  • This question also applies to pandas.DataFrame.plot with subplots, and to seaborn axes-level plots (those with the ax parameter): sns.lineplot(..., ax=ax) Commented May 2, 2022 at 17:34

11 Answers 11

758

Please review matplotlib: Tight Layout guide and try using matplotlib.pyplot.tight_layout, or matplotlib.figure.Figure.tight_layout

As a quick example:

import matplotlib.pyplot as plt

fig, axes = plt.subplots(nrows=4, ncols=4, figsize=(8, 8))
fig.tight_layout() # Or equivalently,  "plt.tight_layout()"

plt.show()

Without Tight Layout

enter image description here


With Tight Layout

enter image description here

3
  • 31
    So tight_layout() makes the layout.... less tight. Interesting name.
    – Harry
    Commented Nov 22, 2023 at 21:09
  • 2
    @Harry It's short for "hold tight. Imma layout this," Commented Mar 5 at 8:59
  • @Harry "Make it as tight as possible, but no more" :) One thing that's not clear from the MWE: tight_layout() should be called right before the plot is created/saved. If data is added after calling it, it won't work so nicely.
    – Dominik
    Commented Apr 26 at 10:44
537

You can use plt.subplots_adjust to change the spacing between the subplots.

call signature:

subplots_adjust(left=None, bottom=None, right=None, top=None, wspace=None, hspace=None)

The parameter meanings (and suggested defaults) are:

left  = 0.125  # the left side of the subplots of the figure
right = 0.9    # the right side of the subplots of the figure
bottom = 0.1   # the bottom of the subplots of the figure
top = 0.9      # the top of the subplots of the figure
wspace = 0.2   # the amount of width reserved for blank space between subplots
hspace = 0.2   # the amount of height reserved for white space between subplots

The actual defaults are controlled by the rc file

1
  • 2
    Nice! So, this works for me to increase the vertical separation between two subplots: plt.subplots_adjust(hspace=0.3). I call it just before plt.show(). Note to anyone wondering: the 0.3 is in units of inches I'm pretty sure. I'm not sure how they handle different screen sizes though... Commented May 22, 2023 at 21:31
110

Similar to tight_layout matplotlib now (as of version 2.2) provides constrained_layout. In contrast to tight_layout, which may be called any time in the code for a single optimized layout, constrained_layout is a property, which may be active and will optimze the layout before every drawing step.

Hence it needs to be activated before or during subplot creation, such as figure(constrained_layout=True) or subplots(constrained_layout=True).

Example:

import matplotlib.pyplot as plt

fig, axes = plt.subplots(4,4, constrained_layout=True)

plt.show()

enter image description here

constrained_layout may as well be set via rcParams

plt.rcParams['figure.constrained_layout.use'] = True

See the what's new entry and the Constrained Layout Guide

3
  • Does using constrained_layout = True squeeze the text? I understand using tight_layout changes all sizes, such that for example the font become smaller.
    – tetukowski
    Commented Jul 11, 2023 at 10:05
  • 1
    In many cases, constrained_layout is much better than tight_layout if one needs the subplots to fill the window, and even better and simpler than setting subplots_adjust manually. (tight_layout is actually far from tight... it's rather "loose".) Commented Dec 13, 2023 at 12:30
  • @tetukowski tight_layout definitely does not change fonts sizes. Commented Jan 16 at 15:38
98

Using subplots_adjust(hspace=0) or a very small number (hspace=0.001) will completely remove the whitespace between the subplots, whereas hspace=None does not.

import numpy as np
import matplotlib.pyplot as plt
import matplotlib.ticker as tic

fig = plt.figure(figsize=(8, 8))

x = np.arange(100)
y = 3.*np.sin(x*2.*np.pi/100.)

for i in range(1, 6):
    temp = 510 + i
    ax = plt.subplot(temp)
    plt.plot(x, y)
    plt.subplots_adjust(hspace=0)
    temp = tic.MaxNLocator(3)
    ax.yaxis.set_major_locator(temp)
    ax.set_xticklabels(())
    ax.title.set_visible(False)

plt.show()

hspace=0 or hspace=0.001

enter image description here

hspace=None

enter image description here

2
  • I believe you can move the call to plt.subplots_adjust(hspace=0) outside the loop and achieve the same result.
    – als0052
    Commented Oct 5, 2023 at 12:20
  • This didn't work for me at first, until I removed the plt.tight_layout() statement, which interferes...
    – AstroFloyd
    Commented May 6 at 15:34
37
import matplotlib.pyplot as plt

fig = plt.figure(figsize=(10,60))
plt.subplots_adjust( ... )

The plt.subplots_adjust method:

def subplots_adjust(*args, **kwargs):
    """
    call signature::

      subplots_adjust(left=None, bottom=None, right=None, top=None,
                      wspace=None, hspace=None)

    Tune the subplot layout via the
    :class:`matplotlib.figure.SubplotParams` mechanism.  The parameter
    meanings (and suggested defaults) are::

      left  = 0.125  # the left side of the subplots of the figure
      right = 0.9    # the right side of the subplots of the figure
      bottom = 0.1   # the bottom of the subplots of the figure
      top = 0.9      # the top of the subplots of the figure
      wspace = 0.2   # the amount of width reserved for blank space between subplots
      hspace = 0.2   # the amount of height reserved for white space between subplots

    The actual defaults are controlled by the rc file
    """
    fig = gcf()
    fig.subplots_adjust(*args, **kwargs)
    draw_if_interactive()

or

fig = plt.figure(figsize=(10,60))
fig.subplots_adjust( ... )

The size of the picture matters.

"I've tried messing with hspace, but increasing it only seems to make all of the graphs smaller without resolving the overlap problem."

Thus to make more white space and keep the sub plot size the total image needs to be bigger.

0
31

You could try the .subplot_tool()

plt.subplot_tool()
0
9
  • Resolving this issue when plotting a dataframe with pandas.DataFrame.plot, which uses matplotlib as the default backend.
    • The following works for whichever kind= is specified (e.g. 'bar', 'scatter', 'hist', etc.).
  • Tested in python 3.8.12, pandas 1.3.4, matplotlib 3.4.3

Example

import pandas as pd
import numpy as np
import matplotlib.pyplot as plt

# sinusoidal sample data
sample_length = range(1, 15+1)
rads = np.arange(0, 2*np.pi, 0.01)
data = np.array([np.sin(t*rads) for t in sample_length])
df = pd.DataFrame(data.T, index=pd.Series(rads.tolist(), name='radians'), columns=[f'freq: {i}x' for i in sample_length])

# default plot with subplots; each column is a subplot
axes = df.plot(subplots=True)

enter image description here

Adjust the Spacing

  • Adjust the default parameters in pandas.DataFrame.plot
    1. Change figsize: a width of 5 and a height of 4 for each subplot is a good place to start.
    2. Change layout: (rows, columns) for the layout of subplots.
    3. sharey=True and sharex=True so space isn't taken for redundant labels on each subplot.
  • The .plot method returns a numpy array of matplotlib.axes.Axes, which should be flattened to easily work with.
  • Use .get_figure() to extract the DataFrame.plot figure object from one of the Axes.
  • Use fig.tight_layout() if desired.
axes = df.plot(subplots=True, layout=(3, 5), figsize=(25, 16), sharex=True, sharey=True)

# flatten the axes array to easily access any subplot
axes = axes.flat

# extract the figure object
fig = axes[0].get_figure()

# use tight_layout
fig.tight_layout()

enter image description here

df

# display(df.head(3))
         freq: 1x  freq: 2x  freq: 3x  freq: 4x  freq: 5x  freq: 6x  freq: 7x  freq: 8x  freq: 9x  freq: 10x  freq: 11x  freq: 12x  freq: 13x  freq: 14x  freq: 15x
radians                                                                                                                                                            
0.00     0.000000  0.000000  0.000000  0.000000  0.000000  0.000000  0.000000  0.000000  0.000000   0.000000   0.000000   0.000000   0.000000   0.000000   0.000000
0.01     0.010000  0.019999  0.029996  0.039989  0.049979  0.059964  0.069943  0.079915  0.089879   0.099833   0.109778   0.119712   0.129634   0.139543   0.149438
0.02     0.019999  0.039989  0.059964  0.079915  0.099833  0.119712  0.139543  0.159318  0.179030   0.198669   0.218230   0.237703   0.257081   0.276356   0.295520
6
import matplotlib.pyplot as plt

# create the figure with tight_layout=True
fig, axes = plt.subplots(nrows=4, ncols=4, figsize=(8, 8), tight_layout=True)

enter image description here

4

If passing tight_layout=True to plt.subplots() or fig.tight_layout() is not adding sufficient spacing between subplots, consider tuning with pad like tight_layout(pad=2.0) to get desired spacing.

1
  • This works 100%. The underdog 🙌
    – reverie_ss
    Commented Jan 7 at 11:18
0

For me next config works the best:

plt.rcParams['figure.constrained_layout.use'] = True
0

I you wish to have control over the subplots spacing, you can specify it when calling plt.subplots through gridspace_kw:

fig, axes = plt.subplots(..., gridspec_kw=dict(hspace=..., wspace=...) )

Or specify the layout through the figure kwargs:

fig, axes = plt.subplots(..., layout="tight")

Not the answer you're looking for? Browse other questions tagged or ask your own question.