Working with Repeated Measures Data#


Learning Objectives#

  • Learn how to use split-apply-combine to average repeated measures data appropriately

  • Learn how to properly visualize repeated measures data with Seaborn


Introduction#

In the previous chapter on Seaborn plots, we worked with some simulated reaction time (RT) data from 5 individual participants. We showed various Seaborn plots that allowed us to examine the variance in the data sets. However, for the sake of focusing on the different types of Seaborn plots and what they show us, we glossed over one really important aspect of the data — its nesting structure, and in particular the fact that we have repeated measures. Repeated measures data are very common in cognitive psychology and neuroscience. In fact, while they are often treated as a “special case” in statistical and data science texts, in this context we focus on them because most of the data we will encounter later in this course include repeated measures — and it’s very easy to treat the data incorrectly and draw incorrect conclusions if you’re not used to working with repeated measures data.

Recall that in the Thinking About Data for Plotting lesson, we learned that variance within individual units of measurement (such as individual participants) is not the same as the variance between participants. Furthermore, in a typical experiment our interest is in examining the average effects across individuals, and the range of differences between individuals. When we have nested data, such as many individual trials for each participant, this is because we recognize that any individual measurement may not be accurate, and so by averaging over many measurements we are likely to get a more accurate estimate of the true measurement for each individual. However, if we average the trial-to-trial variance of each individual (which is typically low, within the individual) across individuals, we lose the true inter-individual differences.

For this reason, when plotting nested data it is usually appropriate to first average within each individual (for each condition), and then compute and visualize measures of the true inter-individual variability. Seaborn does not do this by default, and so the plots we generated in the previous lesson were actually inappropriate for assessing whether there was a true experimental effect. In this lesson we’ll see how to use pandas .groupby() to use Seaborn appropriately with nested data.

About the data#

First, let’s load in a new data set (flanker_rt_data.csv). This is real data from a flanker experiment (similar to that described in the section on spreadsheets). In this experiment participants had to press either the left or right arrow key, to indicate whether an arrow shown on the screen is pointing left or right, respectively. However, the catch is that the centre arrow is “flanked” by two other arrows on each side; these can be pointing the same way as the target arrow (congruent):

flanker_congruent

or in the opposite direction (incongruent)

flanker_incongruent

We expect that the incongruent condition would be associated with slower RTs, because the flanking arrows create some visual confusion and response competition (cognitive interference; this process is not unlike that underlying the famous Stroop experiment, which you are likely familiar with). In order to confirm this hypothesis, we would want to get an estimate of the RT for each condition, from a representative sample of human participants. We need a number of participants because we know that there is variability in the average RT from person to person (between-subject variability). Here we have 27 participants.

To estimate the average RT for each condition for an individual, we would want to present many trials of each condition, in random order. We do this because there is always some trial-to-trial variability in measuring human RTs (within-subject variability). In this study each participant completed 40 trials in each condition. Thus we have repeated measures of RT for each participant, in each experimental condition.

This experiment also had a third condition, neutral. On neutral trials, only the centre arrow was presented, without flankers. This condition was included to control for the fact that 5 arrows all pointing in the same direction might facilitate (speed up) responses relative to a single arrow — there might be additive, beneficial effects of congruent flankers. Comparing both the congruent and incongruent conditions to the neutral condition allows us to estimate the relative cost or benefit of flankers that are incongruent or congruent with the direction of the centre arrow. To ensure that there were as many neutral (single-arrow) as flanker (five-arrow) trials, 80 neutral trials were included per participant.

So we have data from 27 participants, and approximately 160 trials per participant in total. The nesting structure of this data set is thus trials (n=160) nested within conditions (n=3) nested within participants (n=27).

Load the data#

First we’ll import the packages we’ll need, read in the data, and view a sample of 10 rows of the DataFrame:

import pandas as pd
import seaborn as sns
import matplotlib.pyplot as plt

df = pd.read_csv('data/flanker_rt_data.csv')

df.sample(10)
participant block trial flankers rt
626 s22 5 19 neutral 0.449923
4263 s15 5 7 congruent 0.569543
2463 s14 3 22 neutral 0.521081
3589 s20 4 2 congruent 0.374114
2048 s12 5 21 neutral 0.532976
453 s16 5 6 congruent 0.478145
3441 s3 4 14 neutral 0.425251
1810 s7 3 1 congruent 0.668527
2335 s26 4 21 congruent 0.489032
328 s16 1 10 congruent 0.434179

