Note
Go to the end to download the full example code. or to run this example in your browser via Binder
ROI-based decoding analysis in Haxby et al. dataset¶
In this script we reproduce the data analysis conducted by Haxby et al.[1].
Specifically, we look at decoding accuracy for different objects in three different masks: the full ventral stream (mask_vt), the house selective areas (mask_house) and the face selective areas (mask_face), that have been defined via a standard GLM-based analysis.
# Fetch data using nilearn dataset fetcher
from nilearn import datasets
from nilearn.plotting import show
Load and prepare the data¶
# by default we fetch 2nd subject data for analysis
haxby_dataset = datasets.fetch_haxby()
func_filename = haxby_dataset.func[0]
# Print basic information on the dataset
print(
"First subject anatomical nifti image (3D) located is "
f"at: {haxby_dataset.anat[0]}"
)
print(
f"First subject functional nifti image (4D) is located at: {func_filename}"
)
# load labels
import pandas as pd
# Load nilearn NiftiMasker, the practical masking and unmasking tool
from nilearn.maskers import NiftiMasker
labels = pd.read_csv(haxby_dataset.session_target[0], sep=" ")
stimuli = labels["labels"]
# identify resting state labels in order to be able to remove them
task_mask = stimuli != "rest"
# find names of remaining active labels
categories = stimuli[task_mask].unique()
# extract tags indicating to which acquisition run a tag belongs
run_labels = labels["chunks"][task_mask]
# apply the task_mask to fMRI data (func_filename)
from nilearn.image import index_img
task_data = index_img(func_filename, task_mask)
[get_dataset_dir] Dataset found in /home/runner/nilearn_data/haxby2001
First subject anatomical nifti image (3D) located is at: /home/runner/nilearn_data/haxby2001/subj2/anat.nii.gz
First subject functional nifti image (4D) is located at: /home/runner/nilearn_data/haxby2001/subj2/bold.nii.gz
Decoding on the different masks¶
The classifier used here is a support vector classifier (svc).
We use
Decoder
and specify the classifier.
import numpy as np
# Make a data splitting object for cross validation
from sklearn.model_selection import LeaveOneGroupOut
from nilearn.decoding import Decoder
cv = LeaveOneGroupOut()
We use Decoder
to estimate a baseline.
mask_names = ["mask_vt", "mask_face", "mask_house"]
mask_scores = {}
mask_chance_scores = {}
for mask_name in mask_names:
print(f"Working on {mask_name}")
# For decoding, standardizing is often very important
mask_filename = haxby_dataset[mask_name][0]
masker = NiftiMasker(mask_img=mask_filename, standardize="zscore_sample")
mask_scores[mask_name] = {}
mask_chance_scores[mask_name] = {}
for category in categories:
print(f"Processing {mask_name} {category}")
classification_target = stimuli[task_mask] == category
# Specify the classifier to the decoder object.
# With the decoder we can input the masker directly.
# We are using the svc_l1 here because it is intra subject.
decoder = Decoder(
estimator="svc_l1",
cv=cv,
mask=masker,
scoring="roc_auc",
standardize="zscore_sample",
)
decoder.fit(task_data, classification_target, groups=run_labels)
mask_scores[mask_name][category] = decoder.cv_scores_[1]
mean = np.mean(mask_scores[mask_name][category])
std = np.std(mask_scores[mask_name][category])
print(f"Scores: {mean:1.2f} +- {std:1.2f}")
dummy_classifier = Decoder(
estimator="dummy_classifier",
cv=cv,
mask=masker,
scoring="roc_auc",
standardize="zscore_sample",
)
dummy_classifier.fit(
task_data, classification_target, groups=run_labels
)
mask_chance_scores[mask_name][category] = dummy_classifier.cv_scores_[
1
]
Working on mask_vt
Processing mask_vt scissors
Scores: 0.92 +- 0.05
Processing mask_vt face
Scores: 0.98 +- 0.03
Processing mask_vt cat
Scores: 0.96 +- 0.04
Processing mask_vt shoe
Scores: 0.92 +- 0.07
Processing mask_vt house
Scores: 1.00 +- 0.00
Processing mask_vt scrambledpix
Scores: 0.99 +- 0.01
Processing mask_vt bottle
Scores: 0.89 +- 0.08
Processing mask_vt chair
Scores: 0.93 +- 0.04
Working on mask_face
Processing mask_face scissors
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Scores: 0.70 +- 0.16
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Processing mask_face face
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Scores: 0.90 +- 0.06
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Processing mask_face cat
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Scores: 0.76 +- 0.12
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Processing mask_face shoe
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Scores: 0.73 +- 0.17
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Processing mask_face house
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Scores: 0.71 +- 0.16
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Processing mask_face scrambledpix
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Scores: 0.87 +- 0.09
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Processing mask_face bottle
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Scores: 0.68 +- 0.17
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Processing mask_face chair
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Scores: 0.63 +- 0.10
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:796: UserWarning: After clustering and screening, the decoding model will be trained only on 30 features. Consider raising clustering_percentile or screening_percentile parameters.
warnings.warn(
Working on mask_house
Processing mask_house scissors
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Scores: 0.83 +- 0.08
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Processing mask_house face
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Scores: 0.90 +- 0.07
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Processing mask_house cat
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Scores: 0.86 +- 0.09
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Processing mask_house shoe
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Scores: 0.82 +- 0.12
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Processing mask_house house
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Scores: 1.00 +- 0.00
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Processing mask_house scrambledpix
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Scores: 0.96 +- 0.05
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Processing mask_house bottle
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Scores: 0.86 +- 0.10
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Processing mask_house chair
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
Scores: 0.90 +- 0.10
/home/runner/work/nilearn/nilearn/.tox/doc/lib/python3.9/site-packages/nilearn/decoding/decoder.py:774: UserWarning: Brain mask is smaller than .5% of the size of the standard human brain. This object is probably not tuned to be used on such data.
selector = check_feature_screening(
We make a simple bar plot to summarize the results¶
import matplotlib.pyplot as plt
plt.figure(constrained_layout=True)
tick_position = np.arange(len(categories))
plt.xticks(tick_position, categories, rotation=45)
for color, mask_name in zip("rgb", mask_names):
score_means = [
np.mean(mask_scores[mask_name][category]) for category in categories
]
plt.bar(
tick_position, score_means, label=mask_name, width=0.25, color=color
)
score_chance = [
np.mean(mask_chance_scores[mask_name][category])
for category in categories
]
plt.bar(
tick_position,
score_chance,
width=0.25,
edgecolor="k",
facecolor="none",
)
tick_position = tick_position + 0.2
plt.ylabel("Classification accuracy (AUC score)")
plt.xlabel("Visual stimuli category")
plt.ylim(0.3, 1)
plt.legend(loc="lower right")
plt.title("Category-specific classification accuracy for different masks")
show()
References¶
Total running time of the script: (2 minutes 40.201 seconds)
Estimated memory usage: 1448 MB