The oem package has been on CRAN for some time now, but with the latest update I expect few structural changes to the user interface. oem
is a package for the estimation of various penalized regression models using the oem algorithm of Xiong et al. (2016). The focus of oem
is to provide high performance computation for big tall data. Many applications not only have a large number of variables, but a vast number of observations; oem
is designed to perform well in these settings.
 Fast computation for big tall data
 Efficient computation for computation for multiple penalties simultaneously
 Efficient crossvalidation
In this post I’ll give a brief overview of what is in the oem
package and how to use it.
Installation
The simplest way to install oem
is via the CRAN repositories as the following:
install.packages("oem")
To install the development version, first install the devtools
package
install.packages("devtools")
and then install oem
via the install_github
function
devtools::install_github("jaredhuling/oem")
Quick Start
First load oem
library(oem)
Simulate data
nobs < 1e4
nvars < 100
x < matrix(rnorm(nobs * nvars), ncol = nvars)
y < drop(x %*% c(0.5, 0.5, 0.5, 0.5, 1, rep(0, nvars  5))) + rnorm(nobs, sd = 4)
Fit a penalized regression model using the oem
function
fit1 < oem(x = x, y = y, penalty = "lasso")
Plot the solution path
plot(fit1)
Key Features
Available functions
Function Name  Functionality 

oem() 
Main fitting function 
predict.oemfit() 
Prediction for oem objects 
plot.oemfit() 
Plotting for oem objects 
logLik.oemfit() 
log Likelihood for oem objects 
cv.oem() 
Crossvalidation function 
xval.oem() 
Accelerated crossvalidation for linear models 
predict.cv.oem() 
Prediction for cv.oem objects 
plot.cv.oem() 
Plotting for cv.oem objects 
logLik.cv.oem() 
log Likelihood for cv.oem objects 
Available Penalties
Penalty  Option Name  Penalty Form 

Lasso  lasso 
\(\lambda \sum_{j = 1}^pd_j\vert\beta_j\vert\) 
Elastic Net  elastic.net 
\(\lambda \sum_{j = 1}^p\alpha d_j\vert\beta_j\vert + (1  \alpha)\lambda \sum_{j = 1}^pd_j\beta_j^2\) 
MCP  mcp 
\(\lambda \sum_{j = 1}^pd_j \int_0^{\beta_j}(1  x/(\gamma\lambda d_j))_+\mathrm{d}x\) 
SCAD  scad 
\(\sum_{j = 1}^p p^{SCAD}_{\lambda d_j,\gamma}(\beta_j)\) 
Group Lasso  grp.lasso 
\(\lambda \sum_{k = 1}^Gd_k\sqrt{\sum_{j \in g_k}\beta_j^2}\) 
Group MCP  grp.mcp 
\(\sum_{k = 1}^G p_{\lambda d_k,\gamma}^{MCP}\left(\vert \vert \boldsymbol\beta_{g_k}\vert \vert_2\right)\) 
Group SCAD  grp.scad 
\(\sum_{k = 1}^G p_{\lambda d_k,\gamma}^{SCAD}\left(\vert \vert \boldsymbol\beta_{g_k}\vert \vert_2\right)\) 
Sparse Group Lasso  sparse.grp.lasso 
\(\lambda \alpha\sum_{j = 1}^pd_j\vert\beta_j\vert + \lambda (1\alpha)\sum_{k = 1}^Gd_k\sqrt{\sum_{j \in g_k}\beta_j^2}\) 
where \(\vert\vert\boldsymbol\beta_{g_k}\vert\vert_2 = \sqrt{\sum_{j \in g_k}\beta_j^2}\).
Any penalty with .net
at the end of its name has a ridge term of \((1  \alpha)\lambda \sum_{j = 1}^pd_j\beta_j^2\) added to it and the original penalty multiplied by \(\alpha\). For example, grp.mcp.net
is the penalty
\[\lambda \sum_{k = 1}^G\alpha p^{MCP}_{\lambda d_k,\gamma}(\vert\vert\boldsymbol\beta_{g_k}\vert\vert_2) + (1  \alpha)\lambda \sum_{j = 1}^pd_j\beta_j^2. \]
Available Model Families
The following models are available currently.
Model  Option Name  Loss Form 