Examine the data without considering repeated measures#

First we’ll get some descriptive statistics for RT in each condition. We do this using pandas .groupby() method to group by flanker condition, and then we select the rt column for further processing by putting it in square brackets (which is pandas’ way of selecting columns). Finally we chain all this with the .describe() method:

df.groupby('flankers')['rt'].describe()
count mean std min 25% 50% 75% max
flankers
congruent 1076.0 0.470568 0.098868 0.009837 0.401727 0.449814 0.521514 0.953989
incongruent 1064.0 0.515996 0.107581 0.061839 0.449544 0.504629 0.573016 0.965726
neutral 2149.0 0.498310 0.098934 0.031539 0.433521 0.485642 0.552863 0.980535

Note the count (total number of data points), standard deviation (std) as well as the min and max values for each condition.

Let’s look at a box plot of values in each condition:

sns.catplot(kind='box',
           data=df,
           x='flankers', y='rt')
plt.show()
../_images/ec8135b5df913854ae9e832c478a7f2c2065f9474641e2bfa594ba19c839b67b.png

The data show the skew that is expected of RT data — there are fundamental human performance limits on how fast a person can respond (short RTs), but the upper limit of RTs is theoretically unlimited. In the present experiment, participants actually only had a 1 s window in which to respond, so the RTs are all less than 1. Nevertheless, the data are skewed because there is a wider range of RTs above the median than below.

Another thing we can see about the distribution of values is that there are many outliers (individual points in the box plot) at the long end of the RT range, but few at the short end. This is, again, a consequence of the skewed nature of the data. However, notably some of the outliers on the short end of the distributions are suspicious: human RTs in very simple tasks (like pressing a button any time a light comes on) are rarely shorter than 200 ms, yet we have a few RT values below that. So our box plots suggest that we may have some anomalous data, perhaps from trials on which someone pressed a button accidentally prior to processing the actual direction of the arrow.

Finally, the box plots suggest that RTs are slowest in the incongruent condition and fastest in the congruent condition — consistent with our predictions based on past studies of the flanker effect. RTs for the neutral condition are in between the other two conditions. This suggests that perhaps both facilitation effects in the congruent condition, and interference effects in the incongruent condition, are present in the data. However, it’s hard to know how large or believable those differences are, because the distributions overlap quite a bit between the three conditions.

Stingray plot#

Let’s look at a couple of other ways of viewing the data that we learned about in the Seaborn lesson. First, we’ll use a violin (stingray) plot to see both the KDE and box plot:

sns.catplot(kind='violin',
           data=df,
           x='flankers', y='rt')
plt.show()
../_images/db6cfe62433c43b6cfc42a7affd71d0dc577f987d86d845b0b579ecb7b2d1494.png

The skewed nature of the data are again evident, but we can see that not only are the median values for the three conditions different, but the bulk of values in each distribution of RTs also appear to be rather different. That is, the “humps” in the middle of each distribution are relatively non-overlapping, although neutral and incongruent are fairly similar.

Strip plot#

Next let’s look at a strip plot of the data, which plots each data point separately:

sns.catplot(kind='strip',
           data=df,
           x='flankers', y='rt')
plt.show()
../_images/8c596eed370bbcc6fc68ea67397ebc5cd330a5ed71c7d852a32451745552fc7a.png

This is really less informative than the above plots, because there are so many data points! Mostly we just see big patches of the plots for each condition with lots of individual values.

However, the strip plots do show us something interesting that wasn’t evident in the previous plots: the very fast outlier data points are quite anomalous, even compared to the very slow outliers. That is, there are few “fast” data points, and they are generally quite far from the rest of the distribution of RTs. In contrast, the slower data points are more plentiful, and seem to form a rather continuous set of values, albeit more sparse than the points closer to the centres of the distributions. This suggests that the slower data points might be quite believable as real data — mosly people make faster responses, but sometimes they’re a bit slower. In contrast, the strip plot gives us greater confidence that the very fast RTs should be eliminated from analysis, because they are not just slightly-faster reponses than the next-fastest responses.

Point plot#

Finally, let’s look at a point plot. Recall that point plots show the mean for each condition, along with 95% confidence intervals (CIs) as the measure of variance. CIs are much narrower than the whiskers in box plots, because they are meant to represent our confidence in the estimate of the mean, rather than showing the overall distribution of values. In general, we can interpret conditions with non-overlapping 95% CIs as being “believably different” from each other, and likely to be (statistically) significantly different as well.

