Contents

1 tl;dr

To demonstrate, we’ll use one of the mammary gland datasets from the scRNAseq package. We will subset it down to a random set of 500 cells for speed.

library(scRNAseq)
sce <- BachMammaryData(samples="G_2")

set.seed(1000)
sce <- sce[,sample(ncol(sce), 500)]

For the purposes of this demonstration, we’ll perform an extremely expedited analysis. One would usually take more care here and do some quality control, create some diagnostic plots, etc., but we don’t have the space for that.

library(scuttle)
sce <- logNormCounts(sce)

library(scran)
dec <- modelGeneVar(sce)

library(scater)
set.seed(1000)
sce <- runPCA(sce, ncomponents=10, subset_row=getTopHVGs(dec, n=1000))

library(bluster)
clusters <- clusterRows(reducedDim(sce, "PCA"), NNGraphParam())

sce <- runTSNE(sce, dimred="PCA")
plotTSNE(sce, colour_by=I(clusters), text_by=I(clusters))

We then run findDoubletClusters() to test each cluster against the null hypothesis that it does consist of doublets. The null is rejected if a cluster has many DE genes that lie outside the expression limits defined by the “source” clusters. On the other hand, if num.de is low, the cluster’s expression profile is consistent with the doublet hypothesis.

library(scDblFinder)
tab <- findDoubletClusters(sce, clusters)
tab
## DataFrame with 7 rows and 9 columns
##       source1     source2    num.de median.de               best     p.value
##   <character> <character> <integer> <integer>        <character>   <numeric>
## 6           2           1         0       122 ENSMUSG00000019256 1.00000e+00
## 3           6           5         8        46 ENSMUSG00000002985 8.41352e-05
## 7           5           1        11       393 ENSMUSG00000075705 2.04690e-10
## 4           7           2        24        63 ENSMUSG00000022491 4.15645e-13
## 1           7           6        86       715 ENSMUSG00000001349 8.49703e-08
## 5           7           2       112      1007 ENSMUSG00000024610 2.38765e-11
## 2           6           5       124       336 ENSMUSG00000023505 4.06936e-06
##   lib.size1 lib.size2      prop
##   <numeric> <numeric> <numeric>
## 6  0.927081  0.443667     0.084
## 3  2.026090  0.859424     0.242
## 7  1.195040  1.249946     0.030
## 4  0.630479  1.646729     0.150
## 1  0.800035  2.253942     0.192
## 5  0.836792  2.185591     0.032
## 2  1.078655  0.457542     0.270

2 Mathematical background

Consider a cell population \(i\) that has mean transcript count \(\lambda_{gi}\) for gene \(g\). Assume that each population exhibits a unique scaling bias \(s_i\), representing the efficiency of library preparation for that population. The observed read/UMI count for each gene is then \(\mu_{gi}=s_i\lambda_{gi}\). (For simplicity, we will ignore gene-specific scaling biases, as this is easily accommodated by considering \(\lambda_{gi} \equiv \phi_g \lambda_{gi}\) for some bias \(\phi_g\).) The expected total count for each population is \(N_i = \sum_g \mu_{gi}\).

Now, let us consider a doublet population \(j\) that forms from two parent populations \(i_1\) and \(i_2\). The observed read count for \(g\) in \(j\) is \(\mu_{gj} = s_j (\lambda_{gi_1} + \lambda_{gi_2})\). Note that \(s_j\) need not be any particular function of \(s_{i_1}\) and \(s_{i_2}\). Rather, this relationship depends on how quickly the reverse transcription and amplification reagents are saturated during library preparation, which is difficult to make assumptions around.

3 Normalization by library size

We obtain log-normalized expression values for each cell based on the library size. Assume that the library size-normalized expression values are such that \(\mu_{gi_1}N_{i_1}^{-1} < \mu_{gi_2}N_{i_2}^{-1}\), i.e., the proportion of \(g\) increases in \(i_2\) compared to \(i_1\). The contribution of each \(s_i\) cancels out, yielding \[ \frac{\lambda_{gi_1}}{\sum_g \lambda_{gi_1}} < \frac{\lambda_{gi_2}}{\sum_g \lambda_{gi_2}} \;. \] The normalized expression value of the doublet cluster \(j\) is subsequently \[ \frac{\lambda_{gi_1} + \lambda_{gi_2}}{\sum_g (\lambda_{gi_1} + \lambda_{gi_2})} \;, \] and it is fairly easy to show that \[ \frac{\lambda_{gi_1}}{\sum_g \lambda_{gi_1}} < \frac{\lambda_{gi_1} + \lambda_{gi_2}}{\sum_g (\lambda_{gi_1} + \lambda_{gi_2})} < \frac{\lambda_{gi_2}}{\sum_g \lambda_{gi_2}} \;. \] In other words, the expected library size-normalized expression of our gene in the doublet cluster lies between that of the two parents.

