Last updated: 2019-04-01
workflowr checks: (Click a bullet for more information) ✔ R Markdown file: up-to-date
Great! Since the R Markdown file has been committed to the Git repository, you know the exact version of the code that produced these results.
✔ Environment: empty
Great job! The global environment was empty. Objects defined in the global environment can affect the analysis in your R Markdown file in unknown ways. For reproduciblity it’s best to always run the code in an empty environment.
✔ Seed:
set.seed(20181103)
The command set.seed(20181103)
was run prior to running the code in the R Markdown file. Setting a seed ensures that any results that rely on randomness, e.g. subsampling or permutations, are reproducible.
✔ Session information: recorded
Great job! Recording the operating system, R version, and package versions is critical for reproducibility.
✔ Repository version: 57d374a
wflow_publish
or wflow_git_commit
). workflowr only checks the R Markdown file, but you know if there are other scripts or data files that it depends on. Below is the status of the Git repository when the results were generated:
Ignored files:
Ignored: .DS_Store
Ignored: .Rhistory
Ignored: .Rproj.user/
Ignored: data/GSE103354_Trachea_droplet_UMIcounts.txt
Ignored: output/.DS_Store
Ignored: output/sc_comparisons/.DS_Store
Unstaged changes:
Modified: code/sc_comparisons.R
Modified: output/sc_comparisons/allres.rds
Note that any generated files, e.g. HTML, png, CSS, etc., are not included in this status report because it is ok for generated content to have uncommitted changes.
File | Version | Author | Date | Message |
---|---|---|---|---|
Rmd | 57d374a | Jason Willwerscheid | 2019-04-01 | wflow_publish(“analysis/sc_comparisons.Rmd”) |
html | cb2c9c5 | Jason Willwerscheid | 2019-03-25 | Build site. |
Rmd | d4e8255 | Jason Willwerscheid | 2019-03-25 | wflow_publish(“analysis/sc_comparisons.Rmd”) |
html | 60843e0 | Jason Willwerscheid | 2019-03-22 | Build site. |
Rmd | b792cfe | Jason Willwerscheid | 2019-03-22 | wflow_publish(“analysis/sc_comparisons.Rmd”) |
Here I compare various methods for fitting flash objects to single-cell data. I use the drop-seq dataset discussed in Montoro et al., which includes counts for approximately 18000 genes and 7000 cells. Close to 90% of counts are equal to zero.
I perform 50 trials. For each trial, I subsample 200 cells, then I subsample 500 genes from the set of genes that have at least 3 non-zero counts among the subsampled cells. I remove 1000 entries at random from the resulting data matrix, fit a series of flash objects, and calculate mean squared errors, using flash posterior means as fitted values and the original data as ground truth. I compare different variance structures, data transformations, scaling methods, priors, and numbers of factors.
Additionally, to paint a more granular picture of how additional factors affect mean squared error, I incrementally fit “greedy” flash objects with 1 to 60 factors using the full drop-seq dataset (not just a subsample).
Click “Code” to view the code used to run the tests.
# library(ashr) -- uncomment after negative mixture proportions bug is fixed
devtools::load_all("~/Github/ashr")
# library(flashier) -- uncomment after pushing 0.1.1 updates to master
devtools::load_all("~/Github/flashier")
library(Matrix)
library(ggplot2)
# Load data ---------------------------------------------------------------
# The drop-seq dataset in Montoro et al. can be downloaded here:
# https://www.ncbi.nlm.nih.gov/geo/query/acc.cgi?acc=GSE103354
trachea <- read.table("./data/GSE103354_Trachea_droplet_UMIcounts.txt")
trachea <- as.matrix(trachea)
trachea <- Matrix(trachea)
# Set test parameters -----------------------------------------------------
# Number of times to subsample data and run fits.
ntrials <- 50
# Size of subsampled data matrices.
ncells <- 200
ngenes <- 500
# Minimum number of nonzero counts needed to include a gene.
min.cts <- 3
# Proportion of entries to impute.
prop.missing <- 0.01
nmissing <- ceiling(prop.missing * ncells * ngenes)
# Number of factors to add to each fit.
K <- 5
# Verbose flag for flashier output.
verbose <- FALSE
# Set up data frames ------------------------------------------------------
create.df <- function(names) {
names <- paste0(names, ".")
df.names <- outer(names, c("mse", "elbo"), FUN = paste0)
df <- data.frame(rep(list(numeric(0)), length(df.names)))
names(df) <- t(df.names)
return(df)
}
# Test 1: variance structure.
var.df <- create.df(c("constant", "genewise", "fixed", "noisyA", "noisyB"))
# Test 2: data transformation.
trans.df <- create.df(c("log1p", "anscombe", "arcsin", "raw", "pearson"))
# Test 3: normalization method.
norm.df <- create.df(c("none", "fitmean", "scale"))
# Test 4: prior type.
prior.df <- create.df(c("normal", "nngeneA", "nngeneB", "nncellA", "nncellB"))
# Test 5: number of factors and backfit.
nfactors.df <- create.df(c("g1", "bf1", "g2", "bf2", "g3", "bf3"))
# Functions for populating data frames ------------------------------------
get.fitted <- function(fl) {
# Get fitted values on the log1p scale.
fitted.vals <- flashier:::lowrank.expand(fl$fit$EF)
# Convert to raw counts.
return(exp(fitted.vals) - 1)
}
log1p.mse <- function(true, preds) {
# Threshold out negative counts.
preds <- pmax(preds, 0)
return(mean((log1p(true) - log1p(preds))^2))
}
get.elbo <- function(fl, adjustment) {
return(fl$objective + adjustment)
}
all.metrics <- function(data, missing.vals,
fl, fitted.vals, elbo.adjustment) {
missing.idx <- which(is.na(data))
if (is.null(fitted.vals)) {
fitted.vals <- get.fitted(fl)
}
return(list(mse = log1p.mse(missing.vals, fitted.vals[missing.idx]),
elbo = get.elbo(fl, elbo.adjustment)))
}
get.df.row <- function(data, missing.vals,
fl.list,
mean.factors = NULL,
fitted.vals = NULL,
elbo.adjustment = NULL) {
if (is.null(fitted.vals)) {
fitted.vals = rep(list(NULL), length(fl.list))
}
if (is.null(elbo.adjustment)) {
elbo.adjustment = rep(list(0), length(fl.list))
}
return(unlist(mapply(all.metrics,
fl = fl.list,
fitted.vals = fitted.vals,
elbo.adjustment = elbo.adjustment,
MoreArgs = list(data = data,
missing.vals = missing.vals))))
}
# Run tests ---------------------------------------------------------------
for (i in 1:ntrials) {
cat("TRIAL", i, "\n")
set.seed(i)
rand.cells <- sample(1:ncol(trachea), ncells)
samp <- trachea[, rand.cells]
rand.genes <- sample(which(rowSums(samp > 0) >= min.cts), ngenes)
samp <- samp[rand.genes, ]
missing.idx <- sample(1:length(samp), nmissing)
missing.vals <- samp[missing.idx]
samp[missing.idx] <- NA
# Test 1: variance structure.
cat(" Running variance structure tests.\n")
fl.var0 <- flashier(log1p(samp), var.type = 0,
prior.type = "normal.mix",
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
fl.var1 <- flashier(log1p(samp), var.type = 1,
prior.type = "normal.mix",
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
S <- sqrt(samp) / (samp + 1)
nz.prop <- sum(samp > 0, na.rm = TRUE) / (length(samp) - length(missing.idx))
S[S == 0] <- sqrt(nz.prop) / (nz.prop + 1)
fl.fixS <- flashier(log1p(samp), S = S,
var.type = NULL,
prior.type = "normal.mix",
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
suppressMessages({
fl.noisyA <- flashier(log1p(samp), S = S,
var.type = 0,
prior.type = "normal.mix",
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
})
suppressMessages({
fl.noisyB <- flashier(log1p(samp), S = sqrt(samp) / (samp + 1),
var.type = 0,
prior.type = "normal.mix",
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
})
var.df[i, ] <- get.df.row(samp, missing.vals,
fl.list = list(fl.var0, fl.var1, fl.fixS,
fl.noisyA, fl.noisyB))
# Test 2: data transformation.
cat(" Running data transformation tests.\n")
fl.log1p <- fl.var0
log1p.adj <- -sum(log1p(samp), na.rm = TRUE)
fl.ans <- flashier(sqrt(samp + 0.375), var.type = 0,
prior.type = "normal.mix",
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
ans.fitted <- flashier:::lowrank.expand(fl.ans$fit$EF)^2 - 0.375
ans.adj <- -sum(log(2) + 0.5 * log(samp + 0.375), na.rm = TRUE)
cell.sums <- colSums(samp, na.rm = TRUE)
props <- samp / rep(cell.sums, each = nrow(samp))
fl.arcsin <- flashier(asin(sqrt(props)), var.type = 0,
prior.type = "normal.mix",
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
arcsin.fitted <- (sin(flashier:::lowrank.expand(fl.arcsin$fit$EF)^2)
* rep(cell.sums, each = nrow(samp)))
arcsin.adj <- NA
fl.raw <- flashier(props, var.type = 0,
prior.type = "normal.mix",
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
raw.fitted <- (flashier:::lowrank.expand(fl.raw$fit$EF)
* rep(cell.sums, each = nrow(samp)))
raw.adj <- -sum(rep(log(cell.sums), each = nrow(samp))[-missing.idx])
gene.props <- rowSums(samp, na.rm = TRUE) / sum(samp, na.rm = TRUE)
mu <- outer(gene.props, cell.sums)
sd.mat <- sqrt(mu - mu^2 / rep(cell.sums, each = nrow(samp)))
resid <- (samp - mu) / sd.mat
fl.pearson <- flashier(resid, var.type = 0,
prior.type = "normal.mix",
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
pearson.fitted <- mu + sd.mat * flashier:::lowrank.expand(fl.pearson$fit$EF)
pearson.adj <- -sum(log(sd.mat)[-missing.idx])
trans.df[i, ] <- get.df.row(samp, missing.vals,
fl.list = list(fl.log1p, fl.ans, fl.arcsin,
fl.raw, fl.pearson),
fitted.vals = list(get.fitted(fl.log1p),
ans.fitted, arcsin.fitted,
raw.fitted, pearson.fitted),
elbo.adjustment = list(log1p.adj,
ans.adj, arcsin.adj,
raw.adj, pearson.adj))
# Test 3: scaling method.
cat(" Running scaling tests.\n")
fl.none <- fl.var0
none.adj <- log1p.adj
fl.ones <- flashier(log1p(samp), var.type = 0,
prior.type = "normal.mix",
fixed.factors = c(ones.factor(2), ones.factor(1)),
greedy.Kmax = K,
backfit.after = 2, final.backfit = FALSE,
verbose.lvl = 1L * verbose)
ones.adj <- log1p.adj
scaled.samp <- samp * median(cell.sums) / rep(cell.sums, each = nrow(samp))
fl.scale <- flashier(log1p(scaled.samp), var.type = 0,
prior.type = "normal.mix",
fixed.factors = ones.factor(2),
greedy.Kmax = K, backfit = "none",
verbose.lvl = 1L * verbose)
scale.fitted <- (get.fitted(fl.scale)
* rep(cell.sums, each = nrow(samp)) / median(cell.sums))
scale.adj <- sum(log(median(cell.sums)) - rep(log(cell.sums), each = nrow(samp))
- log1p(scaled.samp), na.rm = TRUE)
norm.df[i, ] <- get.df.row(samp, missing.vals,
fl.list = list(fl.none, fl.ones, fl.scale),
fitted.vals = list(get.fitted(fl.none),
get.fitted(fl.ones),
scale.fitted),
elbo.adjustment = list(none.adj, ones.adj,
scale.adj))
# Test 4: prior type.
cat(" Running prior type tests.\n")
fl.normalmix <- fl.var0
fl.nngenes <- flashier(log1p(samp), var.type = 0,
prior.type = c("nonnegative", "normal.mix"),
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
fl.nngenes.pm <- flashier(log1p(samp), var.type = 0,
prior.type = c("nonnegative", "normal.mix"),
ash.param = list(method = "fdr"),
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
fl.nncells <- flashier(log1p(samp), var.type = 0,
prior.type = c("normal.mix", "nonnegative"),
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
fl.nncells.pm <- flashier(log1p(samp), var.type = 0,
prior.type = c("normal.mix", "nonnegative"),
ash.param = list(method = "fdr"),
greedy.Kmax = K + 1, backfit = "none",
verbose.lvl = 1L * verbose)
prior.df[i, ] <- get.df.row(samp, missing.vals,
fl.list = list(fl.normalmix,
fl.nngenes, fl.nngenes.pm,
fl.nncells, fl.nncells.pm))
# Test 5: number of factors and backfit.
cat(" Running backfitting tests.\n")
fl.g <- fl.var0
fl.b <- flashier(flash.init = fl.g, backfit = "only",
backfit.reltol = 10,
verbose.lvl = 3L * verbose)
# Add K more factors, then K more.
fl.g2 <- flashier(flash.init = fl.g,
greedy.Kmax = K, backfit = "none",
verbose.lvl = 1L * verbose)
fl.b2 <- flashier(flash.init = fl.b,
greedy.Kmax = K, backfit = "final",
backfit.reltol = 10,
verbose.lvl = 3L * verbose)
fl.g3 <- flashier(flash.init = fl.g2,
greedy.Kmax = K, backfit = "none",
verbose.lvl = 1L * verbose)
fl.b3 <- flashier(flash.init = fl.b2,
greedy.Kmax = K, backfit = "final",
backfit.reltol = 10,
verbose.lvl = 3L * verbose)
nfactors.df[i, ] <- get.df.row(samp, missing.vals,
fl.list = list(fl.g, fl.b,
fl.g2, fl.b2,
fl.g3, fl.b3))
}
# Test 6: incremental addition of factors
set.seed(666)
data <- log1p(trachea[rowSums(trachea > 0) >= min.cts, ])
missing.idx <- sample(1:length(data), ceiling(prop.missing * length(data)))
true.vals <- data[missing.idx]
data[missing.idx] <- NA
Kmax <- 60
fl <- flashier(data, var.type = 0,
prior.type = "normal.mix",
fixed.factors = c(ones.factor(1), ones.factor(2)),
greedy.Kmax = 1, backfit = "none",
final.nullchk = FALSE,
verbose.lvl = 3)
mse.vec <- mse(true.vals, preds(fl, missing.idx), 1:length(true.vals))
for (k in 2:Kmax) {
fl <- flashier(flash.init = fl,
greedy.Kmax = 1, backfit = "none",
final.nullchk = FALSE,
verbose.lvl = 3)
mse.vec <- c(mse.vec,
mse(true.vals, preds(fl, missing.idx), 1:length(true.vals)))
}
mse.diff <- c(NA, mse.vec[2:length(mse.vec)] - mse.vec[1:(length(mse.vec) - 1)])
mse.df <- data.frame(k = 1:Kmax, mse = mse.vec, mse.diff = mse.diff)
# Save results ------------------------------------------------------------
all.res <- list(var.df = var.df,
trans.df = trans.df,
norm.df = norm.df,
prior.df = prior.df,
nfactors.df = nfactors.df,
mse.df = mse.df)
saveRDS(all.res, "./output/sc_comparisons/allres.rds")
allres <- readRDS("./output/sc_comparisons/allres.rds")
For each comparison, I choose a “reference” method and plot the difference in mean squared error for each other method over all 50 trials. A positive value indicates worse performance. Importantly, mean squared error is calculated on the log1p scale in all cases.
Note that the violin plots all have the same y-axis limits (from -0.025 to +0.1). This is to underscore the relative importance of each aspect. For example, the scrunched-up appearance of the scaling methods plot is intentional: as it turns out, the scaling method used does not affect the mean squared error nearly as much as, say, the variance structure. In several cases, there are a few outlying fits with very large MSEs. When this occurs, I display all trials in addition to displaying a “zoom” view with y-axis limits from -0.025 to +0.1.
The red horizontal lines indicate the median and the 10% and 90% quantiles (unfortunately, ggplot2
ignores any outlying data points when calculating these quantities).
For reference, I also plot (without comment) differences in ELBO, adjusting via a change-of-variables formula where necessary. In principle, a higher ELBO should indicate a better fit.
The most commonly used transformations of single-cell data are log transforms with pseudocounts. Throughout these tests, my default procedure is to transform the raw counts using the log1p transform: \[ Y_{ij} = \log \left( X_{ij} + 1 \right) \]
Thus, the model that flashier
fits is \[ \log \left( X_{ij} + 1 \right) = \sum_k L_{ik} F_{jk} + E_{ij}, \] with \[E_{ij} \sim N(0, \sigma_{ij}^2)\]
If one disregards sampling error, one can simply fit a constant variance structure \(\sigma_{ij}^2 = \sigma^2\) or a gene-wise variance structure \(\sigma_{ij}^2 = \sigma_j^2\), with \(\sigma^2\) (viz. \(\sigma_j^2\)) to be estimated.
However, if (as is reasonable to assume) the data is Poisson for some true expression levels \(\lambda_{ij}\): \[ X_{ij} \sim \text{Poisson} (\lambda_{ij}), \] then sampling errors will be much different for large and small counts. Expanding around the MLE \(\hat{\lambda}_{ij} = X_{ij}\): \[ \log (\lambda_{ij} + 1) \approx \log(X_{ij} + 1) + \frac{\lambda_{ij} - X_{ij}}{X_{ij} + 1} \] so that \[ \text{Var} \left( \log (\lambda_{ij} + 1) \right) \approx \frac{\lambda_{ij}}{(X_{ij} + 1)^2} \approx \frac{X_{ij}}{(X_{ij} + 1)^2}\] Ignoring approximation error, one can thus fix the standard errors at \[ S_{ij} = \frac{\sqrt{X_{ij}}}{X_{ij} + 1} \] Unfortunately, this sets \(S_{ij} = 0\) when \(X_{ij} = 0\). To circumvent this problem, I replace zeros with a very rough estimate of the baseline Poisson noise: \[ \lambda_0 := \frac{\#\{X_{ij}: X_{ij} \ne 0\}}{\#\{X_{ij}\}} \] That is, when \(X_{ij} = 0\), I set \[ S_{ij} = \frac{\sqrt{\lambda_0}}{\lambda_0 + 1} \]
Using flashier
, one can also fit a model that includes both fixed sampling errors and an approximation error to be estimated, so that \[E_{ij} \sim N(0, S_{ij}^2 + \sigma^2)\] I try two approaches. One fixes the standard errors as described above, replacing \(S_{ij} = 0\) with \(S_{ij} = \sqrt{\lambda_0} / (\lambda_0 + 1)\). The other leaves standard errors of zero as they are and hopes that the estimated \(\sigma^2\) will be able to make up the difference. In both cases, the estimated \(\sigma^2\) is constant across genes and cells (estimating a noisy gene-wise variance structure is at present far too slow for large datasets).
library(ggplot2)
get.plot.df <- function(df, ref.lvl, type) {
tests <- colnames(df)
category <- sapply(strsplit(tests, '[.]'), `[`, 1)
metric <- sapply(strsplit(tests, '[.]'), `[`, 2)
df <- df[, which(metric == type)]
category <- category[which(metric == type)]
which.ref <- which(category == ref.lvl)
df <- df - df[, which.ref]
df <- df[, -which.ref]
category <- category[-which.ref]
return(data.frame(value = as.vector(as.matrix(df)),
category = rep(category, each = nrow(df))))
}
var.mse <- get.plot.df(allres$var.df, ref.lvl = "constant", type = "mse")
ggplot(var.mse, aes(x = category, y = value)) +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in MSE",
title = "Variance structures, MSE (reference: Constant)") +
scale_x_discrete(limits = c("genewise", "fixed", "noisyA", "noisyB"),
labels = c("Gene-wise", "Fixed", "Noisy (no zero SEs)",
"Noisy (zero SEs)"))
ggplot(var.mse, aes(x = category, y = value)) +
geom_violin(adjust = 1.5,
draw_quantiles = c(0.1, 0.5, 0.9),
color = "red") +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in MSE",
title = "Variance structures, MSE (zoom)") +
scale_x_discrete(limits = c("genewise", "fixed", "noisyA", "noisyB"),
labels = c("Gene-wise", "Fixed", "Noisy (no zero SEs)",
"Noisy (zero SEs)")) +
ylim(-0.1, 0.2)
The constant variance structure clearly works best. A gene-wise variance structure can do as well (and occasionally better), but also results in some very poor fits. I’ve found that a gene-wise variance structure often estimates the variances for genes with few non-zero counts to be very small, and that this can lead to severe overfitting.
Results for fixed standard errors are nearly identical to results for the noisy variance structure that uses the same standard errors. With the latter, \(\sigma^2\) is almost always estimated to be close to zero.
var.elbo <- get.plot.df(allres$var.df, ref.lvl = "constant", type = "elbo")
ggplot(var.elbo, aes(x = category, y = value)) +
geom_boxplot(colour = "blue", outlier.shape = NA) +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in ELBO",
title = "Variance structures, ELBO (reference: Constant)") +
scale_x_discrete(limits = c("genewise", "fixed", "noisyA", "noisyB"),
labels = c("Gene-wise", "Fixed", "Noisy (no zero SEs)",
"Noisy (zero SEs)"))
ggplot(subset(var.elbo, category %in% c("fixed", "noisyA")),
aes(x = category, y = value)) +
geom_boxplot(colour = "blue", outlier.shape = NA) +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in ELBO",
title = "Variance structures, ELBO (zoom)") +
scale_x_discrete(labels = c("Fixed", "Noisy (no zero SEs)"))
Another way to handle unequal sampling errors is to use a variance-stabilizing transformation. I compare the log1p transform to the Anscombe transform (which stabilizes variance for Poisson data): \[ Y_{ij} = \sqrt{X_{ij} + \frac{3}{8}},\] and the arcsine transform (which stabilizes variance for proportions): \[ Y_{ij} = \text{arcsin}\left(\sqrt{\frac{X_{ij}}{\sum_j X_{ij}}}\right)\] In both cases, the magnitude of the sampling errors should be similar across all entries, so it should suffice to fit a constant variance structure.
Note that almost all gene proportions are small, and that the arcsine function is approximately linear for small \(x\). Thus the arcsine transformation is not much different from a square-root transformation of the proportions, which in turn might not be much different from the untransformed proportions. For purposes of comparison, then, I also fit a flash object to the untransformed proportions: \[ Y_{ij} = \frac{X_{ij}}{\sum_j X_{ij}} \]
Finally, I fit a flash object to Pearson residuals, using a binomial approximation to the multinomial distribution as recommended by Townes et al. (The authors prefer to use deviance residuals, but transforming predicted deviance residuals to raw counts is not trivial.)
In all cases, I calculate mean-squared error on the log1p scale. While it is true that this might bias the results in favor of the log1p transform, it can be justified as the relative error in the fitted counts (adding a pseudocount to avoid division by zero).
For the log1p and Anscombe transforms, I fit an additional rank-one “mean” factor (which simply amounts to fitting 6 factors instead of 5). Since the other transforms fit scaled data, no additional mean factor needs to be fitted.
trans.mse <- get.plot.df(allres$trans.df, ref.lvl = "log1p", type = "mse")
ggplot(trans.mse, aes(x = category, y = value)) +
geom_violin(adjust = 1.5,
draw_quantiles = c(0.1, 0.5, 0.9),
color = "red") +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in MSE",
title = "Data transformations, MSE (reference: log1p)") +
scale_x_discrete(limits = c("anscombe", "arcsin", "raw", "pearson"),
labels = c("Anscombe", "Arcsine",
"Proportions (raw)", "Pearson")) +
ylim(-0.1, 0.2)
The Anscombe and arcsine transforms are both outperformed by the log1p transform, and (unsurprisingly) fitting the untransformed matrix of proportions does worst of all (but maybe not as poorly as expected). Interestingly, fitting the Pearson residuals performs nearly as poorly as fitting the matrix of raw proportions.
The log1p transform performs best in terms of mean squared error, but it also allows for a simple interpretation of factors as multiplicative effects. Further, it is possible to fit scaling factors so that loadings are comparable among one another (as discussed in the following section). (Since factors are not multiplicative under the Anscombe transform, one cannot directly fit scaling factors.)
In each case, the ELBO has been adjusted using a change-of-variables formula. Since the arcsine transformation is not differentiable at zero, its ELBO cannot be adjusted.
trans.elbo <- get.plot.df(allres$trans.df, ref.lvl = "log1p", type = "elbo")
ggplot(subset(trans.elbo, category != "arcsin"),
aes(x = category, y = value)) +
geom_boxplot(colour = "blue", outlier.shape = NA) +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in ELBO",
title = "Data transformations, ELBO (reference: log1p)") +
scale_x_discrete(limits = c("anscombe", "raw", "pearson"),
labels = c("Anscombe", "Proportions (raw)", "Pearson")) +
ylim(NA, 0)
Ideally, one would like factor loadings to be comparable across genes even though mean expression can vary by several orders of magnitude. Mean expression varies much less across cells, but one should also account for differences in cell size. One way to achieve both of these goals is to add fixed mean factors to perform row- and column-specific scaling. Specifically, I fit a fixed row vector of all ones with column loadings \(c_j\) to be estimated and a fixed column vector of all ones with row loadings \(r_i\) to be estimated. This is approximately equivalent to estimating separate scaling factors for the rows and columns of the count data: \[ X_{ij} + 1 = e^{r_i}e^{c_j} \]
Compare to the case where FLASH estimates a single rank-one factor with row loadings \(r_i\) and column loadings \(c_j\): \[ X_{ij} + 1 = e^{r_i c_j} \] Here, the scaling factors are not independent. Do note, however, that when \(r_i\) and \(c_j\) are both small, \[ X_{ij} = e^{r_i c_j} - 1 \approx r_i c_j, \] so fitting a single factor might actually work well for small counts.
A third possible method scales the cells in advance (that is, before the log1p transform). Letting \(R_i\) be the total count for cell \(i\), I scale each cell by the factor \[ \frac{\text{median}(R_i)}{R_i} \] so that, in particular, each scaled cell has the same total count. (A mean factor for genes still needs to be fit so that gene loadings are comparable.) Although this method is conceptually simpler, it risks over-relying on a few genes with large counts. In principle, a bi-scaling method like the two discussed above should be more accurate.
norm.mse <- get.plot.df(allres$norm.df, ref.lvl = "none", type = "mse")
ggplot(norm.mse, aes(x = category, y = value)) +
geom_violin(adjust = 1.5,
draw_quantiles = c(0.1, 0.5, 0.9),
color = "red") +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in MSE",
title = "Scaling methods, MSE (reference: Rank-1 mean factor)") +
scale_x_discrete(labels = c("Fixed ones vectors",
"Pre-scaled cells")) +
ylim(-0.1, 0.2)
All methods do about equally well. Since it is difficult to understand exactly what the rank-one mean factor is fitting, I prefer using fixed ones vectors or pre-scaling cells when doing an in-depth analysis of a particular dataset. But the rank-one approach is easier to implement (and a bit faster), so I adopt it as the default scaling method in all other tests throughout this analysis.
As above, the ELBOs have been adjusted using a change-of-variables formula.
norm.elbo <- get.plot.df(allres$norm.df, ref.lvl = "none", type = "elbo")
ggplot(norm.elbo,
aes(x = category, y = value)) +
geom_boxplot(colour = "blue", outlier.shape = NA) +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in ELBO",
title = "Scaling methods, ELBO (reference: Rank-1 mean factor)") +
scale_x_discrete(labels = c("Fixed ones vectors",
"Pre-scaled cells"))
ggplot(norm.elbo,
aes(x = category, y = value)) +
geom_boxplot(colour = "blue", outlier.shape = NA) +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in ELBO",
title = "Scaling methods, ELBO (zoom)") +
scale_x_discrete(labels = c("Fixed ones vectors",
"Pre-scaled cells")) +
ylim(-3000, 3000)
For simplicity, I use normal-mixture priors as the default priors throughout this analysis, but using nonnegative priors for either genes or cells can enhance interpretability. The former yields sets of genes that co-express in the same direction, with cell loadings indicating whether expression levels for the gene set are above or below the mean. The latter can work better for clustering cells, since it yields sets of cells with one set of genes that is overexpressed and a second set that is underexpressed.
In both cases, I test two ashr
parameter settings. method = "fdr"
includes a point mass at zero in the prior, whereas method = "shrink"
includes small mixture components but no point mass. In general, fdr
is better for false discovery rate control while shrink
tends to be slightly more accurate.
Since the normal-mixture prior is in principle the most flexible of all of these priors, I don’t expect the other priors to improve on the mean squared error. The primary reason for choosing to put a nonnegative prior along one of the two dimensions is interpretability, not accuracy.
prior.mse <- get.plot.df(allres$prior.df, ref.lvl = "normal", type = "mse")
ggplot(prior.mse, aes(x = category, y = value)) +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in MSE",
title = "Priors, MSE (reference: Normal mixture)") +
scale_x_discrete(labels = c("NN cells (shrink)",
"NN cells (fdr)",
"NN genes (shrink)",
"NN genes (fdr)"))
ggplot(prior.mse, aes(x = category, y = value)) +
geom_violin(adjust = 1.5,
draw_quantiles = c(0.1, 0.5, 0.9),
color = "red") +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in MSE",
title = "Priors, MSE (zoom)") +
scale_x_discrete(labels = c("NN cells (shrink)",
"NN cells (fdr)",
"NN genes (shrink)",
"NN genes (fdr)")) +
ylim(-0.025, 0.1)
When the nonnegative prior is on genes rather than cells, there are a few outlying fits that have very poor mean squared error.
If the outlying fits are ignored, each type of prior does approximately as well as normal-mixture priors. This is fairly surprising since 5 normal-mixture factors is in some sense equivalent to 10 semi-nonnegative factors. I’m not sure whether most of the useful information in a given factor is in one direction, or whether 5 normal-mixture factors are already beginning to overfit the data (as discussed in the following section).
prior.elbo <- get.plot.df(allres$prior.df, ref.lvl = "normal", type = "elbo")
ggplot(prior.elbo,
aes(x = category, y = value)) +
geom_boxplot(colour = "blue", outlier.shape = NA) +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in ELBO",
title = "Priors, ELBO (reference: Normal mixture)") +
scale_x_discrete(labels = c("NN cells (shrink)",
"NN cells (fdr)",
"NN genes (shrink)",
"NN genes (fdr)")) +
ylim(NA, 0)
In all previous tests, I fit 5 factors to each flash object. Here, I also fit 10 and 15 factors to see whether there’s any evidence of overfitting. Additionally, I look at how backfitting changes mean squared error.
nfactors.mse <- get.plot.df(allres$nfactors.df, ref.lvl = "g1", type = "mse")
ggplot(nfactors.mse, aes(x = category, y = value)) +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in MSE",
title = "Number of factors, MSE (reference: Greedy, 5 factors)") +
scale_x_discrete(limits = c("bf1", "g2", "bf2", "g3", "bf3"),
labels = c("Backfit, 5 f.",
"Greedy, 10 f.",
"Backfit, 10 f.",
"Greedy, 15 f.",
"Backfit, 15 f."))
Version | Author | Date |
---|---|---|
60843e0 | Jason Willwerscheid | 2019-03-22 |
ggplot(nfactors.mse, aes(x = category, y = value)) +
geom_violin(adjust = 1.5,
draw_quantiles = c(0.1, 0.5, 0.9),
color = "red") +
geom_jitter(position = position_jitter(0.25)) +
labs(x = NULL, y = "Difference in MSE",
title = "Number of factors, MSE (zoom)") +
scale_x_discrete(limits = c("bf1", "g2", "bf2", "g3", "bf3"),
labels = c("Backfit, 5 f.",
"Greedy, 10 f.",
"Backfit, 10 f.",
"Greedy, 15 f.",
"Backfit, 15 f.")) +
ylim(-0.1, 0.2)
Version | Author | Date |
---|---|---|
60843e0 | Jason Willwerscheid | 2019-03-22 |
The results suggest that overfitting (in the sense of predictive accuracy) occurs well before the log likelihood of the model stops improving. The ten- and fifteen-factor fits are nearly always worse than the five-factor fits, and the backfits are on average worse than the corresponding greedy fits. In surprisingly many cases, backfitting turns out to be disastrous.
To further analyze the dynamics of overfitting, I fit a single factor to the full drop-seq dataset (after removing 1% of entries at random) and calculate mean squared error, then I fit a second factor, and so on until 60 factors have been added. Even though flashier
continues to add factors throughout the process, the mean squared error no longer improves monotonically after 32 factors, and bottoms out at 40 factors.
The results imply that flashier
is unable to add the “correct” number of factors for count data, and that some type of cross validation might be needed.
mse.df <- allres$mse.df
ggplot(mse.df, aes(x = k, y = mse)) + geom_point() +
labs(x = "Number of factors", y = "Mean squared error",
title = "Incremental greedy additions (full dataset)")
Version | Author | Date |
---|---|---|
60843e0 | Jason Willwerscheid | 2019-03-22 |
ggplot(subset(mse.df, k > 19), aes(x = k, y = mse)) + geom_point() +
labs(x = "Number of factors", y = "Mean squared error",
title = "Incremental greedy additions (zoom)")
Version | Author | Date |
---|---|---|
60843e0 | Jason Willwerscheid | 2019-03-22 |
sessionInfo()
R version 3.4.3 (2017-11-30)
Platform: x86_64-apple-darwin15.6.0 (64-bit)
Running under: macOS High Sierra 10.13.6
Matrix products: default
BLAS: /Library/Frameworks/R.framework/Versions/3.4/Resources/lib/libRblas.0.dylib
LAPACK: /Library/Frameworks/R.framework/Versions/3.4/Resources/lib/libRlapack.dylib
locale:
[1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8
attached base packages:
[1] stats graphics grDevices utils datasets methods base
other attached packages:
[1] ggplot2_3.1.0
loaded via a namespace (and not attached):
[1] Rcpp_1.0.0 bindr_0.1 knitr_1.21.6
[4] whisker_0.3-2 magrittr_1.5 workflowr_1.0.1
[7] munsell_0.5.0 colorspace_1.3-2 R6_2.3.0
[10] rlang_0.3.0.1 dplyr_0.7.4 stringr_1.3.1
[13] plyr_1.8.4 tools_3.4.3 grid_3.4.3
[16] gtable_0.2.0 xfun_0.4 R.oo_1.21.0
[19] withr_2.1.2.9000 git2r_0.21.0 htmltools_0.3.6
[22] assertthat_0.2.0 yaml_2.2.0 lazyeval_0.2.1
[25] digest_0.6.18 rprojroot_1.3-2 tibble_1.4.2
[28] bindrcpp_0.2 R.utils_2.6.0 glue_1.3.0
[31] evaluate_0.12 rmarkdown_1.11 labeling_0.3
[34] stringi_1.2.4 pillar_1.2.1 compiler_3.4.3
[37] scales_1.0.0 backports_1.1.2 R.methodsS3_1.7.1
[40] pkgconfig_2.0.1
This reproducible R Markdown analysis was created with workflowr 1.0.1