sns.catplot(kind='point', join=False,
           data=df,
           x='flankers', y='rt', hue='flankers')
plt.show()
../_images/bd82539081a8c537d824d9a6edb0bed24d32efe74c53569a4bc9c4a50ac428cc.png

We can see that the CIs are quite far apart, suggesting that there are very large differences between RTs for each pair of conditions.

While any researcher would be thrilled to see such a big difference and small confidence intervals, this plot is misleading, because we have not distinguished within-participant variance from between-participant variance.

Examine individual participant data#

Now, let’s use split-apply-combine to average the RTs within each individual participant (and within each condition within individuals) prior to the group-level descriptive statistics. We can do this in one line of code by chaining, combining the following steps:

  • group by both participant ID and condition (split). Note that the argument to .groupby() is a list containing the two columns we want to group by

  • select the rt column: ["rt"]

  • apply descriptive statistics and combine the results in a table: .describe()

df.groupby(['participant', 'flankers'])['rt'].describe()
count mean std min 25% 50% 75% max
participant flankers
s1 congruent 39.0 0.455259 0.062264 0.345359 0.405289 0.445225 0.495277 0.597253
incongruent 40.0 0.471838 0.076799 0.373203 0.422353 0.463064 0.506307 0.777096
neutral 80.0 0.476104 0.071508 0.357297 0.429230 0.467111 0.514231 0.783683
s10 congruent 40.0 0.471231 0.078493 0.329720 0.403671 0.463479 0.521448 0.689298
incongruent 40.0 0.499031 0.068776 0.301630 0.467616 0.499406 0.553274 0.637271
... ... ... ... ... ... ... ... ... ...
s8 incongruent 40.0 0.494878 0.133459 0.061839 0.453581 0.501628 0.541888 0.881690
neutral 80.0 0.460459 0.123738 0.031539 0.412689 0.478907 0.525675 0.644778
s9 congruent 40.0 0.431829 0.074206 0.316626 0.382794 0.413070 0.468954 0.646603
incongruent 39.0 0.437926 0.083530 0.301302 0.381454 0.421338 0.487607 0.705617
neutral 80.0 0.432177 0.061369 0.304624 0.389656 0.431551 0.462674 0.609026

81 rows × 8 columns

This is a lot of numbers to look at! Indeed, so many that pandas doesn’t show us all the rows. From what is shown, we can see that there is certainly variance in mean RTs for each individual, and in variance (as measured by standard deviation), but it’s hard to get a real sense of how much variability there is, or whether there are any participants whose data may be anomalous relative to most others.

Visualizing nested data with Seaborn#

Seaborn makes it easy to separate data by participants in visualization. For example, we can generate box plots as we did above, but put participant on the x axis, and use the hue kwarg to colour-code flanker condition. Because there are lots of individual participants, we add the aspect=3 kwarg to tell Seaborn to make the plot 3x as wide as it is tall.

sns.catplot(kind='box',
           data=df,
           x='participant', y='rt', hue='flankers',
           aspect=3
           )
plt.show()
../_images/38c9efddb93f24da26c538385f33d00af6946dc9ae92cb0655180e9dbc03fa3a.png

Another way to visualize so many different box plots is with Seaborn’s col and col_wrap kwargs. On its own, col= will plot whatever variable you give it in separate columns — in our case, participants (rather than as different locations on the x axis, as shown above). Adding the col_wrap kwarg is necessary when the number of categories passed to col is large, as in our 27 participants. This kwarg tells Seaborn how many columns to put on one row before wrapping around to a new row.

One last trick we will use is to set Seaborn’s context to talk, to make the font size easier to read when we have so many plots. We’ll set it back to paper after showing the plot, so our future plots aren’t affected.

sns.set_context('talk')

sns.catplot(kind='box',
           data=df,
           x='flankers', y='rt', col='participant', col_wrap=6,
           )
plt.show()
sns.set_context('paper')
../_images/c4a999fc87be6164e85d97d69ee81ac8a791188bc2bf221f9bd15c3b20d8689d.png

This plot shows us that, indeed, there is variability among participants in both the median RTs and the variance (as indicated by the length of the whiskers, and the number and range of outlier points). Encouragingly, however, each individual seems to show the expected effect, of longer RTs for incongruent than congruent flakers. It also shows us that those anomalously fast RTs we saw above are contributed only by a couple of participants, again leading us to think that these are not believable data points.