Linear Regression  gaussian 
\(\frac{1}{2n}\sum_{i=1}^n(y_i  x_i^T\beta) ^ 2\) 
Logistic Regression  binomial 
\(\frac{1}{n}\sum_{i=1}^n\left[y_i x_i^T\beta  \log (1 + \exp\{ x_i^T\beta \} ) \right]\) 
There are plans to include support for multiple responses, binomial models (not just logistic regression), Cox’s proportional hazards model, and more if requested.
Fitting multiple penalties at once
The oem algorithm is wellsuited to quickly estimate a solution path for multiple penalties simultaneously if the number of variables is not too large. The oem algorithm is only efficient for multiple penalties for linear models.
For the group lasso penalty, the groups
argument must be used. groups
should be a vector which indicates the group number for each variable.
fit2 < oem(x = x, y = y, penalty = c("lasso", "mcp", "grp.lasso", "grp.mcp"),
groups = rep(1:20, each = 5))
Plot the solution paths for all models
layout(matrix(1:4, ncol = 2))
plot(fit2, which.model = 1, xvar = "lambda")
plot(fit2, which.model = 2, xvar = "lambda")
plot(fit2, which.model = 3, xvar = "lambda")
plot(fit2, which.model = "grp.mcp", xvar = "lambda")
Timing Comparison
The following is a demonstration of oem’s efficiency for computing solutions for tuning parameter paths for multiple penalties at once.
Linear Regression
The efficiency oem for fitting multiple penalties at once only applies to linear models. However, for linear models it is quite efficient, even for a high number of tuning parameters for many different penalties.
nobs < 1e5
nvars < 100
x2 < matrix(rnorm(nobs * nvars), ncol = nvars)
y2 < drop(x2 %*% c(0.5, 0.5, 0.5, 0.5, 1, rep(0, nvars  5))) + rnorm(nobs, sd = 4)
system.time(fit2a < oem(x = x2, y = y2, penalty = c("grp.lasso"),
groups = rep(1:20, each = 5), nlambda = 100L))
## user system elapsed
## 0.139 0.016 0.156
system.time(fit2b < oem(x = x2, y = y2,
penalty = c("grp.lasso", "lasso", "mcp",
"scad", "elastic.net", "grp.mcp",
"grp.scad", "sparse.grp.lasso"),
groups = rep(1:20, each = 5), nlambda = 100L))
## user system elapsed
## 0.159 0.016 0.176
system.time(fit2c < oem(x = x2, y = y2,
penalty = c("grp.lasso", "lasso", "mcp",
"scad", "elastic.net", "grp.mcp",
"grp.scad", "sparse.grp.lasso"),
groups = rep(1:20, each = 5), nlambda = 500L))
## user system elapsed
## 0.228 0.023 0.250
Logistic Regression
It is still more efficient to fit multiple penalties at once instead of individually for logistic regression, but the benefit is not as dramatic as for linear models.
nobs < 5e4
nvars < 100
x2 < matrix(rnorm(nobs * nvars), ncol = nvars)
y2 < rbinom(nobs, 1, prob = 1 / (1 + exp(drop(x2 %*% c(0.15, 0.15, 0.15, 0.15, 0.25, rep(0, nvars  5))))))
system.time(fit2a < oem(x = x2, y = y2, penalty = c("grp.lasso"),
family = "binomial",
groups = rep(1:20, each = 5), nlambda = 100L))
## user system elapsed
## 0.958 0.035 1.000
system.time(fit2b < oem(x = x2, y = y2, penalty = c("grp.lasso", "lasso", "mcp", "scad", "elastic.net"),
family = "binomial",
groups = rep(1:20, each = 5), nlambda = 100L))
## user system elapsed
## 4.997 0.072 5.093
Cross Validation
Here we use the nfolds
argument to specify the number of folds for \(k\)fold cross validation
system.time(cvfit1 < cv.oem(x = x, y = y,
penalty = c("lasso", "mcp",
"grp.lasso", "grp.mcp"),
gamma = 2,
groups = rep(1:20, each = 5),
nfolds = 10))
## user system elapsed
## 1.171 0.225 1.400
Plot the cross validation mean squared error results for each model
layout(matrix(1:4, ncol = 2))
plot(cvfit1, which.model = 1)
plot(cvfit1, which.model = 2)
plot(cvfit1, which.model = 3)
plot(cvfit1, which.model = 4)
Extremely Fast Cross Validation for Linear Models
The function xval.oem
offers accelerated cross validation for penalized linear models. In many cases is is orders of magnitude faster than cv.oem. It is only recommended for scenarios where the number of observations is larger than the number of variables. In addition to the computational gains in singlecore usage, it also benefits from parallelizaton using OpenMP (instead of using foreach, as used by cv.oem). For large enough problems, it has on a similar order of computation time as just fitting one OEM model.
nobsc < 1e5
nvarsc < 100
xc < matrix(rnorm(nobsc * nvarsc), ncol = nvarsc)
yc < drop(xc %*% c(0.5, 0.5, 0.5, 0.5, 1, rep(0, nvarsc  5))) + rnorm(nobsc, sd = 4)
system.time(cvalfit1 < cv.oem(x = xc, y = yc, penalty = "lasso",
groups = rep(1:20, each = 5),
nfolds = 10))
## user system elapsed
## 4.392 0.990 5.400
system.time(xvalfit1 < xval.oem(x = xc, y = yc, penalty = "lasso",
groups = rep(1:20, each = 5),
nfolds = 10))
## user system elapsed
## 0.538 0.021 0.562
system.time(xvalfit2 < xval.oem(x = xc, y = yc, penalty = "lasso",
groups = rep(1:20, each = 5),
nfolds = 10, ncores = 2))
## user system elapsed
## 0.657 0.020 0.395
system.time(ofit1 < oem(x = xc, y = yc, penalty = "lasso",
groups = rep(1:20, each = 5)))
## user system elapsed
## 0.229 0.015 0.143
Evaluation Metrics
A variety of evaluation metrics can be used for cross validation. The available metrics can be found in the table below
Model  Metric  type.measure= 

