Data Handling Overview#

%load ../../rapaio-bootstrap.ipynb
Adding dependency io.github.padreati:rapaio-lib:7.0.1
Solving dependencies
Resolved artifacts count: 1
Add to classpath: /home/ati/work/rapaio-jupyter-kernel/target/mima_cache/io/github/padreati/rapaio-lib/7.0.1/rapaio-lib-7.0.1.jar

Data types, arrays and collections#

Most programming languages have constructs for data and instructions or commands. This is true for both low and high level languages. If the language does not provide those kind of constructs, commonly used libraries, like system libraries supply them. This is true also for Java language. We start our discussion about data handling from what the language itself provides.

Java provides value and reference data types. There are few values types like: double, int, char and so on. For each value type there is also a corresponding refference boxing type like Double, Integer, Character. Since Java language is an object oriented language, it should be apparent the reason for existence of the corresponding refference types: abstractization and encapsulation. If we add how the generics were implemented in language on those reasons we understood why we have collections implemented as they are, with their rich offerings in terms of functionality. Still, value types are part of the language together with the construct called array. They seem so dulled and non idiomatic, so one may ask why there are still part of the language. The main reason of their existence is performance: both in terms of memory and computation.

Rapaio uses under the hood value types and arrays to achieve speed and memory efficience, carrying the burden of writting much more non idiomatic code. For example there are many vaue types array features in utilitary classes like rapaio.util.collection.DoubleArrays and similar. Those utilitary classes exists in order to offer support for high order constructs with compact and performant implementations.

Working directly with value type arrays, even having at one’s disposal a rich set of features, claims a lot of effort and attention. The resulting code would be probably very hard to write or read. Thus, Rapaio uses for most of the algorithms implemented higher constructs like rapaio.data.Var and rapaio.data.Frame to offer a rich and flexible set of features, easy to be used and leveraged further. Those constructs hides the complexity and intricaties implied at implementation times dues to working directly with value types vectors, without performance penalties.

Because those abstractions works directly with value type arrays, a lot of care was spent to offer many easy ways to transform data from value type array into variables, frames, vectors and matrices and viceversa, sometimes avoiding copying the data entirely. Java collections are also well conected with Rapaio data structures, thus being very easy to transfer data to and from those collections.

Var and Frame#

There are two ubiquous data structures used all over the place in this library: variables and frames. A variable is a list of values of the same type which implements the Var interface. You can think of a variable as a higher abstractization of an array because all values have the same type and there is random acces available. In fact, sometimes, the arrays are the only data storage for a Var implementation. Another way to look at a variable is like a column of a table.

A set of variables makes a data frame, described by the interface rapaio.data.Frame. A frame is a table having observations as rows and variables as columns.

Let’s take a simple example. We will load the iris data set, which is provided by the the library.

Frame df = Datasets.loadIrisDataset();
df.printSummary();
Frame Summary
=============
* rowCount: 150
* complete: 150/150
* varCount: 5
* varNames: 

0. sepal-length : dbl | 3.  petal-width : dbl | 
1.  sepal-width : dbl | 4.        class : nom | 
2. petal-length : dbl | 

* summary: 
 sepal-length [dbl]      sepal-width [dbl]      petal-length [dbl]      petal-width [dbl]     
       Min. : 4.3000000       Min. : 2.0000000        Min. : 1.0000000       Min. : 0.1000000 
    1st Qu. : 5.1000000    1st Qu. : 2.8000000     1st Qu. : 1.6000000    1st Qu. : 0.3000000 
     Median : 5.8000000     Median : 3.0000000      Median : 4.3500000     Median : 1.3000000 
       Mean : 5.8433333       Mean : 3.0573333        Mean : 3.7580000       Mean : 1.1993333 
    2nd Qu. : 6.4000000    2nd Qu. : 3.3000000     2nd Qu. : 5.1000000    2nd Qu. : 1.8000000 
       Max. : 7.9000000       Max. : 4.4000000        Max. : 6.9000000       Max. : 2.5000000 
                                                                                              

       class [nom] 
versicolor :    50 
    setosa :    50 
 virginica :    50 
                   
                   
                   
                   

Frame summary is a simple way to see some general information about a data frame. We see the data frame contains \(150\) observations/rows and \(5\) variables/columns.

The listing continues with name and type of the variables. Notice that there are four double variables and one nominal variable, named class.

The summary listing ends with a section which describes each variable. For double variables the summary contains the well-known six number summary. We have there the minimum and maximum values, median, first and third quartile and the mean. For nominal values we have an enumeration of the first most frequent levels and the associated counts. For our class variable we see that there are three levels, each with \(50\) instances.

Variables#

In statistics a variable has multiple meanings. A random variable can be thought as a process which produces values according to a distribution. Var objects models the concept of values drawn from a unidimensional random variable, in other words a sample of values. As a consequence a Var object has a size and uses integer indices to access values from sample.