Next we’ll generate a strip plot, in which we colour-code the points according to which participant they are from:

sns.catplot(kind='strip',
           data=df,
           x='flankers', y='rt', hue='participant')
plt.show()
../_images/63ea2abd77f7e53087e8520323bb6db4bdc4bdb3ca139275be63330e7f433303.png

This is not a lot more useful than the strip plot above, because Seaborn draws dots for each participant over those of previous participants. So we see lots of pink in the plot above, because that is the colour used for the last participant plotted. This is addressed by using a swarm plot (which will take a long time to execute if you try to run this, due to the large number of data points it’s trying to plot):

sns.catplot(kind='swarm',
           data=df,
           x='flankers', y='rt', hue='participant')
plt.show()
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 55.9% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 73.5% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 50.8% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 72.8% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 83.8% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 69.3% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 69.1% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 81.6% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 65.3% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 73.0% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 83.8% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 69.5% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
/Users/hyosubkim/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3544: UserWarning: 73.0% of the points cannot be placed; you may want to decrease the size of the markers or use stripplot.
  warnings.warn(msg, UserWarning)
---------------------------------------------------------------------------
KeyboardInterrupt                         Traceback (most recent call last)
Cell In[11], line 4
      1 sns.catplot(kind='swarm',
      2            data=df,
      3            x='flankers', y='rt', hue='participant')
----> 4 plt.show()

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/pyplot.py:387, in show(*args, **kwargs)
    343 """
    344 Display all open figures.
    345 
   (...)
    384 explicitly there.
    385 """
    386 _warn_if_gui_out_of_main_thread()
--> 387 return _get_backend_mod().show(*args, **kwargs)

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib_inline/backend_inline.py:90, in show(close, block)
     88 try:
     89     for figure_manager in Gcf.get_all_fig_managers():
---> 90         display(
     91             figure_manager.canvas.figure,
     92             metadata=_fetch_figure_metadata(figure_manager.canvas.figure)
     93         )
     94 finally:
     95     show._to_draw = []

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/IPython/core/display_functions.py:298, in display(include, exclude, metadata, transient, display_id, raw, clear, *objs, **kwargs)
    296     publish_display_data(data=obj, metadata=metadata, **kwargs)
    297 else:
--> 298     format_dict, md_dict = format(obj, include=include, exclude=exclude)
    299     if not format_dict:
    300         # nothing to display (e.g. _ipython_display_ took over)
    301         continue

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/IPython/core/formatters.py:177, in DisplayFormatter.format(self, obj, include, exclude)
    175 md = None
    176 try:
--> 177     data = formatter(obj)
    178 except:
    179     # FIXME: log the exception
    180     raise

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/decorator.py:232, in decorate.<locals>.fun(*args, **kw)
    230 if not kwsyntax:
    231     args, kw = fix(args, kw, sig)
--> 232 return caller(func, *(extras + args), **kw)

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/IPython/core/formatters.py:221, in catch_format_error(method, self, *args, **kwargs)
    219 """show traceback on failed format call"""
    220 try:
--> 221     r = method(self, *args, **kwargs)
    222 except NotImplementedError:
    223     # don't warn on NotImplementedErrors
    224     return self._check_return(None, args[0])

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/IPython/core/formatters.py:338, in BaseFormatter.__call__(self, obj)
    336     pass
    337 else:
--> 338     return printer(obj)
    339 # Finally look for special method names
    340 method = get_real_method(obj, self.print_method)

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/IPython/core/pylabtools.py:152, in print_figure(fig, fmt, bbox_inches, base64, **kwargs)
    149     from matplotlib.backend_bases import FigureCanvasBase
    150     FigureCanvasBase(fig)
--> 152 fig.canvas.print_figure(bytes_io, **kw)
    153 data = bytes_io.getvalue()
    154 if fmt == 'svg':

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/backend_bases.py:2319, in FigureCanvasBase.print_figure(self, filename, dpi, facecolor, edgecolor, orientation, format, bbox_inches, pad_inches, bbox_extra_artists, backend, **kwargs)
   2315 try:
   2316     # _get_renderer may change the figure dpi (as vector formats
   2317     # force the figure dpi to 72), so we need to set it again here.
   2318     with cbook._setattr_cm(self.figure, dpi=dpi):
