Chapter 5 Using multiple references
5.1 Overview
In some cases, we may wish to use multiple references for annotation of a test dataset. This yields a more comprehensive set of cell types that are not covered by any individual reference, especially when differences in the resolution are considered. However, it is not trivial due to the presence of batch effects across references (from differences in technology, experimental protocol or the biological system) as well as differences in the annotation vocabulary between investigators.
Several strategies are available to combine inferences from multiple references:
- using reference-specific labels in a combined reference
- using harmonized labels in a combined reference
- combining scores across multiple references
This chapter discusses the various strengths and weaknesses of each strategy and provides some practical demonstrations of each. Here, we will use the HPCA and Blueprint/ENCODE datasets as our references and (yet another) PBMC dataset as the test.
5.2 Using reference-specific labels
In this strategy, each label is defined in the context of its reference dataset. This means that a label - say, “B cell” - in reference dataset X is considered to be different from a “B cell” label in reference dataset Y. Use of reference-specific labels is most appropriate if there are relevant biological differences between the references; for example, if one reference is concerned with healthy tissue while the other reference considers diseased tissue, it can be helpful to distinguish between the same cell type in different biological contexts.
We can easily implement this approach by combining the expression matrices together
and pasting the reference name onto the corresponding character vector of labels.
This modification ensures that the downstream SingleR()
call
will treat each label-reference combination as a distinct entity.
hpca2 <- hpca
hpca2$label.main <- paste0("HPCA.", hpca2$label.main)
bpe2 <- bpe
bpe2$label.main <- paste0("BPE.", bpe2$label.main)
shared <- intersect(rownames(hpca2), rownames(bpe2))
combined <- cbind(hpca2[shared,], bpe2[shared,])
It is then straightforward to perform annotation with the usual methods.
library(SingleR)
com.res1 <- SingleR(pbmc, ref=combined, labels=combined$label.main, assay.type.test=1)
table(com.res1$labels)
##
## BPE.B-cells BPE.CD4+ T-cells BPE.CD8+ T-cells BPE.HSC
## 1178 1708 2656 20
## BPE.Monocytes BPE.NK cells HPCA.HSC_-G-CSF HPCA.Platelets
## 2349 460 1 7
## HPCA.T_cells
## 2
However, this strategy identifies markers by directly comparing expression values across references, meaning that the marker set is likely to contain genes responsible for uninteresting batch effects. This will increase noise during the calculation of the score in each reference, possibly leading to a loss of precision and a greater risk of technical variation dominating the classification results. The use of reference-specific labels also complicates interpretation of the results as the cell type is always qualified by its reference of origin.
5.3 Comparing scores across references
5.3.1 Combining inferences from individual references
Another strategy - and the default approach implemented in SingleR()
-
involves performing classification separately within each reference,
and then collating the results to choose the label with the highest score across references.
This is a relatively expedient approach that avoids the need for explicit harmonization
while also reducing exposure to reference-specific batch effects.
To use this method, we simply pass multiple objects to the ref=
and label=
argument in SingleR()
.
The combining strategy is as follows:
- The function first annotates the test dataset with each reference individually
in the same manner as described in Section 1.2.
This step is almost equivalent to simply looping over all individual references and running
SingleR()
on each. - For each cell, the function collects its predicted labels across all references. In doing so, it also identifies the union of markers that are upregulated in the predicted label in each reference.
- The function identifies the overall best-scoring label as the final prediction for that cell. This step involves a recomputation of the scores across the identified marker subset to ensure that these scores are derived from the same set of genes (and are thus comparable across references).
The function will then return a DataFrame
of combined results for each cell in the test dataset,
including the overall label and the reference from which it was assigned.
com.res2 <- SingleR(test = pbmc, assay.type.test=1,
ref = list(BPE=bpe, HPCA=hpca),
labels = list(bpe$label.main, hpca$label.main))
# Check the final label from the combined assignment.
table(com.res2$labels)
##
## B-cells B_cell CD4+ T-cells CD8+ T-cells
## 1170 14 1450 2936
## GMP HSC Monocyte Monocytes
## 1 22 753 1560
## NK cells NK_cell Platelets Pre-B_cell_CD34-
## 372 10 9 16
## T_cells
## 68
##
## 1 2
## 7510 871
The main appeal of this approach lies in the fact that it is based on the results of annotation with individual references. This avoids batch effects from comparing expression values across references; it reduces the need for any coordination in the label scheme between references; and simultaneously provides the per-reference annotations in the results. The last feature is particularly useful as it allows for more detailed diagnostics, troubleshooting and further analysis.
## [1] "B-cells" "Monocytes" "CD8+ T-cells" "CD8+ T-cells" "Monocytes"
## [6] "Monocytes"
## [1] "B_cell" "Monocyte" "T_cells" "T_cells" "Monocyte" "Monocyte"
The main downside is that it is somewhat suboptimal if there are many labels that are unique to one reference, as markers are not identified with the aim of distinguishing a label in one reference from another label in another reference. The continued lack of consistency in the labels across references also complicates interpretation of the results, though we can overcome this by using harmonized labels as described below.
5.3.2 Combined diagnostics
All of the diagnostic plots in SingleR will naturally operate on these combined results.
For example, we can create a heatmap of the scores in all of the individual references
as well as for the recomputed scores in the combined results (Figure 5.1).
Note that scores are only recomputed for the labels predicted in the individual references,
so all labels outside of those are simply set to NA
- hence the swathes of grey.
The deltas for each individual reference can also be plotted with plotDeltaDistribution()
(Figure 5.2).
No deltas are shown for the recomputed scores as the assumption described in Section 4.3
may not be applicable across the predicted labels from the individual references.
For example, if all individual references suggest the same cell type with similar recomputed scores,
any delta would be low even though the assignment is highly confident.
We can similarly extract marker genes to use in heatmaps as described in Section 4.4.
As annotation was performed to each individual reference,
we can simply extract the marker genes from the nested DataFrame
s as shown in Figure 5.3.
hpca.markers <- metadata(com.res2$orig.results$HPCA)$de.genes
bpe.markers <- metadata(com.res2$orig.results$BPE)$de.genes
mono.markers <- unique(unlist(hpca.markers$Monocyte, bpe.markers$Monocytes))
library(scater)
plotHeatmap(logNormCounts(pbmc),
order_columns_by=list(I(com.res2$labels)),
features=mono.markers)
5.4 Using harmonized labels
5.4.2 Manual label harmonization
The matchReferences()
function provides a simple approach for label harmonization between two references.
Each reference is used to annotate the other and the probability of mutual assignment between each pair of labels is computed,
i.e., for each pair of labels, what is the probability that a cell with one label is assigned the other and vice versa?
Probabilities close to 1 in Figure 5.4 indicate there is a 1:1 relation between that pair of labels;
on the other hand, an all-zero probability vector indicates that a label is unique to a particular reference.
library(SingleR)
bp.se <- BlueprintEncodeData()
hpca.se <- HumanPrimaryCellAtlasData()
matched <- matchReferences(bp.se, hpca.se,
bp.se$label.main, hpca.se$label.main)
pheatmap::pheatmap(matched, col=viridis::plasma(100))
This function can be used to guide harmonization to enforce a consistent vocabulary between two sets of labels. However, some manual intervention is still required in this process given the ambiguities posed by differences in biological systems and technologies. In the example above, neurons are considered to be unique to each reference while smooth muscle cells in the HPCA data are incorrectly matched to fibroblasts in the Blueprint/ENCODE data. CD4+ and CD8+ T cells are also both assigned to “T cells”, so some decision about the acceptable resolution of the harmonized labels is required here.
As an aside, we can also use this function to identify the matching clusters between two independent scRNA-seq analyses. This involves substituting the cluster assignments as proxies for the labels, allowing us to match up clusters and integrate conclusions from multiple datasets without the difficulties of batch correction and reclustering.
Session info
R version 4.4.1 (2024-06-14)
Platform: x86_64-pc-linux-gnu
Running under: Ubuntu 24.04.1 LTS
Matrix products: default
BLAS: /home/biocbuild/bbs-3.20-bioc/R/lib/libRblas.so
LAPACK: /usr/lib/x86_64-linux-gnu/lapack/liblapack.so.3.12.0
locale:
[1] LC_CTYPE=en_US.UTF-8 LC_NUMERIC=C
[3] LC_TIME=en_GB LC_COLLATE=C
[5] LC_MONETARY=en_US.UTF-8 LC_MESSAGES=en_US.UTF-8
[7] LC_PAPER=en_US.UTF-8 LC_NAME=C
[9] LC_ADDRESS=C LC_TELEPHONE=C
[11] LC_MEASUREMENT=en_US.UTF-8 LC_IDENTIFICATION=C
time zone: America/New_York
tzcode source: system (glibc)
attached base packages:
[1] stats4 stats graphics grDevices utils datasets methods
[8] base
other attached packages:
[1] scater_1.34.0 ggplot2_3.5.1
[3] scuttle_1.16.0 SingleR_2.8.0
[5] ensembldb_2.30.0 AnnotationFilter_1.30.0
[7] GenomicFeatures_1.58.0 AnnotationDbi_1.68.0
[9] celldex_1.15.0 TENxPBMCData_1.23.0
[11] HDF5Array_1.34.0 rhdf5_2.50.0
[13] DelayedArray_0.32.0 SparseArray_1.6.0
[15] S4Arrays_1.6.0 abind_1.4-8
[17] Matrix_1.7-1 SingleCellExperiment_1.28.0
[19] SummarizedExperiment_1.36.0 Biobase_2.66.0
[21] GenomicRanges_1.58.0 GenomeInfoDb_1.42.0
[23] IRanges_2.40.0 S4Vectors_0.44.0
[25] BiocGenerics_0.52.0 MatrixGenerics_1.18.0
[27] matrixStats_1.4.1 BiocStyle_2.34.0
[29] rebook_1.16.0
loaded via a namespace (and not attached):
[1] RColorBrewer_1.1-3 jsonlite_1.8.9
[3] CodeDepends_0.6.6 magrittr_2.0.3
[5] ggbeeswarm_0.7.2 gypsum_1.2.0
[7] farver_2.1.2 rmarkdown_2.28
[9] BiocIO_1.16.0 zlibbioc_1.52.0
[11] vctrs_0.6.5 memoise_2.0.1
[13] Rsamtools_2.22.0 DelayedMatrixStats_1.28.0
[15] RCurl_1.98-1.16 htmltools_0.5.8.1
[17] AnnotationHub_3.14.0 curl_5.2.3
[19] BiocNeighbors_2.0.0 Rhdf5lib_1.28.0
[21] sass_0.4.9 alabaster.base_1.6.0
[23] bslib_0.8.0 httr2_1.0.5
[25] cachem_1.1.0 GenomicAlignments_1.42.0
[27] mime_0.12 lifecycle_1.0.4
[29] pkgconfig_2.0.3 rsvd_1.0.5
[31] R6_2.5.1 fastmap_1.2.0
[33] GenomeInfoDbData_1.2.13 digest_0.6.37
[35] colorspace_2.1-1 irlba_2.3.5.1
[37] ExperimentHub_2.14.0 RSQLite_2.3.7
[39] beachmat_2.22.0 labeling_0.4.3
[41] filelock_1.0.3 fansi_1.0.6
[43] httr_1.4.7 compiler_4.4.1
[45] bit64_4.5.2 withr_3.0.2
[47] BiocParallel_1.40.0 viridis_0.6.5
[49] DBI_1.2.3 highr_0.11
[51] alabaster.ranges_1.6.0 alabaster.schemas_1.6.0
[53] rappdirs_0.3.3 rjson_0.2.23
[55] tools_4.4.1 vipor_0.4.7
[57] beeswarm_0.4.0 glue_1.8.0
[59] restfulr_0.0.15 rhdf5filters_1.18.0
[61] grid_4.4.1 generics_0.1.3
[63] gtable_0.3.6 BiocSingular_1.22.0
[65] ScaledMatrix_1.14.0 utf8_1.2.4
[67] XVector_0.46.0 ggrepel_0.9.6
[69] BiocVersion_3.20.0 pillar_1.9.0
[71] dplyr_1.1.4 BiocFileCache_2.14.0
[73] lattice_0.22-6 rtracklayer_1.66.0
[75] bit_4.5.0 tidyselect_1.2.1
[77] Biostrings_2.74.0 knitr_1.48
[79] gridExtra_2.3 bookdown_0.41
[81] ProtGenerics_1.38.0 xfun_0.48
[83] pheatmap_1.0.12 UCSC.utils_1.2.0
[85] lazyeval_0.2.2 yaml_2.3.10
[87] evaluate_1.0.1 codetools_0.2-20
[89] tibble_3.2.1 alabaster.matrix_1.6.0
[91] BiocManager_1.30.25 graph_1.84.0
[93] cli_3.6.3 munsell_0.5.1
[95] jquerylib_0.1.4 Rcpp_1.0.13
[97] dir.expiry_1.14.0 dbplyr_2.5.0
[99] png_0.1-8 XML_3.99-0.17
[101] parallel_4.4.1 blob_1.2.4
[103] beachmat.hdf5_1.4.0 sparseMatrixStats_1.18.0
[105] bitops_1.0-9 viridisLite_0.4.2
[107] alabaster.se_1.6.0 scales_1.3.0
[109] purrr_1.0.2 crayon_1.5.3
[111] rlang_1.1.4 KEGGREST_1.46.0