It is harder to provide theoretical guarantees with arbitrary size factors, which is why we only use the library sizes for normalization instead. The exception is that of spike-in size factors that would estimate \(s_i\) directly. This would allow us to obtain estimates of \(\lambda_{gi}\) for the parent clusters and of \(\lambda_{gi_1} + \lambda_{gi_2}\) for the doublets. In this manner, we could more precisely identify doublet clusters as those where the normalized expression value is equal to the sum of the parents. Unfortunately, spike-ins are generally not available for droplet-based data sets where doublets are most problematic.

4 Testing for (lack of) intermediacy

We want to identify the clusters that may be comprised of doublets of other clusters. For each cluster \(j'\), we test for differential expression in the library size-normalized expression profiles against every other cluster \(i'\). For each pair of other clusters \(i'_1\) and \(i'_2\), we identify genes that change in \(j'\) against both \(i'_1\) and \(i'_2\) in the same direction. The presence of such genes violates the intermediacy expected of a doublet cluster and provides evidence that \(j'\) is not a doublet of \(i'_1\) and \(i'_2\).

Significant genes are identified by an intersection-union test on the \(p\)-values from the pairwise comparisons between \(j'\) and \(i'_1\) or \(i'_2\). (Specifically, \(t\)-tests are used via the findMarkers() function from scran.) The \(p\)-value for a gene is set to unity when the signs of the log-fold changes are not the same between comparisons. Multiple correction testing is applied using the Benjamini-Hochberg method, and the number of genes detected at a specified false discovery rate (usually 5%) is counted. The pair \((i'_1, i'_2)\) with the fewest detected genes are considered as the putative parents of \(j'\).