-> 2319         result = print_method(
   2320             filename,
   2321             facecolor=facecolor,
   2322             edgecolor=edgecolor,
   2323             orientation=orientation,
   2324             bbox_inches_restore=_bbox_inches_restore,
   2325             **kwargs)
   2326 finally:
   2327     if bbox_inches and restore_bbox:

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/backend_bases.py:1648, in _check_savefig_extra_args.<locals>.wrapper(*args, **kwargs)
   1640     _api.warn_deprecated(
   1641         '3.3', name=name, removal='3.6',
   1642         message='%(name)s() got unexpected keyword argument "'
   1643                 + arg + '" which is no longer supported as of '
   1644                 '%(since)s and will become an error '
   1645                 '%(removal)s')
   1646     kwargs.pop(arg)
-> 1648 return func(*args, **kwargs)

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/_api/deprecation.py:415, in delete_parameter.<locals>.wrapper(*inner_args, **inner_kwargs)
    405     deprecation_addendum = (
    406         f"If any parameter follows {name!r}, they should be passed as "
    407         f"keyword, not positionally.")
    408     warn_deprecated(
    409         since,
    410         name=repr(name),
   (...)
    413                  else deprecation_addendum,
    414         **kwargs)
--> 415 return func(*inner_args, **inner_kwargs)

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/backends/backend_agg.py:540, in FigureCanvasAgg.print_png(self, filename_or_obj, metadata, pil_kwargs, *args)
    490 @_check_savefig_extra_args
    491 @_api.delete_parameter("3.5", "args")
    492 def print_png(self, filename_or_obj, *args,
    493               metadata=None, pil_kwargs=None):
    494     """
    495     Write the figure to a PNG file.
    496 
   (...)
    538         *metadata*, including the default 'Software' key.
    539     """
--> 540     FigureCanvasAgg.draw(self)
    541     mpl.image.imsave(
    542         filename_or_obj, self.buffer_rgba(), format="png", origin="upper",
    543         dpi=self.figure.dpi, metadata=metadata, pil_kwargs=pil_kwargs)

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/backends/backend_agg.py:436, in FigureCanvasAgg.draw(self)
    432 # Acquire a lock on the shared font cache.
    433 with RendererAgg.lock, \
    434      (self.toolbar._wait_cursor_for_draw_cm() if self.toolbar
    435       else nullcontext()):
--> 436     self.figure.draw(self.renderer)
    437     # A GUI class may be need to update a window using this draw, so
    438     # don't forget to call the superclass.
    439     super().draw()

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/artist.py:74, in _finalize_rasterization.<locals>.draw_wrapper(artist, renderer, *args, **kwargs)
     72 @wraps(draw)
     73 def draw_wrapper(artist, renderer, *args, **kwargs):
---> 74     result = draw(artist, renderer, *args, **kwargs)
     75     if renderer._rasterizing:
     76         renderer.stop_rasterizing()

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/artist.py:51, in allow_rasterization.<locals>.draw_wrapper(artist, renderer)
     48     if artist.get_agg_filter() is not None:
     49         renderer.start_filter()
---> 51     return draw(artist, renderer)
     52 finally:
     53     if artist.get_agg_filter() is not None:

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/figure.py:2845, in Figure.draw(self, renderer)
   2842         # ValueError can occur when resizing a window.
   2844 self.patch.draw(renderer)
-> 2845 mimage._draw_list_compositing_images(
   2846     renderer, self, artists, self.suppressComposite)
   2848 for sfig in self.subfigs:
   2849     sfig.draw(renderer)

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/image.py:132, in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)
    130 if not_composite or not has_images:
    131     for a in artists:
--> 132         a.draw(renderer)
    133 else:
    134     # Composite any adjacent images together
    135     image_group = []

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/artist.py:51, in allow_rasterization.<locals>.draw_wrapper(artist, renderer)
     48     if artist.get_agg_filter() is not None:
     49         renderer.start_filter()
---> 51     return draw(artist, renderer)
     52 finally:
     53     if artist.get_agg_filter() is not None:

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/axes/_base.py:3091, in _AxesBase.draw(self, renderer)
   3088         a.draw(renderer)
   3089     renderer.stop_rasterizing()
-> 3091 mimage._draw_list_compositing_images(
   3092     renderer, self, artists, self.figure.suppressComposite)
   3094 renderer.close_group('axes')
   3095 self.stale = False

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/matplotlib/image.py:132, in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)
    130 if not_composite or not has_images:
    131     for a in artists:
--> 132         a.draw(renderer)
    133 else:
    134     # Composite any adjacent images together
    135     image_group = []

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:377, in _CategoricalPlotterNew.plot_swarms.<locals>.draw(points, renderer, center)
    375 def draw(points, renderer, *, center=center):
--> 377     beeswarm(points, center)
    379     if self.orient == "h":
    380         scalex = False

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3400, in Beeswarm.__call__(self, points, center)
   3398 # Adjust points along the categorical axis to prevent overlaps
   3399 new_xyr = np.empty_like(orig_xyr)
-> 3400 new_xyr[sorter] = self.beeswarm(orig_xyr)
   3402 # Transform the point coordinates back to data coordinates
   3403 if self.orient == "h":

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3449, in Beeswarm.beeswarm(self, orig_xyr)
   3446 candidates = candidates[np.argsort(offsets)]
   3448 # Find the first candidate that does not overlap any neighbors
-> 3449 new_xyr_i = self.first_non_overlapping_candidate(candidates, neighbors)
   3451 # Place it into the swarm
   3452 swarm = np.vstack([swarm, new_xyr_i])

File ~/opt/anaconda3/envs/jupyter_books/lib/python3.10/site-packages/seaborn/categorical.py:3504, in Beeswarm.first_non_overlapping_candidate(self, candidates, neighbors)
   3502 dx = neighbors_x - x_i
   3503 dy = neighbors_y - y_i
-> 3504 sq_distances = np.square(dx) + np.square(dy)
   3506 sep_needed = np.square(neighbors_r + r_i)
   3508 # Good candidate does not overlap any of neighbors which means that
   3509 # squared distance between candidate and any of the neighbors has
   3510 # to be at least square of the summed radii

KeyboardInterrupt: 

However, in this case there are so many data points that only a relatively small percentage of them are actually drawn (hence the warning messages above). It’s hard to get a sense of the distributions of values for individual participants.

Appropriately dealing with repeated measures: Aggregating data by participants#

Because there are so many data points, and variation in both the mean and range of values between participants, it is necessary to first average (aggregate) within each participant and condition, prior to computing descriptive statistics or visualizing. This ensures that the variance we are examining is the between-subjects variance, unconfounded with within-subjects variance. This is primarily important in generating plots that show confidence intervals (CIs), such as point plots and bar plots, for reasons explained below.

Seaborn has a way of doing this “on the fly” when you generate plots, using the units= kwarg and passing the name of the grouping variable for repeated measures (in this case, participant):

sns.catplot(kind='point', join=False,
           data=df, units='participant',
           x='flankers', y='rt', hue='flankers')
plt.show()
../_images/930e21683425878b57899b7578ce2be2e29be1735118a31510042f516537fe34.png

This is the most important figure in this lesson. Compared to the point plot with 95% CIs that we generated above for the aggregated data, the CIs in this plot are much larger than those we saw earlier, in the point plot for the un-aggregated data. In that previous plot, the width of the CIs was only about 0.005 s (5 ms). Here, with the CIs computed properly (as reflecting the variance across individuals), the width of the CIs is approximately 0.02 s (20 ms). While still small relative to the differences in the means, the CIs are nonetheless roughly 4x wider in the aggregated data.

Why do CIs get larger when we properly aggregate repeated measures data?#

We’ve already seen that the variance in the data actually went down, as quantified by the min-max range of values, and the standard deviation. So it may seem strange that the CIs do the opposite, and get bigger. To understand this, consider how the 95% CIs are computed:

\(CI = 1.96 * \frac{std}{\sqrt n}\)

CIs are computed by dividing the standard deviation (\(std\)) by the square root of \(n\) — the number of data points used to compute the mean and standard deviation. In the un-aggregated data, this was approximately 1000 – 2000 data points for each condition. In contrast, for the aggregated data, \(n\) = 27. Thus the CIs are much wider, because dividing by a larger number (1000 - 2000, rather than 27) results in a smaller number (even factoring in the differences in standard deviation, and the square root calculation). The values are multiplied by 1.96 because this is the z value corresponding to p = .05 in a normal distribution.

Notably, the conclusions that we would draw from this final plot would be quite different if we had, in error, used the first point plot generated from the un-aggregated data. The first plot made it look like RTs in the three conditions were very different from each other, with little question as to the differences between them (due to the small and non-overlapping CIs). In contrast, when the data are properly aggregated as in the above plot, the CIs for neutral overlap a lot with both of the flanker conditions, which makes us less confident in their differences.

