Let the vector \({\bf p}\) be the true fractions across \(D\) categories. Consider \(C\) total counts sampled from a Dirichlet-multinomial (DMN) distribution with overdispersion \(\tau\), where \(\tau=1\) reduces to the multinomial distribution. The centered log ratio (CLR) of the \(i^{th}\) estimated fraction in \({\bf \hat p}\) is
\[\begin{equation} \tag{1} \text{clr}_i({\bf \hat p}) = \log(\hat p_i) - \frac{1}{D}\sum_{j=1}^D \log(\hat p_j) \end{equation}\] and we show that the sampling variance is
\[\begin{equation} \tag{2} \text{var}[\text{clr}_i({\bf \hat p})] = \frac{\tau}{C} \left[ \frac{1}{\hat p_i} - \frac{2}{ D \hat p_i} + \frac{1}{D^2}\sum_{j=1}^D \frac{1}{\hat p_j} \right]. \label{eqn2} \end{equation}\]
The sampling variance is derived from asymototic theory, so we examine its behavior for finite total counts. Here we evaluate the empirical variance from \(1,000\) draws from a Dirichlet-multinomial distribution while varying \(D\), \(\tau\), \(C\). A pseudocount of 0.5 is added to the observed counts since the asymptotic theory is not defined for counts of zero.
Here we plot the standard deviation after CLR transform from the empirical DMN and the asymptotic normal approximation under a range of conditions. Results are shown for instances with at least 2 counts.
The asymptotic standard deviation shows good agreement with the empirical results even for small values of \(C\), when at least 2 counts are observed. In practice, it is often reasonable to assume a sufficient number of counts before a variable is included in an analysis. Importantly, with less than 2 counts the asymptotic theory gives a larger standard deviation than the emprical results (results not shown). Therefore, this approach is conservative and should not underestimate the true amount of variation. The asymptotic normal approximation is most accurate for large total counts \(C\), large proportions \(p\), and small overdispersion \(\tau\).
Based on Equation (2), the variance of the CLR-transformed proportions is a linear function of \(\tau\). Importantly, downstream analysis of the CLR-transformed proportions with a precision-weighted linear (mixed) model or a variance stabilizing transform depends only on the relative variances. Since relative variances are invariant to the scale of \(\tau\), for these applications the value of \(\tau\) can be set to 1 instead of being estimated from the data.
For other applications, crumblr can estimate \(\tau\) from the data by using crumblr(counts, tau=NULL). This calls dmn.mle() to estimate the parameters of the DMN distribution and is substantially faster than alternatives.
## R version 4.5.0 RC (2025-04-04 r88126)
## Platform: x86_64-pc-linux-gnu
## Running under: Ubuntu 24.04.2 LTS
## 
## Matrix products: default
## BLAS:   /home/biocbuild/bbs-3.21-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               LC_TIME=en_GB             
##  [4] LC_COLLATE=C               LC_MONETARY=en_US.UTF-8    LC_MESSAGES=en_US.UTF-8   
##  [7] LC_PAPER=en_US.UTF-8       LC_NAME=C                  LC_ADDRESS=C              
## [10] LC_TELEPHONE=C             LC_MEASUREMENT=en_US.UTF-8 LC_IDENTIFICATION=C       
## 
## time zone: America/New_York
## tzcode source: system (glibc)
## 
## attached base packages:
## [1] parallel  stats     graphics  grDevices utils     datasets  methods   base     
## 
## other attached packages:
##  [1] lubridate_1.9.4          forcats_1.0.0            stringr_1.5.1           
##  [4] dplyr_1.1.4              purrr_1.0.4              readr_2.1.5             
##  [7] tidyr_1.3.1              tibble_3.2.1             tidyverse_2.0.0         
## [10] glue_1.8.0               HMP_2.0.1                dirmult_0.1.3-5         
## [13] variancePartition_1.38.0 BiocParallel_1.42.0      limma_3.64.0            
## [16] crumblr_1.0.0            ggplot2_3.5.2            BiocStyle_2.36.0        
## 
## loaded via a namespace (and not attached):
##   [1] jsonlite_2.0.0              magrittr_2.0.3              magick_2.8.6               
##   [4] farver_2.1.2                nloptr_2.2.1                rmarkdown_2.29             
##   [7] fs_1.6.6                    vctrs_0.6.5                 minqa_1.2.8                
##  [10] ggtree_3.16.0               tinytex_0.57                htmltools_0.5.8.1          
##  [13] S4Arrays_1.8.0              broom_1.0.8                 SparseArray_1.8.0          
##  [16] gridGraphics_0.5-1          sass_0.4.10                 KernSmooth_2.23-26         
##  [19] bslib_0.9.0                 pbkrtest_0.5.3              plyr_1.8.9                 
##  [22] cachem_1.1.0                lifecycle_1.0.4             iterators_1.0.14           
##  [25] pkgconfig_2.0.3             Matrix_1.7-3                R6_2.6.1                   
##  [28] fastmap_1.2.0               GenomeInfoDbData_1.2.14     rbibutils_2.3              
##  [31] MatrixGenerics_1.20.0       digest_0.6.37               numDeriv_2016.8-1.1        
##  [34] aplot_0.2.5                 colorspace_2.1-1            patchwork_1.3.0            
##  [37] S4Vectors_0.46.0            GenomicRanges_1.60.0        vegan_2.6-10               
##  [40] labeling_0.4.3              timechange_0.3.0            mgcv_1.9-3                 
##  [43] httr_1.4.7                  abind_1.4-8                 compiler_4.5.0             
##  [46] aod_1.3.3                   withr_3.0.2                 backports_1.5.0            
##  [49] viridis_0.6.5               gplots_3.2.0                MASS_7.3-65                
##  [52] DelayedArray_0.34.0         corpcor_1.6.10              gtools_3.9.5               
##  [55] caTools_1.18.3              permute_0.9-7               tools_4.5.0                
##  [58] ape_5.8-1                   remaCor_0.0.18              nlme_3.1-168               
##  [61] grid_4.5.0                  cluster_2.1.8.1             reshape2_1.4.4             
##  [64] generics_0.1.3              gtable_0.3.6                tzdb_0.5.0                 
##  [67] hms_1.1.3                   XVector_0.48.0              BiocGenerics_0.54.0        
##  [70] foreach_1.5.2               pillar_1.10.2               yulab.utils_0.2.0          
##  [73] splines_4.5.0               treeio_1.32.0               lattice_0.22-7             
##  [76] tidyselect_1.2.1            SingleCellExperiment_1.30.0 knitr_1.50                 
##  [79] reformulas_0.4.0            gridExtra_2.3               bookdown_0.43              
##  [82] IRanges_2.42.0              SummarizedExperiment_1.38.0 RhpcBLASctl_0.23-42        
##  [85] stats4_4.5.0                xfun_0.52                   Biobase_2.68.0             
##  [88] statmod_1.5.0               matrixStats_1.5.0           stringi_1.8.7              
##  [91] UCSC.utils_1.4.0            lazyeval_0.2.2              ggfun_0.1.8                
##  [94] yaml_2.3.10                 boot_1.3-31                 evaluate_1.0.3             
##  [97] codetools_0.2-20            rpart.plot_3.1.2            BiocManager_1.30.25        
## [100] ggplotify_0.1.2             cli_3.6.4                   rpart_4.1.24               
## [103] RcppParallel_5.1.10         Rdpack_2.6.4                munsell_0.5.1              
## [106] jquerylib_0.1.4             Rcpp_1.0.14                 GenomeInfoDb_1.44.0        
## [109] zigg_0.0.2                  EnvStats_3.0.0              Rfast_2.1.5.1              
## [112] bitops_1.0-9                lme4_1.1-37                 viridisLite_0.4.2          
## [115] mvtnorm_1.3-3               tidytree_0.4.6              lmerTest_3.1-3             
## [118] scales_1.3.0                crayon_1.5.3                fANCOVA_0.6-1              
## [121] rlang_1.1.6