Var interface declares methods useful for various kinds of tasks:

  • manipulate values from the variable by adding, removing, inserting and updating with different representations

  • naming a variable offers an alternate way to identify a variable into a frame and it is also useful as output information

  • manipulate sets of values by with concatenation and mapping

  • streaming allows traversal of variables by java streams

  • numeric computations like mathematical operations or variuos statistical interesting characteristics

  • unique value and grouping with a flexible grammar and short syntax

  • other tools like deep copy, deep compare, summary, etc

VarType: storage and representation of a variable#

There are two main concepts which have to be understood when working with variables: storage and representation. All the variables are able to store data internally using a Java data types like double, int, String, etc. In the same time, the data from variables can be represented in different ways, all of them being available through the Var interface for all types of variables.

However not all the representations are possible for all types of variables, because some of them does not make sense. For example double floating values can be represented as strings, which is fine, however strings in general cannot be represented as double values.

These are the following data representations all the Var-iables can implement:

  • double - double

  • int - int

  • long - long

  • instant - Instant

  • label - String

The Var interface offers methods to get/update/insert values for all those data representations. Not all data representations are available for all variables. For example the label representation is available for all sort of variables. This is acceptable, since when storing information into a text-like data format, any data type should be transformed into a string and also should be able to be read from a string representation.

To accomodate all those legal possibilities, the rapaio library has a set of predefined variable types, which can be found in the enum VarType.

The defined variable types:

VType

Var class

Description

BINARY

VarBinary

Binary variable represented as int values, internally uses bitsets for efficient memory usage

INT

VarInt

Integer variable represented and stored internally as int

NOMINAL

VarNominal

Categorical variable represented as string from a predefined set, with no ordering (for example: male, female)

DOUBLE

VarDouble

Double variable represented and stored internally as double precision floating point values

LONG

VarLong

Long variable represented and stored internally as long 8-byte signed integer values

INSTANT

VarInstant

Instant variable represented and stored as datetime instant

STRING

VarString

String variable used for manipulation of text with free form

A data type is important for the following reasons:

  • gives a certain useful meaning for variables in such a way that machine learning or statistical algorithms can leverage to maximum potential the meta information about variables

  • encapsulates the stored data type artifacts and hide those details from the user, while allowing the usage of a single unified interface for all variables

Frames#

The rapaio.data.Frame is the most common way to handle data. A data frame is a collection of variables organized in rows and columns. One can see the rows of a data frame as instances or observations having various attributes. The columns of a data frame are variables described by rapaio.data.Var.

There is a rich collection of methods offered by a data frame, allowing one to manipulate data in various ways for many purposes. Among those facilities to manipulate data one can have: views by data filtering, frame composition by merging rows or variables or other frames, grouping and group aggregates, streaming and so on.

Solid frames and view frames#

There are more than one implementation of a data frame for different purposes. The most encountered data frame is a SolidFrame. Solid data frames represents data stored in dense solid variables. This allows fast operations since data is stored compact in memory and it is ussually produced when data is loaded from a data storage, by allocating space for new data or by creating a new copy of data from a view frame. A view frame is a type of data frame which does not store itself data, but it is a wrapper on data stored somewhere else, usually in other data frames.

During data handling operations the library tries to not create copies of data, if possible. This give some freedom to the user to decide when data should be copied. We can illustrate with an example:

// create a solid data frame with one variable
var solid = SolidFrame.byVars(VarDouble.copy(1, 2, 3, 4, 5, 6).name("x"));
print(solid);

// filter some rows based on values and obtain a view filter
var view = solid.stream().filter(s -> s.getDouble("x") % 2 == 0).toMappedFrame();
print(view);

// we will bind the rows with old data frame
var bound = solid.bindRows(view);
print(bound);

// we can create o copy of the data to not alter the original one with updates
var copy = bound.copy();
print(copy);
rapaio.data.SolidFrame(rowCount=6, varCount=1)
VarDouble [name:"x", rowCount:6, values: 1.0, 2.0, 3.0, 4.0, 5.0, 6.0]
rapaio.data.MappedFrame(rowCount=3, varCount=1)
MappedVar[type=dbl, name:x, rowCount:3]
rapaio.data.BoundFrame(rowCount=9, varCount=1)
BoundVar(type=dbl) [name:"x", rowCount:9, values: 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 2.0, 4.0, 6.0]
rapaio.data.SolidFrame(rowCount=9, varCount=1)
VarDouble [name:"x", rowCount:9, values: 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 2.0, 4.0, 6.0]

Frames allows manipulation of data sets in a flexible way to infer various knowledge from data. Those abjects are used for most of the machine learning models and for input and output facilities.

Variables and frames are also used in almost all tools from statistical hypothesis tests to graphical plots. There is also the possibility to work with vectors and matrices, objects used in linear algebra. Transition between those types of data structures can be done easily, even if sometimes data transformation can happen due to linear algebra object constrains (all values from a vector or matrix should have the same type).