Recall that the neutral condition was intended to help assess whether the congruent flankers showed facilitation relative to neutral, and whether incongruent flankers showed interference relative to neutral. The fact that the neutral condition overlaps with both others, but congruent and incongruent appear significantly different, suggests that the flanker effect (congruent faster than incongruent) may actually be due to a combination of facilitation and interference, rather than one or the other.

The interpretation of the current data set is less important here than understanding the importance of recognizing and appropriately dealing with repeated-measures data. It is critical to understand the effects of repeated measures and nesting on estimate of variance — and of the differences between measures of variance in the data (as reflected by things like standard deviation, or box plots), and estimates of our confidence in the accuracy of the mean (CIs). CIs can be used for statistical inference (i.e., is there a believable difference?), whereas the variance reflected in box plots (and histograms, KDEs, etc.) is more informative at the early stages of exploring and understanding your data, and diagnosing potential issues in the data that might need to be addressed prior to inference about the true patters in the data. Although this initial example used relatively “clean” data — and we didn’t worry about doing anything to address the skew of the distributions or the outliers — in future lessons we will see cases where this first step can be critically important.

Aggregating using Split-Apply-Combine#

While it’s nice that Seaborn can aggregate repeated-measures data for us, that only works for generating Seaborn plots. It’s also good to know how to create a DataFrame of aggregated data, which you can then use to generate tables of descriptive statistics, and eventually perform statistical tests on.

To do this, we will use two split-apply-combine operations in sequence. In one chained command, we will:

  • split the data by participant and condition: .groupby(['participant', 'flankers'])

  • select the rt column: ['rt']

  • apply a calculation to get mean RT for each participant/condition: .mean()

  • split these results by condition only, to combine across participants: .groupby('flankers')

  • apply descriptive statistics and combine the results in a table: .describe()

df.groupby(['participant', 'flankers'])['rt'].mean().groupby('flankers').describe()
count mean std min 25% 50% 75% max
flankers
congruent 27.0 0.470883 0.060013 0.401993 0.430794 0.446840 0.508700 0.628185
incongruent 27.0 0.516718 0.052936 0.428867 0.484333 0.506722 0.542447 0.650769
neutral 27.0 0.498696 0.053444 0.422821 0.458468 0.480008 0.537580 0.619004

Compare the count, std, min, and max values here to that in our first descriptive statistics table above, which we’ll reproduce here for ease of comparison:

df.groupby('flankers')['rt'].describe()
count mean std min 25% 50% 75% max
flankers
congruent 1076.0 0.470568 0.098868 0.009837 0.401727 0.449814 0.521514 0.953989
incongruent 1064.0 0.515996 0.107581 0.061839 0.449544 0.504629 0.573016 0.965726
neutral 2149.0 0.498310 0.098934 0.031539 0.433521 0.485642 0.552863 0.980535

You can see that the number of data points goes down from the total number of trials across all participants (~4200) to 27 per condition (the number of participants). The mean RTs do not actually change at all. This is expected because averaging is a linear operation, meaning that the average across all trials is the same as averaging first within participants then across trials.

However, the range of values decreases when we look at the min and max values. This is a normal consequence of averaging — the most extreme values found in a set of individual data points will be reduced when we average by some grouping variables (like participant).

The variance (as reflected by the standard deviation, std) gets smaller as well, for a similar reason: the variance is now among the average RTs for each participant in each condition, which have a smaller range than the raw data points.

Plot data aggregated across trials#

Now let’s re-plot the data using the same plots we did above, but now for the data averaged within participants.

First let’s save the aggregated data in a new DataFrame called df_avg. We chain the .reset_index() method to the end of this, because .groupby() sets the indexes of the DataFrame to be the grouping variables (in this case, participant and flankers()). This is ideal for looking at tables of the data (e.g., in the above, RT and condition are in bold), but index columns are not visible to Seaborn as data columns for plotting. So we use .reset_index() to convert participant and flankers back from indexes to regular DataFrame columns.

df_avg = pd.DataFrame(df.groupby(['participant', 'flankers'])['rt'].mean()).reset_index()

Now let’s generate a box plot of these data:

sns.catplot(kind='box',
           data=df_avg,
           x='flankers', y='rt')
plt.show()
../_images/32b4b65c4464506f75cd1f14a396b88e7fa1a50f65122a1b47916501c8a8b968.png

