Linear Least Squares Minimization#
The Problem#
We propose a linear functional relation between 2 measurable variables, \(x\) and \(y\):
where \(a_0\) and \(a_1\) are unknown constants. We wish to find these constants.
The Solution#
To find these unknown coefficients in practice we measure many \(x\), \(y\) pairs (assuming the measurements display some sort of dispersion). We now have a set of measured \((x_i, y_i)\) pairs for \(i = 1, 2, 3, \dots, N\).
If we assume that the \(x_i\) are free of error, we can introduce error terms \(\epsilon_i\) to the \(y_i\) data to make up for the dispersion of the data (i.e. that it doesn’t follow the linear relation exactly).
With this error term, the relation between our data points can be represented as:
\(y_i +\) \(\epsilon_i\) \(= a_0 + a_1 x_i\)
Note that, at this point the error terms we have introduced are unknown to us. They represent the difference between the measured \(y_i\) values and the expected values if we plugged \(x_i\) into our relation (for which we have yet to determine \(a_0\) and \(a_1\)). The error terms can be seen as a means to an end and will soon be done away with.
Now, we need some sort of metric to tell us how much error we have. We can use the sum of the errors squared for this:
We use the squares of the error as it is the magnitude of the errors we are concerned about, and with the errors ranging between positive and negative values, will end up canceling each other out (these are illustrated as points above and below the lines in the figure above).
We can use the relation between our data points to replace the \(\epsilon_i^2\):
Now, we want our choice of \(a_0\) and \(a_1\) to give us the least amount of error possible, or rather to give us the minimum value of \(S\). To achieve this we minimize \(S\) with respect to \(a_0\):
and \(a_1\):
To solve this system of equations we could use a matrix equation and let the computer determine the solution to that numerically, but with only two equations and unknowns, an analytic solution is easy enough to find:
Variance of \(y\)#
If we assume that the \(y_i\) data points are distributed around the “true” \(y\) values for the given \(x_i\) by a Gaussian distribution with constant variance, we can calculate the variance of \(y\) as:
Worked Example - Cepheid Variables
For this worked example we will use data from Cepheid variables. These are pulsating stars with their luminosity (or magnitude \(M\)) related to the period (\(P\)) of their pulsations:
Note that the relation above is can be made more accurate by including the color or temperature of the star, which we shall use later in the chapter.
As this relation is consistent across all specimens, these stars can be used as a standard candle for measuring distances, all that is needed are measurements from stars with known distances from Earth to determine \(a_0\) and \(a_1\).
The standard is to measure Cepheids in the Large Magellanic Cloud, whose distance is known. A few of these measurements can be found in the data file ‘cepheid_data.csv’ provided on Vula (Resources/Exercises/Data/exercise10.1/) or on GitHub. The data file contains measurements of:
\(\log P\)
\(M\)
\(B - V\) (color, not using yet)
We will determine \(a_0\) and \(a_1\) under 2 different assumptions:
The error in the data is associated with \(M\)
The error in the data is associated with \(\log P\) (this will require us to re-arrange things)
Solution:
You are encouraged to attempt this yourself before continuing.
We start by reading in the file. We will read the data into a 2 arrays. This can be achieved using the standard library as in the page Python Standard Library/File IO/Data Files, or using numpy.loadtxt()
(documentation here). We shall use the latter as it is far more convenient:
import numpy as np
import matplotlib.pyplot as plt
logP, M, color = np.loadtxt('./data/cepheid_data.csv', delimiter = ',', skiprows = 1, unpack = True)
The keyword arguments used above are:
delimeter
: the string used to separate the data columnsskiprows
: the number of rows to skip from the data file (in this case the header)unpack
: this makesloadtxt
return each column in the data file as arrays, as apposed to the default of a single 2D array.
As we will be performing 2 minimizations, we will define a function to determine \(a_0\) and \(a_1\), and \(\sigma_y\):
def least_squares(x, y):
mean_x = np.mean(x)
mean_y = np.mean(y)
#Note that the expectation values can be calculated using the mean function
expect_xy = np.mean(x*y)
expect_xx = np.mean(x*x)
a1 = (expect_xy - mean_x*mean_y)/(expect_xx - mean_x*mean_x)
return [mean_y - a1*mean_x, a1]
def sigma(a0, a1, x, y):
return np.sqrt(np.mean((a0 + a1*x - y)**2))
Error in \(M\)
Let’s estimate the coefficients for the relation:
assuming the error resides primarily in \(M\).
a0, a1 = least_squares(logP, M)
Error in \(\log P\)
Now, if we assumed that the error resides primarily in \(\log P\), we want to apply least squares minimization to the relation:
to find the values for the coefficients \(b_0\) and \(b_1\). These values can be used to calculate \(a_0\) and \(a_1\) by re-arranging the relation to put \(M\) as the subject:
which gives us:
b0, b1 = least_squares(M, logP)
Plotting the solutions
Instead of printing out the values of the coefficients, let’s visualize them by plotting the linear relations.
fontsize = 16
linewidth = 2
x = np.array([logP[0], logP[-1]]) #for the relation
y_M = a0 + a1*x #error in M
y_P = -b0/b1 + x/b1 #error in logP
fig_ceph, ax = plt.subplots()
ax.plot(logP, M, 'ro', label = 'Data')
ax.plot(x, y_M, 'k', label = 'Error in M', lw = linewidth)
ax.plot(x, y_P, 'k--', label = 'Error in logP', lw = linewidth)
ax.set_xlabel('logP', fontsize = fontsize)
ax.set_ylabel('M', fontsize = fontsize)
ax.legend(fontsize = fontsize)
plt.show()