Data Manipulation
Using functions to explore data structures
There are also selections of base functions in R that are useful for inspecting your data and summarizing it. Let’s start with a simple data structure such as vectors. A commonly used function is length()
, which tells you how many elements are in a particular vector:
The class()
function is useful in indicating the datatype or data structure of a variable. So for example if we were interested in knowing what was inside glengths
:
We could also use class on a data frame or any other type of object. Let’s load in a data frame to test out some more functions. We will use read.csv to read in data from a csv (comma separated values) file. There are numerous other functions to load in data depending on your filetype, but read.csv
is one of the more commonly used ones.
The function has one required argument and several options that can be changed. The mandatory argument is a path to the file and filename, which in our case is mouse_exp_design.csv
file. We will put the function to the right of the assignment operator, meaning that any output will be saved as the variable name provided on the left.
Take a look at the file by typing out the variable name metadata and pressing return. The file contains information describing the samples in our study. Each row holds information for a single sample, and the columns represent genotype
(WT or KO), celltype
(typeA or typeB), and replicate number.
Note: If you are using older versions of R:
By default, data.frame
converts (= coerces) columns that contain characters (i.e., text) into the factor data type. Depending on what you want to do with the data, you may want to keep these columns as character. To do so, read.csv()
and read.table()
have an argument called stringsAsFactors
which can be set to FALSE
.
Suppose we had a larger file, we might not want to display all the contents in the console. Instead we could check the top (the first 6 lines) of this data.frame using the function head()
:
Let’s now check the str
ucture of this data.frame
in more details with the function str()
:
If you are using older versions of R:
As you can see, the columns genotype
and celltype
are of a special class called factor whereas the replicate column has been interpreted as integer data type.
If you are using R 4.0.0 or newer:
We already saw how the functions head()
and str()
can be useful to check the content and the structure of a data.frame
. Here is a non-exhaustive list of functions to get a sense of the content/structure of the data.
Size:
dim()
- returns a vector with the number of rows in the first element, and the number of columns as the second element (thedim
ensions of the object)nrow()
- returns the number of rowsncol()
- returns the number of columns
Content:
head()
- shows the first 6 rowstail()
- shows the last 6 rows
Names:
names()
- returns the column names (synonym of colnames() for data.frame objects)rownames()
- returns the row names
Summary:
str()
- structure of the object and information about the class, length and content of each columnsummary()
- summary statistics for each column
Note: most of these functions are “generic”, they can be used on other types of objects besides data.frame.
Using indexes and sequences to select data from vectors and dataframes
When analyzing data, we often want to partition the data so that we are only working with selected columns or rows. A data frame or data matrix is simply a collection of vectors combined together. So let’s begin with vectors, then apply those concepts to dataframes.
Vectors
If we want to extract one or several values from a vector, we must provide one or several indexes in square brackets. The index represents the element number within a vector (or the compartment number, if you think of the bucket analogy). R indexes start at 1. Programming languages like Fortran, MATLAB, and R start counting at 1, because that’s what human beings typically do. Languages in the C family (including C++, Java, Perl, and Python) count from 0 because that’s simpler for computers to do.
Let’s start by creating a vector called age
:
Suppose we only wanted the fifth value of this vector, we would use the following syntax:
If we wanted to index more than one element we would still use the square bracket syntax, but rather than using a single value we would pass in a vector of the index values:
To access a sequence of continuous values from a vector, we would use :
` which is a special function that creates numeric vectors of integer in increasing or decreasing order. Let’s select the first five values from age:
Alternatively, if you wanted the reverse could try 5:1 for instance and see what is returned. The function seq()
(for seq
uence) can also be used to create sequences, but allow for more complex patterns. Passing in the by argument will allow you to generate a sequence based on the specified interval:
Additionally, the length.out parameter will provide the restriction on the maximum length of the resulting vector. A combination of parameters can also be used:
Dataframes
Dataframes have 2 dimensions (rows and columns), so if we want to extract some specific data from it we need to specify the “coordinates” we want from it. We use the same square bracket syntax but rather than providing a single index, there are two inputs required. Within the square bracket, row numbers come first followed by column numbers (and the two are separated by a comma). For example:
Now if you only wanted to select based on rows, you would provide the indexes for the rows and leave the columns blank. The key here is to include the comma, to let R know that you are accessing a 2 dimensional data structure:
Similarly, if you were selecting specific columns from the data frame - the rows are left blank:
For larger datasets, it can be tricky to remember the column number that corresponds to a particular variable. (Is celltype in column 1 or 3? oh, right… they are in column 2). In some cases, the column number for a variable can change if the script you are using adds or removes columns. It’s therefore often better to use column names to refer to a particular variable, and it makes your code easier to read and your intentions clearer.
You can do operations on a particular column, by selecting it using the $
sign. In this case, the entire column is a vector. For instance, to extract all the gentotypes from our dataset, we can use: metadata$genotype
. You can use names(metadata)
or colnames(metadata)
to remind yourself of the column names.
To select multiple columns by name the square bracket syntax is used by concatenating a vector of strings that correspond to column names:
Subsetting data using logical operators
Another way of partitioning your data, is by filtering based on the content that is in your dataframe using the subset()
function. For example, we can look at the samples of a specific celltype
"typeA
":
We can also subset using other logical operators in R. For example suppose we wanted to subset to keep only the WT samples from the typeA
celltype
.
Alternatively, we could try looking at only the first two replicates of each sample set. Here, we can use the less than operator since replicate is currently a numeric vector. Adding in the argument select allows us to specify which columns to keep. Which columns are left?
Re-organizing data based on match indices
Often when working with genomic data, we have a data file that corresponds with our metadata file. The data file contains measurements from the biological assay for each individual sample. In our case, the biological assay is gene expression and data was generated using RNA-Seq. Let’s bring in the data matrix of RPKM values:
Take a look at the first few lines of the data matrix to see what’s in there.
It looks as if the sample names (header) in our data matrix are similar to the row names of our metadata file, but it’s hard to tell since they are not in the same order. We can do a quick check of the dimensions and at least see if the numbers match up.
What we want to know is, do we have data for every sample that we have for metadata?
There are many ways to answer this using R. We’ll be using the match function, which takes at least 2 arguments: 1) a vector of values to be matched, and 2) a vector of values to be matched against. The function returns the position of the matches in the second vector. Take a look at the example below where vector B is the reverse of vector A:
Now if we use the match function with A as our first input and B as our second, you will be returned a vector of size length(A). Each number that is returned represents the index of vector B where the matching values was observed.
Let’s change vector B so that only a subset are retained:
Note, for values that don’t match you can specify what values you would have it assigned using nomatch argument (by default this is set to NA). Also, illustrated in the example, if there is more than one matching value found only the first is reported.
Subset data using matching
We are trying to match the row names of our metadata with the column names of our expression data, so these will be the arguments for match. There are base functions in R which allow you to extract the row and column names as a vector:
Using these two arguments we will retrieve a vector of match indices. This vector represents the re-ordering of the column names in our data matrix to be identical to the rows in metadata:
Now we can create a new data matrix in which columns are re-ordered based on the match indices:
Check and see what happened by using head. You can also verfy that column names of this new data matrix matches the metadata row names by using the all function:
An R package for data manipulation
The package dplyr
is a package that tries to provide easy tools for the most common data manipulation tasks. It is built to work directly with data frames. The thinking behind it was largely inspired by the package plyr
which has been in use for some time but suffered from being slow in some cases. dplyr
addresses this by porting much of the computation to C++. An additional feature is the ability to work with data stored directly in an external database. The benefits of doing this are that the data can be managed natively in a relational database, queries can be conducted on that database, and only the results of the query returned.
Last updated