Difference between revisions of "Python:Plotting/Subplots"

From PrattWiki
Jump to navigation Jump to search
 
(4 intermediate revisions by the same user not shown)
Line 18: Line 18:
 
* [https://matplotlib.org/api/_as_gen/matplotlib.figure.Figure.html?highlight=subplots#matplotlib.figure.Figure.subplots fig.subplots()] needs you to create the figure handle first (probably with <code>fig=plt.figure()</code> and then you can use that figure variable to create an entire array of axes.
 
* [https://matplotlib.org/api/_as_gen/matplotlib.figure.Figure.html?highlight=subplots#matplotlib.figure.Figure.subplots fig.subplots()] needs you to create the figure handle first (probably with <code>fig=plt.figure()</code> and then you can use that figure variable to create an entire array of axes.
 
** The two main arguments for <code>fig.subplots(nrows, ncols)</code> will establish how many rows and columns you want to break the figure into.  It will return an array of axes handles (1D array if creating a single column or row, 2D array if there is more than one row and more than one column).  This is useful if you have an array of axes and you are planning to use all of them.  The only real difference between this version and the figure based one above is that this one will accept keyword arguments to be passed to the axes rather than to the figure.  This will become important when creating 3D plots.
 
** The two main arguments for <code>fig.subplots(nrows, ncols)</code> will establish how many rows and columns you want to break the figure into.  It will return an array of axes handles (1D array if creating a single column or row, 2D array if there is more than one row and more than one column).  This is useful if you have an array of axes and you are planning to use all of them.  The only real difference between this version and the figure based one above is that this one will accept keyword arguments to be passed to the axes rather than to the figure.  This will become important when creating 3D plots.
=== Single Subplot For The Whole Figure ===
+
== Single Subplot For The Whole Figure ==
 +
[[File:Subplots111.PNG|thumb|Figure window with one large subplot]]
 +
=== Using <code>fig.add_subplot()</code> ===
 
<syntaxhighlight lang="Python">
 
<syntaxhighlight lang="Python">
 
fig = plt.figure(num=1, clear=True)
 
fig = plt.figure(num=1, clear=True)
 
ax = fig.add_subplot(1, 1, 1)
 
ax = fig.add_subplot(1, 1, 1)
 
</syntaxhighlight>
 
</syntaxhighlight>
or
+
=== Using <code>plt.subplots()</code> ===
 +
The default for this command is to make a single subplot, so there is no row or column information necesary.
 
<syntaxhighlight lang="Python">
 
<syntaxhighlight lang="Python">
 
fig, ax = plt.subplots(num=1, clear=True)
 
fig, ax = plt.subplots(num=1, clear=True)
 
</syntaxhighlight>
 
</syntaxhighlight>
or
+
=== Using <code>fig.subplots()</code> ===
 +
The default for this command is to make a single subplot, so there is no row or column information necesary.
 +
<syntaxhighlight lang="Python">
 +
fig = plt.figure(num=1, clear=True)
 +
ax = fig.subplots()
 +
</syntaxhighlight>
 +
 
 +
== Single Row or Column of Subplots ==
 +
[[File:Subplots132.PNG|thumb|Figure window with one row of three subplots and random numbers in the middle subplot (after fig.tight_layout())]]
 +
The examples below will show how to set up and access a single row or column of subplots. 
 +
Imagine you want to have a single row with three columns of subplots.  You can either set up three variables, one to access each set of axes, or you can set up a single variable that stores all four axes handles.  In the latter case, the variable storing the axes will be a 1D array of handles, so you will access each one with a single index value.
 +
 
 +
=== Using <code>fig.add_subplot()</code> ===
 +
This version requires that you set each up individually.  This can be useful if each set of axes has different characteristics (for instance, if some have 3D graphs and others do not).
 
<syntaxhighlight lang="Python">
 
<syntaxhighlight lang="Python">
 
fig = plt.figure(num=1, clear=True)
 
fig = plt.figure(num=1, clear=True)
 +
ax1 = fig.add_subplot(1, 3, 1)
 +
ax2 = fig.add_subplot(1, 3, 2)
 +
ax3 = fig.add_subplot(1, 3, 3)
 +
</syntaxhighlight>
  
 +
If you want to plot 10 uniformly distributed random numbers in the second subplot, you would use:
 +
<syntaxhighlight lang="Python">
 +
ax2.plot(np.random.uniform(size=10))
 +
</syntaxhighlight>
 +
=== Using <code>plt.subplots()</code> ===
 +
The first two positional arguments are the number of rows and the number of columns of plots to make.  In this case, the <code>ax</code> variable will be a 1D array with four elements - one per subplot.
 +
<syntaxhighlight lang="Python">
 +
fig, ax = plt.subplots(1, 3, num=1, clear=True)
 
</syntaxhighlight>
 
</syntaxhighlight>
  
=== Single Row or Column of Subplots ===
+
If you want to plot 10 uniformly distributed random numbers in the second subplot, you would use:
If you want to have a single row with four columns of subplots, you can either set up four variables, one for each, or you can set up a single variable that stores all four axes handles.  In the latter case, the variable storing the axes will be a 1D array of handles, so you will access each one with a single index value.
+
<syntaxhighlight lang="Python">
 +
ax[1].plot(np.random.uniform(size=10))
 +
</syntaxhighlight>
 +
=== Using <code>fig.subplots()</code> ===
 +
The first two positional arguments are the number of rows and the number of columns of plots to make.  In this case, the <code>ax</code> variable will be a 1D array with four elements - one per subplot.
 
<syntaxhighlight lang="Python">
 
<syntaxhighlight lang="Python">
 
fig = plt.figure(num=1, clear=True)
 
fig = plt.figure(num=1, clear=True)
ax1 = fig.add_subplot(1, 4, 1)
+
ax = fig.subplots(1, 3)
ax2 = fig.add_subplot(1, 4, 2)
+
</syntaxhighlight>
ax3 = fig.add_subplot(1, 4, 3)
+
 
ax4 = fig.add_subplot(1, 4, 4)
+
If you want to plot 10 uniformly distributed random numbers in the second subplot, you would use:
 +
<syntaxhighlight lang="Python">
 +
ax[1].plot(np.random.uniform(size=10))
 +
</syntaxhighlight>
 +
 
 +
== Single Column of Subplots with Shared x Axis ==
 +
If you have a single column of subplots and you want all of them to have the same $$x$$ axis (basically, to have a single labeled $$x$$ axis showing on the bottom plot), you can use code similar to that below (example shows three rows of plots):
 +
<syntaxhighlight lang="Python">
 +
fig, ax = plt.subplots(3, 1, num=1, clear=True, sharex=True)
 +
# plot things
 +
ax[2].set(xlabel="label")
 
</syntaxhighlight>
 
</syntaxhighlight>
 
or
 
or
 
<syntaxhighlight lang="Python">
 
<syntaxhighlight lang="Python">
fig, ax = plt.subplots(1, 4, num=1, clear=True)
+
fig = plt.figure(num=1, clear=True)
 +
ax = fig.subplots(3, 1, sharex=True)
 +
# plot things
 +
ax[2].set(xlabel="label")
 
</syntaxhighlight>
 
</syntaxhighlight>
  
If you want to plot 10 uniformly distributed random numbers in the second subplot, for the first case you would use:
+
== Multiple Rows and Columns of Subplots ==
 +
[[File:Subplots232.PNG|thumb|Figure window with two rows of three subplots and random numbers in the top middle subplot (after fig.tight_layout())]]
 +
If you want more than one row and more than one column of subplots, you can also create them three different ways.  The <code>add_subplot</code> command will still create one at a time while the two different <code>subplots()</code> commands will now create 2D arrays of handles, meaning you will need to give two index values to access a plot.
 +
Imagine you want to have two rows with three columns of subplots.  You can either set up six variables, one to access each axes, or you can set up a single 2D array that stores all six axes handles.
 +
 
 +
=== Using <code>fig.add_subplot()</code> ===
 +
This version requires that you set each up individually.  This can be useful if each set of axes has different characteristics (for instance, if some have 3D graphs and others do not) but starts to get tedious with a large number of subplots.  Note that the subplot index (the third argument) starts with 1 in the top left corner, counts up left to right, and then goes to the next row (similar to a phone dial).
 +
<syntaxhighlight lang="Python">
 +
fig = plt.figure(num=1, clear=True)
 +
ax1 = fig.add_subplot(2, 3, 1)
 +
ax2 = fig.add_subplot(2, 3, 2)
 +
ax3 = fig.add_subplot(2, 3, 3)
 +
ax4 = fig.add_subplot(2, 3, 4)
 +
ax5 = fig.add_subplot(2, 3, 5)
 +
ax6 = fig.add_subplot(2, 3, 6)
 +
</syntaxhighlight>
 +
 
 +
If you want to plot 10 uniformly distributed random numbers in the second subplot, you would use:
 
<syntaxhighlight lang="Python">
 
<syntaxhighlight lang="Python">
 
ax2.plot(np.random.uniform(size=10))
 
ax2.plot(np.random.uniform(size=10))
 
</syntaxhighlight>
 
</syntaxhighlight>
and for the second case you would use
+
 
 +
=== Using <code>plt.subplots()</code> ===
 +
The first two positional arguments are the number of rows and the number of columns of plots to make.  In this case, the <code>ax</code> variable will be a 2D array with six elements - one per subplot.
 +
<syntaxhighlight lang="Python">
 +
fig, ax = plt.subplots(2, 3, num=1, clear=True)
 +
</syntaxhighlight>
 +
 
 +
If you want to plot 10 uniformly distributed random numbers in the second subplot (top row, middle column), you would use:
 +
<syntaxhighlight lang="Python">
 +
ax[0][1].plot(np.random.uniform(size=10))
 +
</syntaxhighlight>
 +
 
 +
=== Using <code>fig.subplots()</code> ===
 +
The first two positional arguments are the number of rows and the number of columns of plots to make.  In this case, the <code>ax</code> variable will be a 2D array with six elements - one per subplot.
 
<syntaxhighlight lang="Python">
 
<syntaxhighlight lang="Python">
ax[1].plot(np.random.uniform(size=10))
+
fig = plt.figure(num=1, clear=True)
 +
ax = fig.subplots(2, 3)
 +
</syntaxhighlight>
 +
 
 +
If you want to plot 10 uniformly distributed random numbers in the second subplot (top row, middle column), you would use:
 +
<syntaxhighlight lang="Python">
 +
ax[0][1].plot(np.random.uniform(size=10))
 +
</syntaxhighlight>
 +
 
 +
== Mondrian Subplots ==
 +
If you want to build a figure out of a series of rectangular subplots that are of differing sizes, you can do that with the <code>fig.add_subplot()</code> command.  The key is that no two subplots can take up any of the same space. 
 +
Also, each subplot has to be a single location that you can create in an array of subplots - you cannot have a subplot that takes up 2/3rds of the total width of the screen, for instance.
 +
Here are some examples of how this works:
 +
 
 +
=== One Left, Two Right ===
 +
[[File:Subplots1L2R.PNG|thumb|Figure window with large subplot on the left and two smaller ones on the right (after fig.tight_layout())]]
 +
<syntaxhighlight lang="Python">
 +
fig  = plt.figure(num=1, clear=True)
 +
axL  = fig.add_subplot(1, 2, 1)
 +
axR1 = fig.add_subplot(2, 2, 2)
 +
axR2 = fig.add_subplot(2, 2, 4)
 +
</syntaxhighlight>
 +
Notice that for <code>axL</code> the arguments split the whole figure into one row of two columns and picked the first (left one), then for the <code>axR1</code> and <code>axR2</code> lines the arguments split the figure into '''two''' rows of two columns and picked the top right and bottom right.  These three do not overlap at all, so the figure can support these three shapes together.
 +
<br clear=all>
 +
 
 +
=== Two Top, Three Bottom ===
 +
[[File:Subplots2T3B.PNG|thumb|Figure window with two subplots at the top and three on the bottom (cannot use fig.tight_layout())]]
 +
<syntaxhighlight lang="Python">
 +
fig  = plt.figure(num=1, clear=True)
 +
axTL = fig.add_subplot(2, 2, 1)
 +
axTR = fig.add_subplot(2, 2, 2)
 +
axB1 = fig.add_subplot(2, 3, 4)
 +
axB2 = fig.add_subplot(2, 3, 5)
 +
axB3 = fig.add_subplot(2, 3, 6)
 
</syntaxhighlight>
 
</syntaxhighlight>
 +
Note that while this will work, <code>fig.tight_layout()</code> no longer works - the warning is that "number of columns in subplot specifications must be multiples of one another."

Latest revision as of 13:55, 22 September 2020

There are some examples in the main Python:Plotting page showing how to set up subplots within a figure window. Here are some more examples, where each example assumes

import numpy as np
import matplotlib.pyplot as plt

has already run. Note: once you have your subplots created, labeled, and titled, you definitely want to run

fig.tight_layout()

before saving; Python does not do a great job with leaving space for things with subplots.

Basics

There are three different ways to create subplots:

  • fig.add_subplot() needs you to create the figure handle first (probably with fig=plt.figure() and then you can use that figure variable to create one set of axes within an array of axes.
    • The three main arguments for fig.add_subplot(nrows, ncols, index) will establish how many rows and columns you want to break the figure into and then which one of those subplots you want to work with. This method can be especially useful if you want to build a figure with several non-overlapping subplots of different shapes - more on that below.
  • plt.subplots() allows you to create the figure and the axes handles at the same time.
    • The two main arguments for plt.subplots(nrows, ncols) will establish how many rows and columns you want to break the figure into. It will return a figure handle as well an array of axes handles (1D array if creating a single column or row, 2D array if there is more than one row and more than one column). This is useful if you have an array of axes and you are planning to use all of them. You can also give the typical figure keyword arguments such as num=1, clear=True
  • fig.subplots() needs you to create the figure handle first (probably with fig=plt.figure() and then you can use that figure variable to create an entire array of axes.
    • The two main arguments for fig.subplots(nrows, ncols) will establish how many rows and columns you want to break the figure into. It will return an array of axes handles (1D array if creating a single column or row, 2D array if there is more than one row and more than one column). This is useful if you have an array of axes and you are planning to use all of them. The only real difference between this version and the figure based one above is that this one will accept keyword arguments to be passed to the axes rather than to the figure. This will become important when creating 3D plots.

Single Subplot For The Whole Figure

Figure window with one large subplot

Using fig.add_subplot()

fig = plt.figure(num=1, clear=True)
ax = fig.add_subplot(1, 1, 1)

Using plt.subplots()

The default for this command is to make a single subplot, so there is no row or column information necesary.

fig, ax = plt.subplots(num=1, clear=True)

Using fig.subplots()

The default for this command is to make a single subplot, so there is no row or column information necesary.

fig = plt.figure(num=1, clear=True)
ax = fig.subplots()

Single Row or Column of Subplots

Figure window with one row of three subplots and random numbers in the middle subplot (after fig.tight_layout())

The examples below will show how to set up and access a single row or column of subplots. Imagine you want to have a single row with three columns of subplots. You can either set up three variables, one to access each set of axes, or you can set up a single variable that stores all four axes handles. In the latter case, the variable storing the axes will be a 1D array of handles, so you will access each one with a single index value.

Using fig.add_subplot()

This version requires that you set each up individually. This can be useful if each set of axes has different characteristics (for instance, if some have 3D graphs and others do not).

fig = plt.figure(num=1, clear=True)
ax1 = fig.add_subplot(1, 3, 1)
ax2 = fig.add_subplot(1, 3, 2)
ax3 = fig.add_subplot(1, 3, 3)

If you want to plot 10 uniformly distributed random numbers in the second subplot, you would use:

ax2.plot(np.random.uniform(size=10))

Using plt.subplots()

The first two positional arguments are the number of rows and the number of columns of plots to make. In this case, the ax variable will be a 1D array with four elements - one per subplot.

fig, ax = plt.subplots(1, 3, num=1, clear=True)

If you want to plot 10 uniformly distributed random numbers in the second subplot, you would use:

ax[1].plot(np.random.uniform(size=10))

Using fig.subplots()

The first two positional arguments are the number of rows and the number of columns of plots to make. In this case, the ax variable will be a 1D array with four elements - one per subplot.

fig = plt.figure(num=1, clear=True)
ax = fig.subplots(1, 3)

If you want to plot 10 uniformly distributed random numbers in the second subplot, you would use:

ax[1].plot(np.random.uniform(size=10))

Single Column of Subplots with Shared x Axis

If you have a single column of subplots and you want all of them to have the same $$x$$ axis (basically, to have a single labeled $$x$$ axis showing on the bottom plot), you can use code similar to that below (example shows three rows of plots):

fig, ax = plt.subplots(3, 1, num=1, clear=True, sharex=True)
# plot things
ax[2].set(xlabel="label")

or

fig = plt.figure(num=1, clear=True)
ax = fig.subplots(3, 1, sharex=True)
# plot things
ax[2].set(xlabel="label")

Multiple Rows and Columns of Subplots

Figure window with two rows of three subplots and random numbers in the top middle subplot (after fig.tight_layout())

If you want more than one row and more than one column of subplots, you can also create them three different ways. The add_subplot command will still create one at a time while the two different subplots() commands will now create 2D arrays of handles, meaning you will need to give two index values to access a plot. Imagine you want to have two rows with three columns of subplots. You can either set up six variables, one to access each axes, or you can set up a single 2D array that stores all six axes handles.

Using fig.add_subplot()

This version requires that you set each up individually. This can be useful if each set of axes has different characteristics (for instance, if some have 3D graphs and others do not) but starts to get tedious with a large number of subplots. Note that the subplot index (the third argument) starts with 1 in the top left corner, counts up left to right, and then goes to the next row (similar to a phone dial).

fig = plt.figure(num=1, clear=True)
ax1 = fig.add_subplot(2, 3, 1)
ax2 = fig.add_subplot(2, 3, 2)
ax3 = fig.add_subplot(2, 3, 3)
ax4 = fig.add_subplot(2, 3, 4)
ax5 = fig.add_subplot(2, 3, 5)
ax6 = fig.add_subplot(2, 3, 6)

If you want to plot 10 uniformly distributed random numbers in the second subplot, you would use:

ax2.plot(np.random.uniform(size=10))

Using plt.subplots()

The first two positional arguments are the number of rows and the number of columns of plots to make. In this case, the ax variable will be a 2D array with six elements - one per subplot.

fig, ax = plt.subplots(2, 3, num=1, clear=True)

If you want to plot 10 uniformly distributed random numbers in the second subplot (top row, middle column), you would use:

ax[0][1].plot(np.random.uniform(size=10))

Using fig.subplots()

The first two positional arguments are the number of rows and the number of columns of plots to make. In this case, the ax variable will be a 2D array with six elements - one per subplot.

fig = plt.figure(num=1, clear=True)
ax = fig.subplots(2, 3)

If you want to plot 10 uniformly distributed random numbers in the second subplot (top row, middle column), you would use:

ax[0][1].plot(np.random.uniform(size=10))

Mondrian Subplots

If you want to build a figure out of a series of rectangular subplots that are of differing sizes, you can do that with the fig.add_subplot() command. The key is that no two subplots can take up any of the same space. Also, each subplot has to be a single location that you can create in an array of subplots - you cannot have a subplot that takes up 2/3rds of the total width of the screen, for instance. Here are some examples of how this works:

One Left, Two Right

Figure window with large subplot on the left and two smaller ones on the right (after fig.tight_layout())
fig  = plt.figure(num=1, clear=True)
axL  = fig.add_subplot(1, 2, 1)
axR1 = fig.add_subplot(2, 2, 2)
axR2 = fig.add_subplot(2, 2, 4)

Notice that for axL the arguments split the whole figure into one row of two columns and picked the first (left one), then for the axR1 and axR2 lines the arguments split the figure into two rows of two columns and picked the top right and bottom right. These three do not overlap at all, so the figure can support these three shapes together.

Two Top, Three Bottom

Figure window with two subplots at the top and three on the bottom (cannot use fig.tight_layout())
fig  = plt.figure(num=1, clear=True)
axTL = fig.add_subplot(2, 2, 1)
axTR = fig.add_subplot(2, 2, 2)
axB1 = fig.add_subplot(2, 3, 4)
axB2 = fig.add_subplot(2, 3, 5)
axB3 = fig.add_subplot(2, 3, 6)

Note that while this will work, fig.tight_layout() no longer works - the warning is that "number of columns in subplot specifications must be multiples of one another."