Another, somewhat confusing, thing is that the order of the conditions on the x axis is reversed: whereas before the order was congruent- neutral - incongruent, now neutral is last. This is a consequence of the .reset_index() method, which led to the conditions being re-ordered alphabetically. This highlights that it’s always good to pay close attention to your plot labels!

Seaborn actually has an easy way to specify the order of values on a categorical axis, if you want to: using the order= kwarg, and passing a list of conditions in the order you want them to appear:

sns.catplot(kind='box',
           data=df_avg,
           x='flankers', y='rt',
           order=['congruent', 'neutral', 'incongruent'])
plt.show()
../_images/cc60b63aa7b6bbf4cef3120f0e888bda626718f6557258f7184345c6f2216577.png

If you compare this to the box plot based on every data point that we generated earlier, you’ll see that the range covered by the whiskers and the box are both smaller. This is consistent with the reduced range in the data that we observed in the tables of descriptive statistics.

Strip plot#

A strip plot of the data is now much easier to make sense of, because we only have one point per participant in each condition:

sns.catplot(kind='strip',
           data=df_avg,
           x='flankers', y='rt', hue='participant')
plt.show()
../_images/2e8e98bec687c7fc4c8abc1b8d90741362309394cfa6166448a5b454228c97f0.png

Line plot#

This makes it easier to get a sense of the variability between individuals. However, It’s a bit hard to see how each individual’s RT changes between the two experimental conditions. To facilitate this, let’s use a point plot, which by default draws lines to join data from each individual:

sns.catplot(kind='point',
           data=df_avg,
           x='flankers', y='rt', hue='participant',
           order=['congruent', 'neutral', 'incongruent'])
plt.show()
../_images/0a6b6db2bb57d717b4bfff5f718522cf6d3c3708b4eb2abd399e5cf3253c526b.png

You may recall that point plots typically show means and 95% CIs, but the plot above lacks CIs. This is because in the df_avg DataFrame, there is only one value for each participant in each condition (because we averaged across trials). If we wanted to see CIs for each individual, we could use the un-aggregated data (df).

sns.catplot(kind='point',
           data=df,
           x='flankers', y='rt', hue='participant',
           order=['congruent', 'neutral', 'incongruent'])
plt.show()
../_images/9d07da5435578e24e94b7bacc3109f952d219dfd787da3318cb74ded24742bc6.png

Point plots aggregated across participants#

The CIs in the above plot don’t, however, help us to make a simple inference as to whether, across all participants, there is a significant RT difference between congruent and incongruent. The data are certainly suggestive of this, because all participants show slower RTs for incongruent. But this is where a point plot with 95% CIs, across participants, is useful:

sns.catplot(kind='point', join=False,
           data=df_avg,
           x='flankers', y='rt', hue='flankers',
           order=['congruent', 'neutral', 'incongruent'])
plt.show()
../_images/137df2f7bd448bb4dd5fb1312dc6a76b5923ba6ffe16432ed4274b1ca8353fb9.png

Summary#

  • Repeated measures occur when we take repeated samples of the same measurement. This is common in cognitive psychology and neuroscience data, because individual measurements typically include noise. Averaging across repeated measurements yields a more robust estimate of the true value for that individual

  • When we have repeated measures data, we need to be concerned about the difference between within-subjects variance, and between-subjects variance

  • Within-subjects variance is the variation within an individual, across the repeated measurements we take

  • Between-subjects variance is the variation between individuals, in the average response across the repeated measures taken from that individual

  • If we simply average across all trials and participants, our measures of variance will be inaccurate, because we are conflating within- and between-subjects variance

  • Conflating within-and between-subjects variance will lead to incorrect conclusions, particularly because confidence intervals and statistical tests consider the number of samples in their computations. When assessing effects across a group of participants, the number of participants is the appropriate number of samples to use. The much larger number of samples that comes from summing the number of individual trials across all participants will artificially inflate estimates of the significance of experimental effects

  • The proper way to work with repeated measures data is to first average across all trials for each experimental condition, within an individual. We call this aggregation

  • After we have properly aggregated the data within each individual, then we can average across participants and accurately estimate the between-subjects variance

  • Between-subject variance is typically what is important in assessing the significance of experimental manipulations

  • Proper aggregation of repeated measures data is easy using the pandas .groupby() method


This section was adapted from Aaron J. Newman’s Data Science for Psychology and Neuroscience - in Python and Software Carpentry’s Plotting and Programming in Python workshop.