Introduction to Plotting with Seaborn#
Learning Objectives#
Be familiar with the use-cases for Seaborn plots
Generate a variety of plots in Seaborn as part of Exploratory Data Analysis (EDA)
Understand the advantages and disadvantages of different types of plots in Seaborn
Recognize and appropriately use strategies for visualizing continuous and categorical data
Visualize and reason about the distributions of data
Introduction#
Hi Hyosub,
Just wanted to say thank you for the recommendation to start using seaborn plotting in python. Wow. That’s it.
-Jonathan
This message was copied verbatim from an email sent to me by a former PhD mentee, and it kind of says it all. Seaborn is a great Python plotting package that is built on top of Matplotlib. That is, it uses Matplotlib “under the hood”, but it offers the user a much simpler API (“Application Programming Interface”; aka, a set of commands) that enable us to generate a variety of great-looking plots that are particularly useful in data science. You can check out Seaborn’s examples gallery to see some of the cool stuff you can do. Seaborn was written by Michael Waskom.
Comparing Seaborn to Matplotlib, you could say that Matplotlib gives you every sharp, hard-to-use tool in the shed, and you have to figure out how to learn them all without chopping off a finger. Fortunately in data science you aren’t likely to lose any fingers, but Seaborn gives you some DIY-friendly power tools to get data science jobs done easily and looking great.
Overview of Seaborn Plotting Functions#
Seaborn is organized into three categories of plot types, as shown below:
Image source: seaborn.pydata.org
You can see that Seaborn’s functions are organized around the different types of data, and questions we typically ask of data in data science: the relplot
function plots relationships between different (continuous) variables; the displot
function plots distributions of data (such as histograms), and thecatplot
function.
Importing the Seaborn package#
By convention, the alias we use when importing Seaborn is sns
.
We’ll also be using some other packages, so we’ll import those packages at the same time.
import seaborn as sns
import pandas as pd
import matplotlib.pyplot as plt
import glob
Plotting Gapminder data#
Mostly we’ll work with long-format data from here on, but since we have worked with the Gapminder data up until now, let’s see how we can generate a plot similar to what we produced in the Introduction to Plotting with Matplotlib lesson.
First we’ll load the Oceania GDP data, and strip the leading text off column names so that they are just the years:
df = pd.read_csv('data/gapminder_gdp_oceania.csv', index_col='country')
df.columns = df.columns.str.strip('gdpPercap_')
In the previous lesson, we plotted GDP for each country as a function of year. That is, year is on the x axis, GDP is on the y axis, and there are separate lines for each country. As discussed previously, in this dataset, year is a continuous variable, as is GDP, but country is a categorical variable.
From the figure of Seaborn plot types above, you can see that relplot()
is the function to use for relational plots — that is, when you want to show the relationships between two continuous variables (year and GDP in this case).
sns.relplot(data=df.T, kind='line') # For wide-format data we need to transpose rows & columns
plt.show() # Always good practice to have this as the last line in a plotting cell
Compared to plotting the data in Matplotlib (below), the results are similar, but Seaborn adds some aesthetic touches by default, such as reducing the number of irrelevant frame lines, putting the legend outside the plot, and making the line for the second condition dashed (which aids in discriminating lines, especially if there are several of them, or if someone has vision problems). As we’ll see, Seaborn isn’t just another pretty face — it also makes a wide variety of plots easy to make with a consistent and relatively intuitive syntax.
# Matplotlib version of the same plot
df.T.plot()
plt.show()
Plotting Long-Format Data#
Let’s move to working with some data of the sort we might find in an experiment exploring the psychology of motor control. Imagine an experiment in which each trial involves presenting a string of letters in the center of the screen that is flanked on the left and right by two equal-sized circular targets. Participants are instructed to make a reach to one of the two targets as quickly as possible depending on whether the string is a real word or not. (You’ve counter-balanced the blocks such that the word and non-word targets show up an equal amount of times on the left and right.) You may be running this experiment because you have some theory regarding sensorimotor confidence as it relates to higher-level properties of a visual stimulus. For instance, people may reach faster and more accurately when they are more certain of their judgments, as in the case of familiar words. The dependent variable in the experiment is reaction time (RT) — the amount of time that passed between the onset of the visual target and the start of the reaching movement.
While this is a made-up example, this type of lexical decision task is frequently used in psycholinguistics to explore porperties of real words, such as tehir length or how frequently they occur in normal usage. More broadly, these types of experiments in which sensorimotor confidence is assessed by manipulating the visual stimulus and measuring some aspect of kinematics or kinetics is common in the psychological and neuroscientific study of motor control (for example). Here, we’ll assume that people respond with a faster RT when an actual word appears.
We’ve created some data, using a random number generator, to use in our plotting examples. The data were deliberately created with certain properties, so that the expected difference between RTs is reliable. We have data from 5 hypothetical participants, stored in separate data files in the data
folder. The naming format for the files is ldt_sXX_data.csv
, where XX
stands for the subject ID number (two digits).
First we’ll load the data from the files, practicing our list comprehension, and storing the result in a pandas DataFrame called df
. We need to include the ignore_index=True
kwarg when concatenating the data, because otherwise there will be multiple copies of the same index in the merged DataFrame (since each data file that is read will start at index 0).
df = pd.concat([pd.read_csv(f) for f in glob.glob('data/rl_s??_data.csv')],
ignore_index=True)
---------------------------------------------------------------------------
ValueError Traceback (most recent call last)
Cell In[5], line 1
----> 1 df = pd.concat([pd.read_csv(f) for f in glob.glob('data/rl_s??_data.csv')],
2 ignore_index=True)
File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/pandas/util/_decorators.py:331, in deprecate_nonkeyword_arguments.<locals>.decorate.<locals>.wrapper(*args, **kwargs)
325 if len(args) > num_allow_args:
326 warnings.warn(
327 msg.format(arguments=_format_argument_list(allow_args)),
328 FutureWarning,
329 stacklevel=find_stack_level(),
330 )
--> 331 return func(*args, **kwargs)
File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/pandas/core/reshape/concat.py:368, in concat(objs, axis, join, ignore_index, keys, levels, names, verify_integrity, sort, copy)
146 @deprecate_nonkeyword_arguments(version=None, allowed_args=["objs"])
147 def concat(
148 objs: Iterable[NDFrame] | Mapping[HashableT, NDFrame],
(...)
157 copy: bool = True,
158 ) -> DataFrame | Series:
159 """
160 Concatenate pandas objects along a particular axis.
161
(...)
366 1 3 4
367 """
--> 368 op = _Concatenator(
369 objs,
370 axis=axis,
371 ignore_index=ignore_index,
372 join=join,
373 keys=keys,
374 levels=levels,
375 names=names,
376 verify_integrity=verify_integrity,
377 copy=copy,
378 sort=sort,
379 )
381 return op.get_result()
File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/pandas/core/reshape/concat.py:425, in _Concatenator.__init__(self, objs, axis, join, keys, levels, names, ignore_index, verify_integrity, copy, sort)
422 objs = list(objs)
424 if len(objs) == 0:
--> 425 raise ValueError("No objects to concatenate")
427 if keys is None:
428 objs = list(com.not_none(*objs))
ValueError: No objects to concatenate
Let’s look at a sample of the data:
df.sample(10)
ID | condition | RT | |
---|---|---|---|
212 | ldt_s05 | word | 0.594286 |
218 | ldt_s05 | word | 0.647573 |
132 | ldt_s02 | nonword | 0.596347 |
150 | ldt_s04 | word | 0.797847 |
66 | ldt_s03 | word | 0.608214 |
67 | ldt_s03 | word | 0.747991 |
107 | ldt_s02 | word | 0.610677 |
225 | ldt_s05 | nonword | 0.815493 |
3 | ldt_s01 | word | 0.497075 |
96 | ldt_s03 | nonword | 0.936968 |
We can see that there are three columns:
ID
is the unique participant identifiercondition
is the type of word (real or gibberish)RT
is reaction time, measured in seconds
Descriptive statistics#
We can use the pandas .describe()
method to examine some basic properties of our data:
df.describe()
RT | |
---|---|
count | 250.000000 |
mean | 0.710869 |
std | 0.159536 |
min | 0.239836 |
25% | 0.597494 |
50% | 0.708552 |
75% | 0.818883 |
max | 1.194955 |
Since RT
is the only continuous variable in this DataFrame, we only get descriptive statistics for this variable.
Examining distributions with histograms#
Histograms are a type of plot that allows us to view distributions of values. The x axis of a histogram represents the values in the data, and the y axis shows the count of how many trials had that value on the x axis. Values on the x axis are grouped into bins — ranges of values.
In Seaborn, we use the displot()
function to plot distributions. Histograms are one type of displot
, but they are the default, so we can just run the following:
sns.displot(data=df, x='RT')
plt.show()
Comparing the range of values on the x axis with the min
and max
values from df.describe()
, we see that the entire range of values is represented. The range of RT values have been binned into 11 categories, in order to draw the 11 bars in the histogram. We can see that the most common values of RT are between approximately 0.5 – 0.6 s, which corresponds to the median (50%
) value shown by pd.describe()
. We can also see that the data are slightly skewed — there is a wider range of values to the right than the left of the peak. Visualizing distributions is thus a useful way of examining your data, and you can likely glean more information from this plot, more quickly and easily, than from the descriptive statistics table above.
Tweaking Seaborn plots#
We can adjust the parameters of the histogram with a few kwargs, including adjusting technical aspects of how the data are treated to create the plot (like the number of bins), and aesthetic aspects of how it is represented visually (like the colour).
sns.displot(data=df, x='RT',
bins=25,
color='red')
plt.show()
In the above plot we increased the number of bins from the default to 25. You can see that the histogram looks less smooth than it did with fewer bins — increasing the number of bins decreases the number of data points in the bins, which in turn results in greater random variability.
By default (when you don’t specify a number of bins), the software makes a decision based on characteristics of the data, which usually results in sensible bin sizes.
What to plot on the histogram y axis#
By default, on the y axis Seaborn plots the count of the number of data points in each bin. This isn’t always the best measure. It’s fine if you just want to look at how values are distributed over the variable on the x axis, but if you want to compare histograms, the scale can be deceiving if the total number of data points in different plots is different. For this reason, we can scale the data in different ways.
Particularly useful is to normalize the data. For example, you can use stat='probability'
to make the sum of every histogram bar heights equal to 1. This means that the y axis now reflects the proportion (percentage) of the data in each bin.
sns.displot(data=df, x='RT',
stat='probability')
plt.show()
Kernel density estimate#
An alternative form of ditribution plot in Seaborn is the kernel density estimate (kde
; often more simply called a “density plot”). This is like a histogram, but plots smooth curves over the range of values rather than bins as bars. This is another kind of normalization, but rather than making the heights of the bars sum to 1, the density estimate makes the area under the smoothed curve equal to 1.
Note however that while histograms represent, literally, the number of data points in each bin, the KDEs are estimates of the true distribution, and are effectively smoothed and thus slightly less accurate representations of the data you have. On the other hand, KDEs are models that predict what the true distirbution would be, if more data was collected.
In practice, the difference between histograms and KDEs is relatively small in terms of how one might interpret what they show, but this distinction between literal presentations of data and estimates meant to generalize beyond the data is an important concept that cuts through all of data science.
To override the default kind of plot generated by displot()
(a histogram), we add a kind
kwarg:
sns.displot(kind='kde',
data=df, x='RT')
plt.show()
Density plots arguably make more sense for continuous variables like reaction time, since they represent continuous variables in a more continuous fashion.
Viewing the data by condition#
As noted, the experiment had essentially two conditions, words and non-words. So far we’ve looked at the data overall, combining RTs from those two conditions. But it’s easy to look at the data broken down by condition.
Firstly, we can use pandas’ .groupby()
method to get descriptive statistics:
df.groupby('condition').describe()
RT | ||||||||
---|---|---|---|---|---|---|---|---|
count | mean | std | min | 25% | 50% | 75% | max | |
condition | ||||||||
nonword | 125.0 | 0.797717 | 0.132815 | 0.340474 | 0.716140 | 0.799589 | 0.879407 | 1.133205 |
word | 125.0 | 0.611654 | 0.133936 | 0.242320 | 0.530431 | 0.612513 | 0.696094 | 1.042468 |
We can add a hue
kwarg to sns.displot()
to separate and colour-code the conditions in a histogram:
sns.displot(data=df, x='RT', hue='condition')
plt.show()
The grey area in the plot represents overlap.
You can see that Seaborn does lots of nice things to the plot automatically, that make it easy to interpret: there are meaningful labels on both axes, and a legend telling us what each colour represents.
Colour Choice#
Seaborn has a default palette, or set of colours, that it uses in sequence. You can see this with: sns.color_palette()
sns.color_palette()
So the first category plotted will be blue, the next orange, then green, etc..
Seaborn provides a number of different colour palettes, although the default is a good choice as the hues are designed to be easily distinguished from each other, even for people with different forms of colourblindness.
One alternative is the Paired
palette (note the capital P
), which is designed for when your data are naturally organized in pairs of categories (e.g., imagine if we had the conditions reward and non-reward, but we wanted to compare these in two different groups).
sns.color_palette('Paired')
sns.color_palette('Paired')
Let’s re-plot our 2-condition histogram with the Paired
palette:
sns.displot(data=df, x='RT', hue='condition', palette='Paired')
plt.show()
Some palettes are better for some things than others, though — for example, in the histogram above, the overlap is not very distinct from the other shades of blue.
We can use the hue
kwarg with other Seaborn plots as well, like kde
:
sns.displot(kind='kde',
data=df,
x='RT', hue='condition',
palette='colorblind')
plt.show()
Categorical Plots#
Histograms and KDEs are used to visualize the distribution of continuous variables, like RT in our example data set. We can also plot categorical variables, by drawing different histograms/KDEs for each category, like for reward vs. nonreward in the plots above. These plots are most useful for visualizing the shape of the distributions of the data. This is an important first step in working with data, because some ways of treating data (including statistical tests like t-tests and ANOVAs) assume that the data are normally distributed, (the classic “bell curve”), and may not work as well or as accurately on data that have very different distributions. On the other hand, in such cases other approaches to analyzing the data could be more appropriate.
That said, once we understand the distribution of the data, we might want to focus more on assessing whether there are differences between different categories (such as experimental conditions, or groups). This is where Seaborn’s catplot()
family of plots comes into play.
sns.catplot()
has 6 different plotting options, as shown in the figure at the top of this lesson. All of them allow us to view the distribution of the data, but in different ways, and often with a focus more on showing the mean, or median. This allows us to more easily compare differences between conditions, while considering the extent to which their distributions overlap or not. We’ll show a few of them in this lesson.
Box plots#
Box plots (sometimes called box-and-whisker plots) are a classic and widely-used approach to comparing between categorical variables. The “box” shows the interquartile range (IQR) — the range between the 25th and 75th percentiles of the data. In other words, half of the data falls within this box. The line inside the box represents the median, or the value exactly in the middle of all the values in the data set. The whiskers (lines extending out from the box) end at ±1.5 x the IQR (more on that in a minute), and then any points plotted outside the whiskers are considered outliers. The idea behind this is that the whiskers end at what might reasonably be predicted to reflect the minimum and maximum values in the distribution, and any outliers are likely to be inaccurate measurements or other types of “noise”. Below is a figure showing how the box plot maps on to the normal distribution, which is a helpful way to think about interpreting box plots.
Image from Jhguch and shared under the Creative Commons Attribution-Share Alike 2.5 Generic license
Drawing box plots with Seaborn#
We use sns.catplot()
with the kind='box'
kwarg:
sns.catplot(kind='box',
data=df,
x='condition', y='RT')
plt.show()
We can see from this that the distributions of the two conditions appear quite different, and indeed their IQRs do not overlap at all. This gives us good confidence that RTs are probably different in the two conditions. You can also see that there are a few outliers (the diamonds outside the whiskers).
Although we lose some of the details of the distribution in going from a histogram to a box plot, we gain some simplicity, and it’s easier to assess how different our different conditions are.
Violin Plots#
A violin plot is in some ways the “best of both worlds”, combining a box plot with a KDE. Note that the syntax of this plot is identical to that for the box plot above, except the kind
kwarg.
sns.catplot(kind='violin',
data=df,
x='condition', y='RT')
plt.show()
The box plot has been made more narrow, but the thinker black line inside each “violin” is the IQR, and the thinner lines the whiskers. The coloured, shaded area is the KDE. For aesthetic reasons, the KDE is shown on both sides of the box plot (although if you have different sub-categories, you could plot different distributions on each side).
In adding the distribution plot to the box plot, you gain detail, but perhaps at the cost of the simplicity of a box plot, and its usefulness in making inferences about differences between conditions.
Strip and Swarm plots#
These are quite similar, and both plot the actual individual data points.
sns.catplot(kind='strip',
data=df,
x='condition', y='RT')
plt.show()
Swarm plots shift data points laterally when they are dense, reducing overlap and making it easier to see where the data are more dense.
sns.catplot(kind='swarm',
data=df,
x='condition', y='RT')
plt.show()
Bar plots#
These are classic ways of representing data, that focus more on showing and comparing means for different conditions, than on the distribution. The “error bars” (black bars at the top of each bar below) represent 95% confidence intervals, which are a way of representing variance that is better-suited for comparing between means, than for assessing the shape or breadth of the distribution. You can see below that the bar plots lose a lot of the information about the distributions of data, relative to box plots or any of the other plots above.
Another limitation of bar plots is that they are “anchored” at zero. For some data this makes sense, especially if the data values relative to zero are meaningful. But in many cases — including our current RT data — this doesn’t make a lot of sense. An RT of zero would be considered invalid data, since the human nervous system can’t execute a response to a stimulus at the speed of light. So our experiment is not really concerned with how much slower than zero a person responded. As well, if your data values are far from zero, and the differences between conditions are relatively small, then bar plots make make the differences look meaninglessly small, because the y axis has to span the entire range from zero to the maximum value in the plot.
sns.catplot(kind='bar',
data=df,
x='condition', y='RT')
plt.show()
Point and Line Plots#
Point plots generally are a better choice when you want to focus on comparing means between categories, because they simply show the means as points, with 95% confidence intervals. The scale of the plots will be more appropriate, without the visual bias that bar plots create by filling the range from zero to the mean of each condition with a big block of colour.
By default, Seaborn draws lines connecting the points in a plot. This is only really appropriate if your categories are “connected” in some meaningful way, such as different time points (e.g., pre- vs. post-treatment for some intervention), or when you have multiple categorical variables.
sns.catplot(kind='point',
data=df,
x='condition', y='RT')
plt.show()
We can remove the lines with join=False
, and colour-code the different levels of condition with hue='condition'
sns.catplot(kind='point', join=False, hue='condition',
data=df,
x='condition', y='RT')
plt.show()
Summary#
Seaborn makes it easy to generate plots useful for EDA
relplot()
is used for visualizing relationships between two continuous variablesdisplot()
is used for visualizing distributions of data, as histograms or kernel density estimates (KDEs)catplot()
is used for making comparisons between different levels of categorical data, such as different experimental conditions or groupsDifferent kinds of
catplot()
plots vary in how much they communicate information about the distributions of data in each category, versus on differences between the means of the categoriesIn EDA, it’s often useful to visualize the data many ways to really understand it. Even still, it’s important to have an understanding of the strengths and limitations of different kinds of plots
Seaborn makes it relatively easy to generate plots and adjust properties such as colour and font size, and as well as making plots accessible to the widest range of viewers
This section was adapted from Aaron J. Newman’s Data Science for Psychology and Neuroscience - in Python.