In theory, it is possible to compute the Simes’ combined \(p\)-value across all genes to reject the doublet hypothesis for \(j'\). This would provide a more rigorous approach to ruling out potential doublet/parent combinations. However, this is very sensitive to misspecification of clusters – see below.

5 Calling doublet clusters

Assuming that most clusters are not comprised of doublets, we identify clusters that have an unusually low number of detected genes that violate the intermediacy condition. This is achieved by identifying small outliers on the log-transformed number of detected genes, using the median absolute deviation-based method in the function. (We use a log-transformation simply to improve resolution at low values.) Clusters are likely to be doublets if they are outliers on this metric.

Doublet clusters should also have larger library sizes than the proposed parent clusters. This is consistent with the presence of more RNA in each doublet, though the library size of the doublet cluster need not be a sum of that of the parent clusters (due to factors such as saturation and composition effects). The proportion of cells assigned to the doublet cluster should also be “reasonable”; exactly what this means depends on the experimental setup and the doublet rate of the protocol in use.

6 Discussion

The biggest advantage of this approach lies in its interpretability. Given a set of existing clusters, we can explicitly identify those that are likely to be doublets. We also gain some insight onto the parental origins of each putative doublet cluster, which may be of some interest. We avoid any assumptions about doublet formation that are otherwise necessary for the simulation-based methods. In particular, we do not require any knowledge about exact the relationship between \(s_j\) and \(s_i\), allowing us to identify doublets even when the exact location of the doublet is unknown (e.g., due to differences in RNA content between the parent clusters).

The downside is that, of course, we are dependent on being supplied with sensible clusters where the parental and doublet cells are separated. The intermediacy requirement is loose enough to provide some robustness against misspecification, but this only goes so far. In addition, this strategy has a bias towards calling clusters with few cells as doublets (or parents of doublets) because the DE detection power is low. This can be somewhat offset by comparing num.de against median.de as latter will be low for clusters involved in systematically low-powered comparisons, though it is difficult to adjust for the exact effect of the differences of power on the IUT.

Session information

sessionInfo()
## R version 4.5.1 Patched (2025-08-23 r88802)
## Platform: x86_64-pc-linux-gnu
## Running under: Ubuntu 24.04.3 LTS
## 
## Matrix products: default
## BLAS:   /home/biocbuild/bbs-3.22-bioc/R/lib/libRblas.so 
## LAPACK: /usr/lib/x86_64-linux-gnu/lapack/liblapack.so.3.12.0  LAPACK version 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] bluster_1.19.1              scDblFinder_1.23.9         
##  [3] scater_1.37.0               ggplot2_4.0.0              
##  [5] scran_1.37.0                scuttle_1.19.0             
##  [7] ensembldb_2.33.2            AnnotationFilter_1.33.0    
##  [9] GenomicFeatures_1.61.7      AnnotationDbi_1.71.2       
## [11] scRNAseq_2.23.1             SingleCellExperiment_1.31.1
## [13] SummarizedExperiment_1.39.2 Biobase_2.69.1             
## [15] GenomicRanges_1.61.6        Seqinfo_0.99.3             
## [17] IRanges_2.43.5              S4Vectors_0.47.4           
## [19] BiocGenerics_0.55.4         generics_0.1.4             
## [21] MatrixGenerics_1.21.0       matrixStats_1.5.0          
## [23] BiocStyle_2.37.1           
## 
## loaded via a namespace (and not attached):
##   [1] RColorBrewer_1.1-3       jsonlite_2.0.0           magrittr_2.0.4          
##   [4] magick_2.9.0             ggbeeswarm_0.7.2         gypsum_1.5.0            
##   [7] farver_2.1.2             rmarkdown_2.30           BiocIO_1.19.0           
##  [10] vctrs_0.6.5              memoise_2.0.1            Rsamtools_2.25.3        
##  [13] RCurl_1.98-1.17          tinytex_0.57             htmltools_0.5.8.1       
##  [16] S4Arrays_1.9.1           AnnotationHub_3.99.6     curl_7.0.0              
##  [19] BiocNeighbors_2.3.1      xgboost_1.7.11.1         Rhdf5lib_1.31.1         
##  [22] SparseArray_1.9.1        rhdf5_2.53.6             sass_0.4.10             
##  [25] alabaster.base_1.9.5     bslib_0.9.0              alabaster.sce_1.9.0     
##  [28] httr2_1.2.1              cachem_1.1.0             GenomicAlignments_1.45.4
##  [31] igraph_2.2.0             lifecycle_1.0.4          pkgconfig_2.0.3         
##  [34] rsvd_1.0.5               Matrix_1.7-4             R6_2.6.1                
##  [37] fastmap_1.2.0            digest_0.6.37            dqrng_0.4.1             
##  [40] irlba_2.3.5.1            ExperimentHub_2.99.6     RSQLite_2.4.3           
##  [43] beachmat_2.25.5          labeling_0.4.3           filelock_1.0.3          
##  [46] httr_1.4.7               abind_1.4-8              compiler_4.5.1          
##  [49] bit64_4.6.0-1            withr_3.0.2              S7_0.2.0                
##  [52] BiocParallel_1.43.4      viridis_0.6.5            DBI_1.2.3               
##  [55] HDF5Array_1.37.0         alabaster.ranges_1.9.1   alabaster.schemas_1.9.0 
##  [58] MASS_7.3-65              rappdirs_0.3.3           DelayedArray_0.35.3     
##  [61] rjson_0.2.23             tools_4.5.1              vipor_0.4.7             
##  [64] beeswarm_0.4.0           glue_1.8.0               h5mread_1.1.1           
##  [67] restfulr_0.0.16          rhdf5filters_1.21.4      grid_4.5.1              
##  [70] Rtsne_0.17               cluster_2.1.8.1          gtable_0.3.6            
##  [73] data.table_1.17.8        BiocSingular_1.25.0      ScaledMatrix_1.17.0     
##  [76] metapod_1.17.0           XVector_0.49.1           ggrepel_0.9.6           
##  [79] BiocVersion_3.22.0       pillar_1.11.1            limma_3.65.7            
##  [82] dplyr_1.1.4              BiocFileCache_2.99.6     lattice_0.22-7          
##  [85] rtracklayer_1.69.1       bit_4.6.0                tidyselect_1.2.1        
##  [88] locfit_1.5-9.12          Biostrings_2.77.2        knitr_1.50              
##  [91] gridExtra_2.3            bookdown_0.45            ProtGenerics_1.41.0     
##  [94] edgeR_4.7.6              xfun_0.53                statmod_1.5.1           
##  [97] UCSC.utils_1.5.0         lazyeval_0.2.2           yaml_2.3.10             
## [100] evaluate_1.0.5           codetools_0.2-20         tibble_3.3.0            
## [103] alabaster.matrix_1.9.0   BiocManager_1.30.26      cli_3.6.5               
## [106] jquerylib_0.1.4          dichromat_2.0-0.1        Rcpp_1.1.0              
## [109] GenomeInfoDb_1.45.12     dbplyr_2.5.1             png_0.1-8               
## [112] XML_3.99-0.19            parallel_4.5.1           blob_1.2.4              
## [115] bitops_1.0-9             viridisLite_0.4.2        alabaster.se_1.9.0      
## [118] scales_1.4.0             purrr_1.1.0              crayon_1.5.3            
## [121] rlang_1.1.6              cowplot_1.2.0            KEGGREST_1.49.2