Linear Regression  Mean squared error  mse or deviance 
Mean absolute error  mae 

Logistic Regression  Deviance  deviance 
Area under the ROC curve  auc 

Misclassification Rate  class 

Mean squared error of fitted mean  mse 

Mean absolute error of fitted mean  mae 
Consider a binary outcome setting with logistic regression.
nobs < 2e3
nvars < 20
x < matrix(runif(nobs * nvars, max = 2), ncol = nvars)
y < rbinom(nobs, 1, prob = 1 / (1 + exp(drop(x %*% c(0.25, 1, 1, 0.5, 0.5, 0.25, rep(0, nvars  6))))))
Misclassification Rate
cvfit2 < cv.oem(x = x, y = y, penalty = c("lasso", "mcp",
"grp.lasso", "grp.mcp"),
family = "binomial",
type.measure = "class",
gamma = 2,
groups = rep(1:10, each = 2),
nfolds = 10, standardize = FALSE)
In this case, misclassification rate is not the best indicator of performance. The classes here are imbalanced:
mean(y)
## [1] 0.0695
Area Under the ROC Curve
Area under the ROC curve is an alternative classification metric to misclassification rate. It is available by setting type.measure = "auc"
.
cvfit2 < cv.oem(x = x, y = y, penalty = c("lasso", "mcp",
"grp.lasso", "grp.mcp"),
family = "binomial",
type.measure = "auc",
gamma = 2,
groups = rep(1:10, each = 2),
nfolds = 10, standardize = FALSE)
Methods for Very Large Scale Problems
OEM with Precomputed \(X^TX\), \(X^TY\) for Linear Models
With a very large dataset and computing cluster, the total size of a dataset may be very large, but if the number of variables is only moderately large (on the order of a few thousands) \(X^TX\) and \(X^TY\) may not be large and may already be available from other computations or can be computed trivially in parallel. The function oem.xtx
computes penalized linear regression models using the OEM algorithm only using \(X^TX\) and \(X^TY\). Standardization can be achieved by providing a vector of scaling factors (usually the standard deviations of the columns of x). The function is used like the following:
xtx < crossprod(xc) / nrow(xc)
xty < crossprod(xc, yc) / nrow(xc)
system.time(fit < oem(x = xc, y = yc,
penalty = c("lasso", "grp.lasso"),
standardize = FALSE, intercept = FALSE,
groups = rep(1:20, each = 5)))
## user system elapsed
## 0.131 0.033 0.164
system.time(fit.xtx < oem.xtx(xtx = xtx, xty = xty,
penalty = c("lasso", "grp.lasso"),
groups = rep(1:20, each = 5)) )
## user system elapsed
## 0.008 0.000 0.009
max(abs(fit$beta[[1]][1,]  fit.xtx$beta[[1]]))
## [1] 2.353673e14
max(abs(fit$beta[[2]][1,]  fit.xtx$beta[[2]]))
## [1] 2.353673e14
col.std < apply(xc, 2, sd)
fit.xtx.s < oem.xtx(xtx = xtx, xty = xty,
scale.factor = col.std,
penalty = c("lasso", "grp.lasso"),
groups = rep(1:20, each = 5))
Outofmemory Computation
The OEM package also provides functionality for ondisk computation with the big.oem
function, allowing for fitting penalized regression models on datasets too large to fit in memory. The big.oem
function uses the tools provided by the bigmemory
package, so a big.matrix object must be used for the design matrix.
set.seed(123)
nrows < 50000
ncols < 100
bkFile < "bigmat.bk"
descFile < "bigmatk.desc"
bigmat < filebacked.big.matrix(nrow=nrows, ncol=ncols, type="double",
backingfile=bkFile, backingpath=".",
descriptorfile=descFile,
dimnames=c(NULL,NULL))
# Each column value with be the column number multiplied by
# samples from a standard normal distribution.
set.seed(123)
for (i in 1:ncols) bigmat[,i] = rnorm(nrows)*i
yb < rnorm(nrows) + bigmat[,1]  bigmat[,2]
## outofmemory computation
fit < big.oem(x = bigmat, y = yb,
penalty = c("lasso", "grp.lasso"),
groups = rep(1:20, each = 5))
## fitting with inmemory computation
fit2 < oem(x = bigmat[,], y = yb,
penalty = c("lasso", "grp.lasso"),
groups = rep(1:20, each = 5))
max(abs(fit$beta[[1]]  fit2$beta[[1]]))
## [1] 1.534783e05
Other Features
Parallelization via OpenMP
Computational time can be reduced a little via OpenMP parallelization of the key computational steps of the OEM algorithm. Simply use the ncores
argument to access parallelization. There is no need for the foreach package.
nobsc < 1e5
nvarsc < 500
xc < matrix(rnorm(nobsc * nvarsc), ncol = nvarsc)
yc < drop(xc %*% c(0.5, 0.5, 0.5, 0.5, 1, rep(0, nvarsc  5))) + rnorm(nobsc, sd = 4)
system.time(fit < oem(x = xc, y = yc,
penalty = c("lasso", "grp.lasso"),
standardize = FALSE, intercept = FALSE,
groups = rep(1:20, each = 25)))
## user system elapsed
## 2.167 0.137 2.305
system.time(fitp < oem(x = xc, y = yc,
penalty = c("lasso", "grp.lasso"),
standardize = FALSE, intercept = FALSE,
groups = rep(1:20, each = 25), ncores = 2))
## user system elapsed
## 2.368 0.137 1.358
Penalty Adjustment
If some variables should not be penalized, this can be specified through the use of the penalty.factor
argument for all penalties other than the group lasso. For the group lasso, the groupspecific weights can be modified by the group.weights
argument. penalty.factor
should be a vector of length equal to the number of columns in x
. Each element in penalty.factor
will be multiplied to the applied tuning parameter for each corresponding variable. For example, for a problem with 5 variables (ncol(x) = 5
), setting penalty.factor = c(1, 1, 1, 0, 0)
will effectively only allow penalization for the first three variables. The group.weights
argument should be a vector with length equal to the number of groups. Similarly to penalty.factor
, these weights will be multiplied to the penalty applied to each group. penalty.factor
and group.weights
can also be used to fit the adaptive lasso and adaptive group lasso, respectively.
The following example shows how to fit an adaptive lasso using oem
nobs < 1e4
nvars < 102
x < matrix(rnorm(nobs * nvars), ncol = nvars)
y < drop(x %*% c(0.5, 0.5, 0.5, 0.5, 1, 0.5, rep(0, nvars  6))) + rnorm(nobs, sd = 4)
lams < exp(seq(log(2.5), log(5e5), length.out = 100L))
ols.estimates < coef(lm.fit(y = y, x = cbind(1, x)))[1]
fit.adaptive < oem(x = x, y = y, penalty = c("lasso"),
penalty.factor = 1 / abs(ols.estimates),
lambda = lams)
group.indicators < rep(1:34, each = 3)
## norms of OLS estimates for each group
group.norms < sapply(1:34, function(idx) sqrt(sum((ols.estimates[group.indicators == idx]) ^ 2)))
fit.adaptive.grp < oem(x = x, y = y, penalty = c("grp.lasso"),
group.weights = 1 / group.norms,
groups = group.indicators,
lambda = lams)
More Information
For further information about oem
, please visit:
 The oem site: casualinference.org/oem
 The oem source code: github.com/jaredhuling/oem
 The oem paper: oem paper