Linear Regression in R, exploring a real-world dataset
In this article we will explore the fundamental concepts of linear regression using a real-world dataset. We will use the Bottle Database that contains a collection of oceanographic data, including temperature, salinity, potential density and others.
This article does not aim to be a comprehensive introduction to linear regression, but rather a practical guide to get you started with the technique in real-world problems. It's important to note that although more advanced techniques may exist for certain variable relationships, we will focus on handpicking variables that are most suitable for linear regression.
Review of Linear Regressionlink
In its most general form, a regression model assumes that the response can be expressed as a function of the predictors variables plus an error term :
In this form estimating can be quite challenging, even asuming it is smoth an continuos function. There are infinitely many possible functions that can be used, and the data alone will not tell us which one is the best [1]. This is where linearity comes in. By assuming that the relationship between the predictors and the response is linear, we can greatly simplify the modeling process. Instead of estimating the entire function , we only need to estimate the unknown parameters in the linear regression equation:
Here are unknown parameters, ( is also called the intercept or bias term). Note that the predictors themselves do not have to be linear. For example, we can have a model of the form:
which is still linear. Meanwhile:
is not. Although the name implies a straight line, linear regression models can actually be quite flexible and can handle complex datasets. To capture non-linear relationships between predictors and the response variable, quadratic terms can be included in the model [2]. Additionally, nonlinear predictors can be transformed to a linear form through suitable transformations, such as logarithmic transformation, which can linearize exponential relationships.
In practice almost all relationships could be represented or reduced to a linear form. This is why linear regression is so widespread and popular. It is a simple yet powerful technique.
Matrix representationlink
To make it easier to work with, lets present our model of a response and predictors in a tabular form:
where is the number of observations or cases in the dataset. Given the actual data values, we may write the model as:
We can write this more conveniently in matrix form as:
where is the response vector, is the error vector, is the parameter vector, and:
is the design matrix. Note that the first column of is all ones. This is to account for the intercept term .
Estimating the parameterslink
In order to achieve the best possible fit between our model and the data, our objective is to estimate the parameters denoted as . Geometrically, this involves finding the vector that minimizes the difference between the product and the target variable . We refer to this optimal choice of as , which is also known as the regression coefficients.
To predict the response variable, our model uses the equation or, equivalently, , where represents an orthogonal projection matrix. The predicted values are commonly referred to as the fitted values. On the other hand, the discrepancy between the actual response variable and the fitted response is referred to as the residual.
Least Squares Estimationlink
Estimating can be approached from both a geometric and a non-geometric perspective. We define the best estimate of as the one that minimizes the sum of squared residuals:
To find this optimal estimate, we differentiate the expression with respect to and set the result to zero, yielding the normal equations:
Interestingly, we can also derive the same result using a geometric approach:
In this context, represents the hat matrix, which serves as the orthogonal projection matrix onto the column space spanned by . While is useful for theoretical manipulations, it is typically not computed explicitly due to its size (an matrix). Instead, the Moore-Penrose pseudoinverse of is employed to calculate efficiently.
Note we won't explicitly calculate the inverse matrix here, but instead use the built-in R lm
function.
About the datasetlink
The CalCOFI dataset is an extensive and comprehensive resource, renowned for its long-running time series spanning from 1949 to the present. With over 50,000 sampling stations, it stands as the most complete repository of oceanographic and larval fish data worldwide.
Our interest is figure out if there are variables that exhibit a linear relationship with the temperature of the water. So, let get started by loading the data getting a summary of the rows and columns.
rdata <- read.csv("bottle.csv")
head(data, 10)
You can check more about what means each column here.
Cleaning the datalink
To keep this blog post concise and reader-friendly, we'll narrow our focus to five key variables: temperature, salinity, dissolved oxygen per liter, depth in meters and potential density. These variables, will be our primary points of interest. However, don't hesitate to explore any other variables that catch your attention.
To maintain data integrity, it's worth mentioning that there are some missing values within the dataset. But we can easily handle this by using the na.omit()
function. It allows us to remove rows with missing values, ensuring a robust analysis.
rdata <- data[, c("T_degC", "Salnty", "O2ml_L", "Depthm, T_degC")]
data <- na.omit(data)
Checking for linearitylink
We want to assure that the linear model assumptions are met. This can be tested by calculating the correlation coefficients. A correlation coefficient near zero indicates that there is no linear relationship between the two variables, meanwhile, a correlation coefficient near -1 or 1 indicates that there is a strong negative or positive linear relationship. We can create a correlation matrix using the cor()
function.
rlibrary(corrplot)
correlation_matrix <- cor(data)
corrplot(correlation_matrix, method = "circle")
Let's narrow our attention to the variables that exhibit stronger correlation: T_degC and STheta. Since we have the flexibility to shape our exploration in this scenario, we can decide what aspects to delve into and how to approach them. However, in a real-world scenario, the variables to analyze are often dictated by the specific problem at hand.
To begin our analysis, let's plot these variables and examine if there exists a linear relationship between them. This initial visualization will provide us with valuable insights into their potential connection.
rplot(data$T_degC, data$STheta,
main = "Temperature vs Potential Density (Sigma Theta)")
Looking for outlierslink
Outliers are data points that are far from other data points. They can be caused by experimental errors, data corruption or other anomalies. Outliers can have a large effect on the linear regression model, since they can disproportionately influence the model's fit. We can use the boxplot()
function to visualize the outliers.
rboxplot(data$T_degC, data$O2ml_L,
main = "Boxplot of temperature and dissolved oxigen per litter")
By observing the plot, it appears that there may be a few data points that could be considered outliers. However, in this analysis, we will choose to retain these points. They appear to be within a reasonable range and are not significantly distant from the median. If you are interested in exploring more precise techniques for outlier detection, you can refer to this helpful resource here.
rboxplot(data$T_degC, data$STheta,
main = "Boxplot of temperature and potential density")
Creating the modellink
To ensure an effective evaluation of our model's performance, we will divide the data into training and testing sets. This division allows us to train our model on a subset of the data and assess its accuracy on unseen data.
For this purpose, we will allocate 80% of the data for training and reserve the remaining 20% for testing. To achieve this, we can utilize the sample() function, which enables us to randomly select the rows that will constitute the training set. This random selection ensures a representative sample and avoids bias in our model training.
rtraining_rows <- sample(1:nrow(data), 0.8 * nrow(data))
training_data <- data[training_rows, ]
testing_data <- data[-training_rows, ]
Now we can create the linear model using the lm()
function. The first argument is the formula, which is the response variable (potential density energy) followed by the predictor variable (temperature). The second argument is the dataset that is going to be used to create the model.
rmodel <- lm(STheta ~ T_degC, data = training_data)
We can use the summary()
function to get a summary of the model.
rsummary(model)
bashCall:
lm(formula = STheta ~ T_degC, data = training_data)
Residuals:
Min 1Q Median 3Q Max
-3.3407 -0.1381 -0.0088 0.1076 2.0716
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 2.831e+01 1.020e-03 27769 <2e-16 ***
T_degC -2.305e-01 8.712e-05 -2645 <2e-16 ***
---
Signif. codes: 0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1
Residual standard error: 0.2676 on 529012 degrees of freedom
Multiple R-squared: 0.9297, Adjusted R-squared: 0.9297
F-statistic: 6.997e+06 on 1 and 529012 DF, p-value: < 2.2e-16
Based on the p-value of the coefficients, we can conclude that the intercept and the slope are statistically significant.
Making predictionslink
We can use the predict()
function to make predictions on the testing set. The first argument is the model, the second argument is the dataset that is going to be used to make the predictions. The predict()
function returns a vector of predictions.
rpredictions <- predict(model, testing_data)
We can use the plot()
function to plot the actual values vs the predicted values.
rplot(testing_data$T_degC, testing_data$STheta,
main = "Actual vs predicted values",
xlab = "Temperature", ylab = "Potential density")
points(testing_data$T_degC, predictions, col = "red")
The red points are the predicted values and the blue points are the actual values. We can see that the predicted values are close to the actual values.