---
title: "jmatrix"
output: rmarkdown::html_vignette
bibliography: jmatrix.bib
vignette: >
%\VignetteIndexEntry{jmatrix}
%\VignetteEngine{knitr::rmarkdown}
%\VignetteEncoding{UTF-8}
---
```{r, include = FALSE}
knitr::opts_chunk$set(
collapse = TRUE,
comment = "#>"
)
```
# Loading package
```{r setup}
library(jmatrix)
```
# Purpose
The package `jmatrix` (@R-jmatrix) was originally conceived as a tool for other
packages, namely `parallelpam` (@R-parallelpam) and `scellpam` (@R-scellpam) which
needed to deal with very big matrices which might not fit in the memory of the
computer, particularly if their elements are of type `double` (in most modern
machines, 8 bytes per element) whereas they could fit if they were matrices of
other data types, in particular of floats (4 bytes per element).
Unfortunately, R is not a strongly typed language. Double is the default
type in R and it is not easy to work with other data types. Trials like
the package float (@R-float) have been done, but to use them you have to coerce a
matrix already loaded in R memory to a float matrix, and then you can delete it.
But, what happens if you computer has not memory enough to hold the matrix in
the first place?. This is the problem this package tries to address.
Our idea is to use the disk as temporarily storage of the matrix in a
file with a internal binary format (`jmatrix` format). This format has a
header of 128 bytes with information like type of matrix (full, sparse or
symmetric), data type of each element (char, short, int, long, float, double or
long double), number of rows and columns and endianness; then comes the content
as binary data (in sparse matrices zeros are not stored; in symmetric matrices
only the lower-diagonal is stored) and finally the metadata (currently,
names for rows/columns if needed and an optional comment).
Such files are created and loaded by functions written in C++ which are
accessible from `R` with Rcpp (@R-Rcpp). The file, once loaded, uses strictly the
needed memory for its data type and can be processed by other C++ functions
(like the PAM algorithm or any other numeric library written in C++) also
from inside `R`.
The matrix contained in a binary data file in `jmatrix` format cannot be
loaded directly in `R` memory as a `R` matrix (that would be impossible,
anyway, since precisely this package is done for the cases in which such
matrix would NOT fit into the available RAM). Nevertheless, limited access
through some functions is provided to read one or more rows or one or
more columns as `R` vectors or matrices (obviously, coerced to double).
The package `jmatrix` must not be considered as a final, finished
software. Currently is mostly an instrumental solution to address our needs and
we make available as a separate package just in case it could be useful for
anyone else.
# Workflow
## Debug messages
First of all, the package can show quite informative (but sometimes
verbose) messages in the console. To turn on/off such messages you can use.
```{r}
JMatrixSetDebug(TRUE)
# Initially, state of debug is FALSE.
```
## Data storage
As stated before, the binary matrix files should normally be created
from C++ getting the data from an external source
like a data file in a format used in bioinformatics or a .csv file.
These files should be read by chunks. As an example,
look at function `CsvToJMat` in package `scellpam`.
As a convenience and only for testing purposes (to be used in this vignette),
we provide the function `JWriteBin` to write a
R matrix as a `jmatrix` file.
```{r}
# Create a 6x8 matrix of random values
Rf <- matrix(runif(48),nrow=6)
# Set row and column names for it
rownames(Rf) <- c("A","B","C","D","E","F")
colnames(Rf) <- c("a","b","c","d","e","f","g","h")
# Let's see the matrix
Rf
# and write it as the binary file Rfullfloat.bin
JWriteBin(Rf,"Rfullfloat.bin",dtype="float",dmtype="full",
comment="Full matrix of floats")
# Also, you can write it with double data type:
JWriteBin(Rf,"Rfulldouble.bin",dtype="double",dmtype="full",
comment="Full matrix of doubles")
```
To get information about the stored file the function `JMatInfo` is provided.
Of course, this funcion does not read the
complete file in memory but just the header.
```{r}
# Information about the float binary file
JMatInfo("Rfullfloat.bin")
# Same information about the double binary file
JMatInfo("Rfulldouble.bin")
```
A jmatrix binary file can be exported to .csv/.tsv table. This is done with
the function `JMatToCsv`
```{r}
# Create a 6x8 matrix of random values
Rf <- matrix(runif(48),nrow=6)
# Set row and column names for it
rownames(Rf) <- c("A","B","C","D","E","F")
colnames(Rf) <- c("a","b","c","d","e","f","g","h")
# Store it as the binary file Rfullfloat.bin
JWriteBin(Rf,"Rfullfloat.bin",dtype="float",dmtype="full",
comment="Full matrix of floats")
# Save the content of this .bin as a .csv file
JMatToCsv("Rfullfloat.bin","Rfullfloat.csv",csep=",",withquotes=FALSE)
```
The generated file will not have quotes neither around the column names (in its first line)
nor around each row name (at the beginning of each line) since withquotes is FALSE but it
can be set to TRUE for the opposite behavior. Also, a .tsv (tabulator separated values)
would have been generated using csep="\\t".
Also, a jmatrix binary file can also be generated from a .csv/.tsv file.
Such file must have a first line with the names of the columns (possibly surrounded by double quotes,
including a first empty double-quote, since the column of row names has no name itself).
The rest of its lines must start with a string (possibly surrounded by double quotes) with the row
name and the values. In all cases (first line and data lines) each column must be separated
from the next by a separation character (usually, a comma). No separation character must be
added at the end of each line. This format is compatible with the .csv generated by R with
the function `write.csv`.
The function to read .csv files is `CsvToJMat`
```{r}
# Create a 6x8 matrix of random values
Rf <- matrix(runif(48),nrow=6)
# Set row and column names for it
rownames(Rf) <- c("A","B","C","D","E","F")
colnames(Rf) <- c("a","b","c","d","e","f","g","h")
# Save it as a .csv file with the standard R function...
write.csv(Rf,"rf.csv")
# ...and read it to create a jmatrix binary file
CsvToJMat("rf.csv","rf.bin",mtype="full",csep=",",ctype="raw",valuetype="float",transpose=FALSE,comment="Test matrix generated reading a .csv file")
# Let's see the characteristics of the binary file
JMatInfo("rf.bin")
```
### Special note for symmetric matrices:
The parameter mtype="symmetric" will consider the content of the .csv file as a symmetric matrix. This implies that
it must be a square matrix (same number of rows and columns) but the upper-diagonal matrix that must be present
(it does not matter with which values) will be read, and immediately ignored, i.e.: only the lower-diagonal matrix
(including the main diagonal) will be stored.
## Data load
As stated before, no function is provided to read the whole matrix in
memory which would contradict the philosophy of this package,
but you can get rows or columns from a file.
```{r}
# Reads row 1 into vector vf. Float values inside the file are
# promoted to double.
(vf<-GetJRow("Rfullfloat.bin",1))
```
Obviously, storage in float provokes a loosing of precision. We have
observed this not to be relevant for `PAM` (partitioning around medoids)
algorihm but it can be important in other cases. It is the price to pay
for halving the needed space.
```{r}
# Checks the precision lost
max(abs(Rf[1,]-vf))
```
Nevertheless, storing as double obviously keeps the data intact.
```{r}
vd<-GetJRow("Rfulldouble.bin",1)
max(abs(Rf[1,]-vd))
```
Now, let us see examples of some functions to read rows or columns by
number or by name, or to read several rows/columns as a R matrix.
In all examples numbers for rows and columns are in R-convention (i.e.
starting at 1)
```{r}
# Read column number 3
(vf<-GetJCol("Rfullfloat.bin",3))
# Test precision
max(abs(Rf[,3]-vf))
# Read row with name C
(vf<-GetJRowByName("Rfullfloat.bin","C"))
# Read column with name c
(vf<-GetJColByName("Rfullfloat.bin","c"))
# Get the names of all rows or columns as vectors of R strings
(rn<-GetJRowNames("Rfullfloat.bin"))
(cn<-GetJColNames("Rfullfloat.bin"))
# Get the names of rows and columns simultaneosuly as a list of two elements
(l<-GetJNames("Rfullfloat.bin"))
# Get several rows at once. The returned matrix has the rows in the
# same order as the passed list,
# and this list can contain even repeated values
(vm<-GetJManyRows("Rfullfloat.bin",c(1,4)))
# Of course, columns can be extrated equally
(vc<-GetJManyCols("Rfulldouble.bin",c(1,4)))
# and similar functions are provided for extracting by names:
(vm<-GetJManyRowsByNames("Rfulldouble.bin",c("A","D")))
(vc<-GetJManyColsByNames("Rfulldouble.bin",c("a","d")))
```
The package can manage and store sparse and symmetric matrices, too.
```{r}
# Generation of a 6x8 sparse matrix
Rsp <- matrix(rep(0,48),nrow=6)
sparsity <- 0.1
nnz <- round(48*sparsity)
where <- floor(47*runif(nnz))
val <- runif(nnz)
for (i in 1:nnz)
{
Rsp[floor(where[i]/8)+1,(where[i]%%8)+1] <- val[i]
}
rownames(Rsp) <- c("A","B","C","D","E","F")
colnames(Rsp) <- c("a","b","c","d","e","f","g","h")
# Let's see the matrix
Rsp
# Write the matrix as sparse with type float
JWriteBin(Rsp,"Rspafloat.bin",dtype="float",dmtype="sparse",
comment="Sparse matrix of floats")
```
Notice that the condition of being a sparse matrix and the storage space
used can be known with the matrix info.
```{r}
JMatInfo("Rspafloat.bin")
```
Be careful: trying to store as sparse a matrix which is not (it has not
a majority of 0-entries) works, but produces
a matrix larger than the corresponding full matrix.
With respect to symmetric matrices, `JWriteBin` works the same way.
Let us generate a $7 \times 7$ symmetric matrix.
```{r}
Rns <- matrix(runif(49),nrow=7)
Rsym <- 0.5*(Rns+t(Rns))
rownames(Rsym) <- c("A","B","C","D","E","F","G")
colnames(Rsym) <- c("a","b","c","d","e","f","g")
# Let's see the matrix
Rsym
# Write the matrix as symmetric with type float
JWriteBin(Rsym,"Rsymfloat.bin",dtype="float",dmtype="symmetric",
comment="Symmetric matrix of floats")
# Get the information
JMatInfo("Rsymfloat.bin")
```
Notice that if you store a R matrix which is NOT symmetric as a symmetric
`jmatrix`, only the lower triangular part (including the
main diagonal) will be saved. The upper-triangular part will be lost.
The functions to read rows/colums stated before works equally independently
of the matrix character (full, sparse or symmetric) so
you can play with them using the `Rspafloat.bin` and `Rsymfloat.bin` file
to check they work.
Finally, if the jmatrix stored in a binary file has names associated to rows
or columns, you can filter it using them and generate another jmatrix file
with only the rows or columns you wish to keep. The function to do so is 'FilterJMatByName'.
```{r}
Rns <- matrix(runif(49),nrow=7)
rownames(Rns) <- c("A","B","C","D","E","F","G")
colnames(Rns) <- c("a","b","c","d","e","f","g")
# Let's see the matrix
Rns
# Write the matrix as full with type float
JWriteBin(Rns,"Rfullfloat.bin",dtype="float",dmtype="full",
comment="Full matrix of floats")
# Extract the first two and the last two columns
FilterJMatByName("Rfullfloat.bin",c("a","b","f","g"),"Rfullfloat_fourcolumns.bin",namesat="cols")
# Let's load the matrix and let's see it
vm<-GetJManyRows("Rfullfloat_fourcolumns.bin",c(1,7))
vm
```