Title: | A Simple Way to Specify Symmetric, Block Diagonal Matrices |
---|---|
Description: | Provides a simple mechanism to specify a symmetric block diagonal matrices (often used for covariance matrices). This is based on the domain specific language implemented in 'nlmixr2' but expanded to create matrices in R generally instead of specifying parts of matrices to estimate. It has expanded to include some matrix manipulation functions that are generally useful for 'rxode2' and 'nlmixr2'. |
Authors: | Matthew L. Fidler [aut, cre] , Bill Denney [ctb] |
Maintainer: | Matthew L. Fidler <[email protected]> |
License: | GPL (>=2) |
Version: | 1.0.0.9000 |
Built: | 2024-12-27 06:14:43 UTC |
Source: | https://github.com/nlmixr2/lotri |
As lower triangular matrix
as.lotri(x, ..., default = "") ## S3 method for class 'matrix' as.lotri(x, ..., default = "") ## S3 method for class 'data.frame' as.lotri(x, ..., default = "") ## Default S3 method: as.lotri(x, ..., default = "")
as.lotri(x, ..., default = "") ## S3 method for class 'matrix' as.lotri(x, ..., default = "") ## S3 method for class 'data.frame' as.lotri(x, ..., default = "") ## Default S3 method: as.lotri(x, ..., default = "")
x |
Matrix or other data frame |
... |
Other factors |
default |
Is the default factor when no conditioning is implemented. |
Lower triangular matrix
Matthew Fidler
Easily Specify block-diagonal matrices with lower triangular info
lotri(x, ..., cov = FALSE, rcm = FALSE, envir = parent.frame(), default = "id")
lotri(x, ..., cov = FALSE, rcm = FALSE, envir = parent.frame(), default = "id")
x |
list, matrix or expression, see details |
... |
Other arguments treated as a list that will be concatenated then reapplied to this function. |
cov |
either a boolean or a function accepting a matrix input. When a boolean, 'cov' describes if this matrix definition is actually a rxode2/nlmixr2-style covariance matrix. If so, 'lotri()' will enforce certain regularity conditions: - When diagonal elements are zero, the off-diagonal elements are zero. This means the covariance element is fixed to zero and not truly part of the covariance matrix in general. - For the rest of the matrix, 'lotri' will check that it is non-positive definite (which is required for covariance matrix in general) It is sometimes difficult to adjust covariance matrices to be non-positive definite. For this reason 'cov' may also be a function accepting a matrix input and returning a non-positive definite matrix from this matrix input. When this is a function, it is equivalent to 'cov=TRUE' with the additional ability to correct the matrix to be non-positive definite if needed. |
rcm |
logical; if 'TRUE', the matrix will be reordered to change the matrix to a banded matrix, which is easier to express in 'lotri' than a full matrix. The RCM stands for the reverse Cuthill McKee (RCM) algorithm which is used for this matrix permutation. (see 'rcm()') |
envir |
the |
default |
Is the default factor when no conditioning is implemented. |
This can take an R matrix, a list including matrices or expressions, or expressions
Expressions can take the form
name ~ estimate
Or the lower triangular matrix when "adding" the names
name1 + name2 ~ c(est1, est2, est3)
The matrices are concatenated into a block diagonal matrix, like
bdiag
, but allows expressions to specify
matrices easier.
named symmetric matrix useful in 'rxode2()' simulations (and perhaps elsewhere)
Matthew L Fidler
## A few ways to specify the same matrix lotri({et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30)}) ## You do not need to enclose in {} lotri(et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30), et5 ~ 6) ## But if you do enclose in {}, you can use ## multi-line matrix specifications: lotri({et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30) et5 ~ 6 }) ## You can also add lists or actual R matrices as in this example: lotri(list(et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30), matrix(1,dimnames=list("et5","et5")))) ## Overall this is a flexible way to specify symmetric block ## diagonal matrices. ## For rxode2, you may also condition based on different levels of ## nesting with lotri; Here is an example: mat <- lotri(lotri(iov.Ka ~ 0.5, iov.Cl ~ 0.6), lotri(occ.Ka ~ 0.5, occ.Cl ~ 0.6) | occ(lower=4,nu=3)) mat ## you may access features of the matrix simply by `$` that is mat$lower # Shows the lower bound for each condition mat$lower$occ # shows the lower bound for the occasion variable ## Note that `lower` fills in defaults for parameters. This is true ## for `upper` true; In fact when accessing this the defaults ## are put into the list mat$upper ## However all other values return NULL if they are not present like mat$lotri ## And values that are specified once are only returned on one list: mat$nu mat$nu$occ mat$nu$id ## You can also change the default condition with `as.lotri` mat <- as.lotri(mat, default="id") mat
## A few ways to specify the same matrix lotri({et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30)}) ## You do not need to enclose in {} lotri(et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30), et5 ~ 6) ## But if you do enclose in {}, you can use ## multi-line matrix specifications: lotri({et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30) et5 ~ 6 }) ## You can also add lists or actual R matrices as in this example: lotri(list(et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30), matrix(1,dimnames=list("et5","et5")))) ## Overall this is a flexible way to specify symmetric block ## diagonal matrices. ## For rxode2, you may also condition based on different levels of ## nesting with lotri; Here is an example: mat <- lotri(lotri(iov.Ka ~ 0.5, iov.Cl ~ 0.6), lotri(occ.Ka ~ 0.5, occ.Cl ~ 0.6) | occ(lower=4,nu=3)) mat ## you may access features of the matrix simply by `$` that is mat$lower # Shows the lower bound for each condition mat$lower$occ # shows the lower bound for the occasion variable ## Note that `lower` fills in defaults for parameters. This is true ## for `upper` true; In fact when accessing this the defaults ## are put into the list mat$upper ## However all other values return NULL if they are not present like mat$lotri ## And values that are specified once are only returned on one list: mat$nu mat$nu$occ mat$nu$id ## You can also change the default condition with `as.lotri` mat <- as.lotri(mat, default="id") mat
Change a matrix or lotri matrix to a lotri expression
lotriAsExpression( x, useIni = FALSE, plusNames = getOption("lotri.plusNames", FALSE), nameEst = getOption("lotri.nameEst", 5L) )
lotriAsExpression( x, useIni = FALSE, plusNames = getOption("lotri.plusNames", FALSE), nameEst = getOption("lotri.nameEst", 5L) )
x |
matrix |
useIni |
use the ini block |
plusNames |
logical, when 'TRUE' use the 'a + b ~ c(1, 0.1, 1)' naming convention. Otherwise use the lotri single line convention 'a ~ 1; b ~ c(0.1, 1)' |
nameEst |
logical or integerish. When logical 'TRUE' will add names to all matrix estimates and 'TRUE' when using the lotri single line convention i.e. 'a~c(a=1); b~c(a=0.1, b=1)'. When an integer, the dimension of the matrix being displayed needs to have a dimension above this number before names are displayed. |
Convert a lotri data frame to a lotri expression
lotriDataFrameToLotriExpression(data, useIni = FALSE)
lotriDataFrameToLotriExpression(data, useIni = FALSE)
data |
lotri data frame |
useIni |
Use 'ini' instead of 'lotri' in the expression |
expression of the lotri syntax equivalent to the data.frame provided
Matthew L. Fidler
x <- lotri({ tka <- 0.45; label("Log Ka") tcl <- 1; label("Log Cl") tv <- 3.45; label("Log V") eta.ka ~ 0.6 eta.cl ~ 0.3 eta.v ~ 0.1 add.err <- 0.7 }) df <- as.data.frame(x) lotriDataFrameToLotriExpression(df) # You may also call as.expression directly from the lotri object as.expression(x)
x <- lotri({ tka <- 0.45; label("Log Ka") tcl <- 1; label("Log Cl") tv <- 3.45; label("Log V") eta.ka ~ 0.6 eta.cl ~ 0.3 eta.v ~ 0.1 add.err <- 0.7 }) df <- as.data.frame(x) lotriDataFrameToLotriExpression(df) # You may also call as.expression directly from the lotri object as.expression(x)
Extract or remove lotri estimate data frame from lotri object
lotriEst(x, drop = FALSE)
lotriEst(x, drop = FALSE)
x |
lotri object |
drop |
boolean indicating if the lotri estimate should be dropped |
data frame with estimates or NULL if there is not a data.frame attached
fix1 <- lotri({ a <- c(0, 1); backTransform("exp"); label("a label") b <- c(0, 1, 2) c <- fix(1) d <- fix(0, 1, 2) e <- c(0, 1, 2, fixed) f+g ~ c(1, 0.5, 1) }) # Extract the attached lotri estimate data frame lotriEst(fix1) # Remove the attached lotri estimate data frame lotriEst(fix1, drop=TRUE)
fix1 <- lotri({ a <- c(0, 1); backTransform("exp"); label("a label") b <- c(0, 1, 2) c <- fix(1) d <- fix(0, 1, 2) e <- c(0, 1, 2, fixed) f+g ~ c(1, 0.5, 1) }) # Extract the attached lotri estimate data frame lotriEst(fix1) # Remove the attached lotri estimate data frame lotriEst(fix1, drop=TRUE)
Determine if the matrix is a block matrix
lotriIsBlockMat(mat)
lotriIsBlockMat(mat)
mat |
matrix to determine if it is a block matrix |
logical value, TRUE if it is a block matrix and FALSE otherwise
Matthew L. Fidler
m <- lotri({ a ~ c(a = 0.4) b ~ c(a = 0, b = 0.3) c ~ c(a = 0, b = 0, c = 0) d ~ c(a = -0.1, b = 0, c = 0, d = 0.2) e ~ c(a = 0, b = 0, c = 0, d = 0, e = 0.5) f ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 1.3) g ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = -0.6, g = 0.8) h ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0) i ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0.2) j ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0.9) k ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0, k = 0.9) l ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = -0.2, k = 0, l = 0.3) m ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0, k = 0, l = 0, m = 2.1) n ~ c(a = 0.2, b = 0, c = 0, d = 0.2, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0, k = 0, l = 0, m = 0, n = 0.4) o ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = -1.1, g = 0.9, h = 0, i = 0, j = 0, k = 0, l = 0, m = 0, n = 0, o = 4.7) p ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0.5, k = 0, l = 0.2, m = 0, n = 0, o = 0, p = 1.9) }) lotriIsBlockMat(m) lotriIsBlockMat(rcm(m))
m <- lotri({ a ~ c(a = 0.4) b ~ c(a = 0, b = 0.3) c ~ c(a = 0, b = 0, c = 0) d ~ c(a = -0.1, b = 0, c = 0, d = 0.2) e ~ c(a = 0, b = 0, c = 0, d = 0, e = 0.5) f ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 1.3) g ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = -0.6, g = 0.8) h ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0) i ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0.2) j ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0.9) k ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0, k = 0.9) l ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = -0.2, k = 0, l = 0.3) m ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0, k = 0, l = 0, m = 2.1) n ~ c(a = 0.2, b = 0, c = 0, d = 0.2, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0, k = 0, l = 0, m = 0, n = 0.4) o ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = -1.1, g = 0.9, h = 0, i = 0, j = 0, k = 0, l = 0, m = 0, n = 0, o = 4.7) p ~ c(a = 0, b = 0, c = 0, d = 0, e = 0, f = 0, g = 0, h = 0, i = 0, j = 0.5, k = 0, l = 0.2, m = 0, n = 0, o = 0, p = 1.9) }) lotriIsBlockMat(m) lotriIsBlockMat(rcm(m))
This creates a named banded symmetric matrix from a list of named symmetric matrices.
lotriMat(matList, format = NULL, start = 1L)
lotriMat(matList, format = NULL, start = 1L)
matList |
list of symmetric named matrices |
format |
The format of dimension names when a sub-matrix is repeated. The format will be called with the dimension number, so "ETA[%d]" would represent "ETA[1]", "ETA[2]", etc |
start |
The number the counter of each repeated dimension should start. |
Named symmetric block diagonal matrix based on concatenating the list of matrices together
Matthew Fidler
testList <- list(lotri({et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30)}), lotri(et5 ~ 6)) testList lotriMat(testList) # Another option is to repeat a matrix a number of times. This # can be done with list(matrix, # times to repeat). # In the example below, the first matrix is repeated 3 times testList <- list(list(lotri({et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30)}), 3), lotri(et5 ~ 6)) lotriMat(testList) # Notice that the dimension names `et2`, `et3` and `et4` are # repeated. # Another option is to name the dimensions. For example it could # be `ETA[1]`, `ETA[2]`, etc by using the 'format' option: lotriMat(testList, "ETA[%d]") # Or could start with ETA[2]: lotriMat(testList, "ETA[%d]", 2)
testList <- list(lotri({et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30)}), lotri(et5 ~ 6)) testList lotriMat(testList) # Another option is to repeat a matrix a number of times. This # can be done with list(matrix, # times to repeat). # In the example below, the first matrix is repeated 3 times testList <- list(list(lotri({et2 + et3 + et4 ~ c(40, 0.1, 20, 0.1, 0.1, 30)}), 3), lotri(et5 ~ 6)) lotriMat(testList) # Notice that the dimension names `et2`, `et3` and `et4` are # repeated. # Another option is to name the dimensions. For example it could # be `ETA[1]`, `ETA[2]`, etc by using the 'format' option: lotriMat(testList, "ETA[%d]") # Or could start with ETA[2]: lotriMat(testList, "ETA[%d]", 2)
Converts a matrix into a list of block matrices
lotriMatInv(mat)
lotriMatInv(mat)
mat |
Matrix to convert to a list of block matrices |
This is the inverse of 'lotriMat()'
A list of block matrixes
Matthew Fidler
# Create a block matrix using `lotri()` mat <- lotri({ a+b ~ c(1, 0.5, 1) c ~ 1 d +e ~ c(1, 0.5, 1) }) print(mat) # now convert t a list of matrices mat2 <- lotriMatInv(mat) print(mat2) # Of course you can convert it back to a full matrix: mat3 <- lotriMat(mat2) print(mat3)
# Create a block matrix using `lotri()` mat <- lotri({ a+b ~ c(1, 0.5, 1) c ~ 1 d +e ~ c(1, 0.5, 1) }) print(mat) # now convert t a list of matrices mat2 <- lotriMatInv(mat) print(mat2) # Of course you can convert it back to a full matrix: mat3 <- lotriMat(mat2) print(mat3)
With 'ensureSymmetry' it makes sure it is symmetric by applying 0.5*(t(x) + x) before using lotriNearPD
lotriNearPD( x, keepDiag = FALSE, do2eigen = TRUE, doDykstra = TRUE, only.values = FALSE, ensureSymmetry = !isSymmetric(x), eig.tol = 1e-06, conv.tol = 1e-07, posd.tol = 1e-08, maxit = 100L, trace = FALSE )
lotriNearPD( x, keepDiag = FALSE, do2eigen = TRUE, doDykstra = TRUE, only.values = FALSE, ensureSymmetry = !isSymmetric(x), eig.tol = 1e-06, conv.tol = 1e-07, posd.tol = 1e-08, maxit = 100L, trace = FALSE )
x |
numeric |
keepDiag |
logical, generalizing |
do2eigen |
logical indicating if a 'posdefify()' (like in the package 'sfsmisc') eigen step should be applied to the result of the Higham algorithm |
doDykstra |
logical indicating if Dykstra's correction should be
used; true by default. If false, the algorithm is basically the
direct fixpoint iteration
|
only.values |
logical; if |
ensureSymmetry |
logical; by default, |
eig.tol |
defines relative positiveness of eigenvalues compared
to largest one, |
conv.tol |
convergence tolerance for Higham algorithm. |
posd.tol |
tolerance for enforcing positive definiteness (in the
final |
maxit |
maximum number of iterations allowed. |
trace |
logical or integer specifying if convergence monitoring should be traced. |
This implements the algorithm of Higham (2002), and then (if
do2eigen
is true) forces positive definiteness using code from
'sfsmisc::posdefify()'. The algorithm of Knol and ten
Berge (1989) (not implemented here) is more general in that it
allows constraints to (1) fix some rows (and columns) of the matrix and
(2) force the smallest eigenvalue to have a certain value.
Note that setting corr = TRUE
just sets diag(.) <- 1
within the algorithm.
Higham (2002) uses Dykstra's correction, but the version by Jens
Oehlschlägel did not use it (accidentally),
and still gave reasonable results; this simplification, now only
used if doDykstra = FALSE
,
was active in nearPD()
up to Matrix version 0.999375-40.
unlike the matrix package, this simply returns the nearest positive definite matrix
Jens Oehlschlägel donated a first version to Matrix. Subsequent changes by the Matrix package authors, later modifications to C++ by Matthew Fidler
Cheng, Sheung Hun and Higham, Nick (1998) A Modified Cholesky Algorithm Based on a Symmetric Indefinite Factorization; SIAM J. Matrix Anal.\ Appl., 19, 1097–1110.
Knol DL, ten Berge JMF (1989) Least-squares approximation of an improper correlation matrix by a proper one. Psychometrika 54, 53–61.
Higham, Nick (2002) Computing the nearest correlation matrix - a problem from finance; IMA Journal of Numerical Analysis 22, 329–343.
A first version of this (with non-optional corr=TRUE
)
has been available as 'sfsmisc::nearcor()' and
more simple versions with a similar purpose
'sfsmisc::posdefify()'
set.seed(27) m <- matrix(round(rnorm(25),2), 5, 5) m <- m + t(m) diag(m) <- pmax(0, diag(m)) + 1 (m <- round(cov2cor(m), 2)) near.m <- lotriNearPD(m) round(near.m, 2) norm(m - near.m) # 1.102 / 1.08 round(lotriNearPD(m, only.values=TRUE), 9) ## A longer example, extended from Jens' original, ## showing the effects of some of the options: pr <- matrix(c(1, 0.477, 0.644, 0.478, 0.651, 0.826, 0.477, 1, 0.516, 0.233, 0.682, 0.75, 0.644, 0.516, 1, 0.599, 0.581, 0.742, 0.478, 0.233, 0.599, 1, 0.741, 0.8, 0.651, 0.682, 0.581, 0.741, 1, 0.798, 0.826, 0.75, 0.742, 0.8, 0.798, 1), nrow = 6, ncol = 6) nc <- lotriNearPD(pr)
set.seed(27) m <- matrix(round(rnorm(25),2), 5, 5) m <- m + t(m) diag(m) <- pmax(0, diag(m)) + 1 (m <- round(cov2cor(m), 2)) near.m <- lotriNearPD(m) round(near.m, 2) norm(m - near.m) # 1.102 / 1.08 round(lotriNearPD(m, only.values=TRUE), 9) ## A longer example, extended from Jens' original, ## showing the effects of some of the options: pr <- matrix(c(1, 0.477, 0.644, 0.478, 0.651, 0.826, 0.477, 1, 0.516, 0.233, 0.682, 0.75, 0.644, 0.516, 1, 0.599, 0.581, 0.742, 0.478, 0.233, 0.599, 1, 0.741, 0.8, 0.651, 0.682, 0.581, 0.741, 1, 0.798, 0.826, 0.75, 0.742, 0.8, 0.798, 1), nrow = 6, ncol = 6) nc <- lotriNearPD(pr)
This is used for creating nesting simulations in 'rxode2()' and may not be useful for external function calls.
lotriSep(x, above, below, aboveStart = 1L, belowStart = 1L)
lotriSep(x, above, below, aboveStart = 1L, belowStart = 1L)
x |
lotri matrix |
above |
Named integer vector listing variability above the id level. Each element lists the number of population differences in the whole data-set (as integer) |
below |
Named integer vector listing variability below the id level. Each element lists the number of items below the individual level. For example with 3 occasions per individual you could use 'c(occ=3L)' |
aboveStart |
Add the attribute of where THETA[#] will be added |
belowStart |
Add the attribute of where ETA[#] will be added |
List of two lotri matrices
Matthew Fidler
omega <- lotri(lotri(eta.Cl ~ 0.1, eta.Ka ~ 0.1) | id(nu=100), lotri(eye.Cl ~ 0.05, eye.Ka ~ 0.05) | eye(nu=50), lotri(iov.Cl ~ 0.01, iov.Ka ~ 0.01) | occ(nu=200), lotri(inv.Cl ~ 0.02, inv.Ka ~ 0.02) | inv(nu=10)) lotriSep(omega, above=c(inv=10L), below=c(eye=2L, occ=4L))
omega <- lotri(lotri(eta.Cl ~ 0.1, eta.Ka ~ 0.1) | id(nu=100), lotri(eye.Cl ~ 0.05, eye.Ka ~ 0.05) | eye(nu=50), lotri(iov.Cl ~ 0.01, iov.Ka ~ 0.01) | occ(nu=200), lotri(inv.Cl ~ 0.02, inv.Ka ~ 0.02) | inv(nu=10)) lotriSep(omega, above=c(inv=10L), below=c(eye=2L, occ=4L))
The RCM stands for the reverse Cuthill McKee (RCM) algorithm which is used to permute the matrix to a banded matrix.
rcm(x)
rcm(x)
x |
A symmetric matrix |
A permuted matrix that should be banded
m <- lotri({ a + b + c + d + e + f + g + h + i + j + k + l + m + n + o + p ~ c(0.4, 0, 0.3, 0, 0, 0, -0.1, 0, 0, 0.2, 0, 0, 0, 0, 0.5, 0, 0, 0, 0, 0, 1.3, 0, 0, 0, 0, 0, -0.6, 0.8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.9, 0, 0, 0, 0, 0, 0, 0, 0, 0, -0.2, 0, 0.3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2.1, 0.2, 0, 0, 0.2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.4, 0, 0, 0, 0, 0, -1.1, 0.9, 0, 0, 0, 0, 0, 0, 0, 4.7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.5, 0, 0.2, 0, 0, 0, 1.9) }) rcm(m)
m <- lotri({ a + b + c + d + e + f + g + h + i + j + k + l + m + n + o + p ~ c(0.4, 0, 0.3, 0, 0, 0, -0.1, 0, 0, 0.2, 0, 0, 0, 0, 0.5, 0, 0, 0, 0, 0, 1.3, 0, 0, 0, 0, 0, -0.6, 0.8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.9, 0, 0, 0, 0, 0, 0, 0, 0, 0, -0.2, 0, 0.3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2.1, 0.2, 0, 0, 0.2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.4, 0, 0, 0, 0, 0, -1.1, 0.9, 0, 0, 0, 0, 0, 0, 0, 4.7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.5, 0, 0.2, 0, 0, 0, 1.9) }) rcm(m)