Based on Gilis et al. (2023)
Gilis J, Perin L, Malfait M, Van den Berge K,
Assefa AT, Verbist B, Risso D, and Clement L:
Differential detection workflows for
multi-sample single-cell RNA-seq data.
bioRxiv (2023). DOI: 10.1101/2023.12.17.572043
Single-cell RNA-sequencing (scRNA-seq) has improved our understanding of complex biological processes by elucidating cell-level heterogeneity in gene expression. One of the key tasks in the downstream analysis of scRNA-seq data is studying differential gene expression (DE). Most DE analysis methods aim to identify genes for which the average expression differs between biological groups of interest, e.g., between cell types or between diseased and healthy cells. As such, most methods allow for assessing only one aspect of the gene expression distribution: the mean. However, in scRNA-seq data, differences in other characteristics between count distributions can commonly be observed.
One such characteristic is gene detection, i.e., the number of cells in which a gene is (detectably) expressed. Analogous to a DE analysis, a differential detection (DD) analysis aims to identify genes for which the average fraction of cells in which the gene is detected changes between groups. In Gilis et al. (2023), we show how DD analysis contain information that is biologically relevant, and that is largely orthogonal to the information obtained from DE analysis on the same data.
In this vignette, we display how muscat
can be used to
perform DD analyses in multi-sample, multi-group,
multi-(cell-)subpopulation scRNA-seq data. Furthermore, we show how DD
and DS analysis results on the same data can be effectively combined
using a two-stage testing approach. This workflow thus allows users to
jointly assess two biological hypotheses containing orthogonal
information, which thus can be expected to improve their understanding
of complex biological phenomena, at no extra cost.
We will use the same data as in the differential state (DS) analyses described in muscat, namely, scRNA-seq data acquired on PBMCs from 8 patients before and after IFN-β treatment. For a more detailed description of these data and subsequent preprocessing, we refer to muscat.
## ExperimentHub with 3 records
## # snapshotDate(): 2024-10-24
## # $dataprovider: NCI_GDC, GEO
## # $species: Homo sapiens
## # $rdataclass: character, SingleCellExperiment, BSseq
## # additional mcols(): taxonomyid, genome, description,
## # coordinate_1_based, maintainer, rdatadateadded, preparerclass, tags,
## # rdatapath, sourceurl, sourcetype
## # retrieve records with, e.g., 'object[["EH1661"]]'
##
## title
## EH1661 | Whole Genome Bisulfit Sequencing Data for 47 samples
## EH1662 | Whole Genome Bisulfit Sequencing Data for 47 samples
## EH2259 | Kang18_8vs8
## class: SingleCellExperiment
## dim: 35635 29065
## metadata(0):
## assays(1): counts
## rownames(35635): MIR1302-10 FAM138A ... MT-ND6 MT-CYB
## rowData names(2): ENSEMBL SYMBOL
## colnames(29065): AAACATACAATGCC-1 AAACATACATTTCC-1 ... TTTGCATGGTTTGG-1
## TTTGCATGTCTTAC-1
## colData names(5): ind stim cluster cell multiplets
## reducedDimNames(1): TSNE
## mainExpName: NULL
## altExpNames(0):
We further apply some minimal filtering to remove low-quality genes
and cells, and use prepSCE()
to standardize cell metadata
such that slots specifying cluster (cell
), sample
(stim
+ind
), and group (stim
)
identifiers conform with the muscat
framework:
sce <- sce[rowSums(counts(sce) > 0) > 0, ]
qc <- perCellQCMetrics(sce)
sce <- sce[, !isOutlier(qc$detected, nmads=2, log=TRUE)]
sce <- sce[rowSums(counts(sce) > 1) >= 10, ]
sce$id <- paste0(sce$stim, sce$ind)
sce <- prepSCE(sce, "cell", "id", "stim")
table(sce$cluster_id, sce$group_id)
##
## ctrl stim
## B cells 1422 1313
## CD14+ Monocytes 2855 2662
## CD4 T cells 5874 5897
## CD8 T cells 1369 1188
## Dendritic cells 90 130
## FCGR3A+ Monocytes 698 791
## Megakaryocytes 117 130
## NK cells 1038 1246
##
## ctrl101 ctrl1015 ctrl1016 ctrl1039 ctrl107 ctrl1244 ctrl1256 ctrl1488
## 912 2880 2062 436 586 2091 2267 2229
## stim101 stim1015 stim1016 stim1039 stim107 stim1244 stim1256 stim1488
## 1197 2494 1901 639 578 1642 2127 2779
In general, aggregateData()
will aggregate the data by
the colData
variables specified with argument
by
, and return a SingleCellExperiment
containing pseudobulk data.
To perform a pseudobulk-level analysis, measurements must be
aggregated at the cluster-sample level (default
by = c("cluster_id", "sample_id"
). In this case, the
returned SingleCellExperiment
will contain one assay per
cluster, where rows = genes and columns = samples. Arguments
assay
and fun
specify the input data and
summary statistic, respectively, to use for aggregation.
In a differential detection (DD) analysis, the default choice of the
summary statistic used for aggregation is
fun = "num.detected"
. This strategy can be thought of as
first binarizing the gene expression values (1: expressed, 0: not
expressed), and subsequently performing a simple summation of the
binarized gene expression counts for cells belonging to the same
cluster-sample level. Hence, the resulting pseudobulk-level expression
count reflects the total number of cells in a particular cluster-sample
level with a non-zero gene expression value.
In a differential state (DS) analysis, the default choice for
aggregation is fun = "sum"
, which amounts to the simple
summation of the raw gene expression counts of cells belonging to the
same cluster-sample level.
pb_sum <- aggregateData(sce,
assay="counts", fun="sum",
by=c("cluster_id", "sample_id"))
pb_det <- aggregateData(sce,
assay="counts", fun="num.detected",
by=c("cluster_id", "sample_id"))
t(head(assay(pb_det)))
## NOC2L HES4 ISG15 TNFRSF18 TNFRSF4 SDF4
## ctrl101 11 0 10 7 1 10
## ctrl1015 37 4 86 43 10 39
## ctrl1016 12 2 30 7 0 15
## ctrl1039 2 1 10 3 1 1
## ctrl107 5 0 4 3 1 5
## ctrl1244 17 5 14 17 4 6
## ctrl1256 21 1 45 22 6 16
## ctrl1488 16 1 28 18 3 20
## stim101 12 8 142 3 1 8
## stim1015 32 26 356 16 2 16
## stim1016 5 7 129 1 3 6
## stim1039 2 2 39 1 1 1
## stim107 3 2 56 1 0 3
## stim1244 11 9 93 5 2 5
## stim1256 14 8 210 8 1 6
## stim1488 16 4 282 9 1 15
Qiu (2020) demonstrated that binarizing scRNA-seq counts generates expression profiles that still accurately reflect biological variation. This finding was confirmed by Bouland, Mahfouz, and Reinders (2021), who showed that the frequencies of zero counts capture biological variability, and further claimed that a binarized representation of the single-cell expression data allows for a more robust description of the relative abundance of transcripts than counts.
pbMDS(pb_sum) + ggtitle("Σ counts") +
pbMDS(pb_det) + ggtitle("# detected") +
plot_layout(guides="collect") +
plot_annotation(tag_levels="A") &
theme(legend.key.size=unit(0.5, "lines"))
Once we have assembled the pseudobulk data, we can test for DD using
pbDD()
. By default, a ∼group_id
model is fit, and the
last coefficient of the linear model is tested to be equal to zero.
Inspection, manipulation, and visualization of DD analysis results follows the same principles as for a DS analysis. For a detailed description, we refer to the DS analysis vignettemuscat. Below, some basic functionalities are being displayed.
## [1] "B cells" "CD14+ Monocytes" "CD4 T cells"
## [4] "CD8 T cells" "Dendritic cells" "FCGR3A+ Monocytes"
## [7] "Megakaryocytes" "NK cells"
## gene cluster_id logFC logCPM F p_val p_adj.loc p_adj.glb
## 1 NOC2L B cells -3.0e-01 7.6 2.4e+00 1.2e-01 2.9e-01 3.1e-01
## 2 HES4 B cells 2.2e+00 6.5 3.5e+01 1.1e-08 2.4e-07 2.8e-07
## 3 ISG15 B cells 2.6e+00 10.3 1.1e+03 9.3e-88 1.1e-84 8.6e-84
## 4 TNFRSF18 B cells -1.4e+00 7.3 3.4e+01 1.6e-08 3.5e-07 4.0e-07
## 5 TNFRSF4 B cells -1.1e+00 5.8 5.7e+00 1.8e-02 7.5e-02 8.0e-02
## 6 SDF4 B cells -8.4e-01 7.3 1.5e+01 1.7e-04 1.7e-03 1.8e-03
# filter FDR < 5%, |logFC| > 1 & sort by adj. p-value
tbl_fil <- lapply(tbl, \(u)
filter(u,
p_adj.loc < 0.05,
abs(logFC) > 1) |>
arrange(p_adj.loc))
# nb. of DS genes & % of total by cluster
n_de <- vapply(tbl_fil, nrow, numeric(1))
p_de <- format(n_de / nrow(sce) * 100, digits = 3)
data.frame("#DD" = n_de, "%DD" = p_de, check.names = FALSE)
## #DD %DD
## B cells 715 10.04
## CD14+ Monocytes 1982 27.84
## CD4 T cells 498 7.00
## CD8 T cells 333 4.68
## Dendritic cells 277 3.89
## FCGR3A+ Monocytes 1110 15.59
## Megakaryocytes 109 1.53
## NK cells 423 5.94
While DD analysis results may contain biologically relevant information in their own right, we show in Gilis et al. (2023) that combing DD and DS analysis results on the same data can further improve our understanding of complex biological phenomena. In the remainder of this vignette, we show how DD and DS analysis results on the same data can be effectively combined.
For this, we build on the two-stage testing paradigm proposed by Van den Berge et al. (2017). In the first stage of this testing procedure, we identify differential genes by using an omnibus test for differential detection and differential expression (DE). The null hypothesis for this test is that the gene is neither differentially detected, nor differentially expressed.
In the second stage, we perform post-hoc tests on the differential genes from stage one to unravel whether they are DD, DE or both. Compared to the individual DD and DS analysis results, the two-stage approach increases statistical power and provides better type 1 error control.
## DataFrame with 6 rows and 8 columns
## gene p_adj p_val.DS p_val.DD cluster_id contrast
## <character> <numeric> <numeric> <numeric> <character> <character>
## 1 NOC2L 3.73143e-01 NA NA B cells stim
## 2 HES4 4.71932e-07 1.35239e-05 5.53293e-08 B cells stim
## 3 ISG15 2.17494e-84 2.62823e-32 4.67394e-87 B cells stim
## 4 TNFRSF18 6.86768e-07 1.00944e-04 8.19769e-08 B cells stim
## 5 TNFRSF4 1.24236e-02 4.70871e-03 9.07348e-02 B cells stim
## 6 SDF4 3.21728e-03 6.89599e-02 8.71664e-04 B cells stim
## res_DS res_DD
## <data.frame> <data.frame>
## 1 NOC2L:B cells:-0.208711:... NOC2L:B cells:-0.303141:...
## 2 HES4:B cells: 2.225287:... HES4:B cells: 2.201890:...
## 3 ISG15:B cells: 5.521586:... ISG15:B cells: 2.568004:...
## 4 TNFRSF18:B cells:-1.271499:... TNFRSF18:B cells:-1.382451:...
## 5 TNFRSF4:B cells:-1.645505:... TNFRSF4:B cells:-1.126530:...
## 6 SDF4:B cells:-0.612369:... SDF4:B cells:-0.844448:...
# for each approach, get adjusted p-values across clusters
ps <- map_depth(res, 2, \(df) {
data.frame(
df[, c("gene", "cluster_id")],
p_adj.stagewise=df$p_adj,
p_adj.DS=df$res_DS$p_adj.loc,
p_adj.DD=df$res_DD$p_adj.loc)
}) |>
lapply(do.call, what=rbind) |>
do.call(what=rbind) |>
data.frame(row.names=NULL)
head(ps)
## gene cluster_id p_adj.stagewise p_adj.DS p_adj.DD
## 1 NOC2L B cells 3.731430e-01 6.064017e-01 2.863777e-01
## 2 HES4 B cells 4.719323e-07 5.686554e-05 2.399617e-07
## 3 ISG15 B cells 2.174939e-84 1.834502e-29 1.077728e-84
## 4 TNFRSF18 B cells 6.867680e-07 3.576604e-04 3.468337e-07
## 5 TNFRSF4 B cells 1.242358e-02 9.526608e-03 7.494398e-02
## 6 SDF4 B cells 3.217278e-03 8.069408e-02 1.727711e-03
To get an overview of how different approaches compare, we can count the number of genes found differential in each cluster for a given FDR threshold:
# for each approach & cluster, count number
# of genes falling below 5% FDR threshold
ns <- lapply(seq(0, 0.2, 0.005), \(th) {
ps |>
mutate(th=th) |>
group_by(cluster_id, th) |>
summarise(
.groups="drop",
across(starts_with("p_"),
\(.) sum(. < th, na.rm=TRUE)))
}) |>
do.call(what=rbind) |>
pivot_longer(starts_with("p_"))
ggplot(ns, aes(th, value, col=name)) +
geom_line(linewidth=0.8, key_glyph="point") +
geom_vline(xintercept=0.05, lty=2, linewidth=0.4) +
guides(col=guide_legend(NULL, override.aes=list(size=3))) +
labs(x="FDR threshold", y="number of significantly\ndifferential genes") +
facet_wrap(~cluster_id, scales="free_y", nrow=2) +
theme_bw() + theme(
panel.grid.minor=element_blank(),
legend.key.size=unit(0.5, "lines"))
We can further identify which hits are shared between or unique to a given approach. In the example below, for instance, the vast majority of hits is common to all approaches, many hits are shared between DD and stagewise testing, and only few genes are specific to any one approach:
# subset adjuster p-values for cluster of interest
qs <- ps[grep("CD4", ps$cluster_id), grep("p_", names(ps))]
# for each approach, extract genes at 5% FDR threshold
gs <- apply(qs, 2, \(.) ps$gene[. < 0.05])
# visualize set intersections between approaches
UpSetR::upset(UpSetR::fromList(gs), order.by="freq")
# extract genes unique to stagewise testing
sw <- grep("stagewise", names(gs))
setdiff(gs[[sw]], unlist(gs[-sw]))
## [1] "OLIG1" "MRPL39" "SLC31A1"
## R version 4.4.2 (2024-10-31)
## Platform: x86_64-pc-linux-gnu
## Running under: Ubuntu 24.04.1 LTS
##
## Matrix products: default
## BLAS: /usr/lib/x86_64-linux-gnu/openblas-pthread/libblas.so.3
## LAPACK: /usr/lib/x86_64-linux-gnu/openblas-pthread/libopenblasp-r0.3.26.so; LAPACK version 3.12.0
##
## locale:
## [1] LC_CTYPE=en_US.UTF-8 LC_NUMERIC=C
## [3] LC_TIME=en_US.UTF-8 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: Etc/UTC
## tzcode source: system (glibc)
##
## attached base packages:
## [1] stats4 stats graphics grDevices utils datasets methods
## [8] base
##
## other attached packages:
## [1] tidyr_1.3.1 UpSetR_1.4.0
## [3] scater_1.35.0 scuttle_1.17.0
## [5] SingleCellExperiment_1.29.1 SummarizedExperiment_1.37.0
## [7] Biobase_2.67.0 GenomicRanges_1.59.1
## [9] GenomeInfoDb_1.43.2 IRanges_2.41.2
## [11] S4Vectors_0.45.2 MatrixGenerics_1.19.0
## [13] matrixStats_1.4.1 ExperimentHub_2.15.0
## [15] AnnotationHub_3.15.0 BiocFileCache_2.15.0
## [17] dbplyr_2.5.0 BiocGenerics_0.53.3
## [19] generics_0.1.3 purrr_1.0.2
## [21] muscat_1.21.0 limma_3.63.2
## [23] ggplot2_3.5.1 dplyr_1.1.4
## [25] cowplot_1.1.3 patchwork_1.3.0
## [27] BiocStyle_2.35.0
##
## loaded via a namespace (and not attached):
## [1] RcppAnnoy_0.0.22 splines_4.4.2 bitops_1.0-9
## [4] filelock_1.0.3 tibble_3.2.1 lifecycle_1.0.4
## [7] Rdpack_2.6.2 edgeR_4.5.1 doParallel_1.0.17
## [10] globals_0.16.3 lattice_0.22-6 MASS_7.3-61
## [13] backports_1.5.0 magrittr_2.0.3 sass_0.4.9
## [16] rmarkdown_2.29 jquerylib_0.1.4 yaml_2.3.10
## [19] sctransform_0.4.1 DBI_1.2.3 buildtools_1.0.0
## [22] minqa_1.2.8 RColorBrewer_1.1-3 abind_1.4-8
## [25] EnvStats_3.0.0 glmmTMB_1.1.10 rappdirs_0.3.3
## [28] circlize_0.4.16 GenomeInfoDbData_1.2.13 ggrepel_0.9.6
## [31] pbkrtest_0.5.3 irlba_2.3.5.1 listenv_0.9.1
## [34] maketools_1.3.1 parallelly_1.41.0 codetools_0.2-20
## [37] DelayedArray_0.33.3 tidyselect_1.2.1 shape_1.4.6.1
## [40] UCSC.utils_1.3.0 farver_2.1.2 lme4_1.1-35.5
## [43] ScaledMatrix_1.15.0 viridis_0.6.5 jsonlite_1.8.9
## [46] GetoptLong_1.0.5 BiocNeighbors_2.1.2 iterators_1.0.14
## [49] foreach_1.5.2 tools_4.4.2 progress_1.2.3
## [52] Rcpp_1.0.13-1 blme_1.0-6 glue_1.8.0
## [55] gridExtra_2.3 SparseArray_1.7.2 xfun_0.49
## [58] mgcv_1.9-1 DESeq2_1.47.1 stageR_1.29.0
## [61] withr_3.0.2 numDeriv_2016.8-1.1 BiocManager_1.30.25
## [64] fastmap_1.2.0 boot_1.3-31 caTools_1.18.3
## [67] digest_0.6.37 rsvd_1.0.5 mime_0.12
## [70] R6_2.5.1 colorspace_2.1-1 Cairo_1.6-2
## [73] gtools_3.9.5 RSQLite_2.3.9 RhpcBLASctl_0.23-42
## [76] variancePartition_1.37.1 data.table_1.16.4 corpcor_1.6.10
## [79] prettyunits_1.2.0 httr_1.4.7 S4Arrays_1.7.1
## [82] uwot_0.2.2 pkgconfig_2.0.3 gtable_0.3.6
## [85] blob_1.2.4 ComplexHeatmap_2.23.0 XVector_0.47.1
## [88] sys_3.4.3 remaCor_0.0.18 htmltools_0.5.8.1
## [91] TMB_1.9.15 clue_0.3-66 scales_1.3.0
## [94] png_0.1-8 fANCOVA_0.6-1 reformulas_0.4.0
## [97] knitr_1.49 reshape2_1.4.4 rjson_0.2.23
## [100] nlme_3.1-166 curl_6.0.1 nloptr_2.1.1
## [103] cachem_1.1.0 GlobalOptions_0.1.2 stringr_1.5.1
## [106] BiocVersion_3.21.1 KernSmooth_2.23-24 parallel_4.4.2
## [109] vipor_0.4.7 AnnotationDbi_1.69.0 pillar_1.10.0
## [112] grid_4.4.2 vctrs_0.6.5 gplots_3.2.0
## [115] BiocSingular_1.23.0 beachmat_2.23.5 cluster_2.1.8
## [118] beeswarm_0.4.0 evaluate_1.0.1 mvtnorm_1.3-2
## [121] cli_3.6.3 locfit_1.5-9.10 compiler_4.4.2
## [124] rlang_1.1.4 crayon_1.5.3 future.apply_1.11.3
## [127] labeling_0.4.3 plyr_1.8.9 ggbeeswarm_0.7.2
## [130] stringi_1.8.4 viridisLite_0.4.2 BiocParallel_1.41.0
## [133] Biostrings_2.75.3 lmerTest_3.1-3 munsell_0.5.1
## [136] aod_1.3.3 Matrix_1.7-1 hms_1.1.3
## [139] bit64_4.5.2 future_1.34.0 KEGGREST_1.47.0
## [142] statmod_1.5.0 rbibutils_2.3 broom_1.0.7
## [145] memoise_2.0.1 bslib_0.8.0 bit_4.5.0.1