Predicting Titanic Survivors: My First Machine Learning Model

August 14, 2018

8 min read

I’ve been taking the Machine Learning course by Dr. Andrew Ng online through Coursera, and so far it’s been an enlightening and enjoyable experience. While I’ve been taking lots of notes on the equations and implementation details on linear and logistic regression models, I thought it would be a fun weekend exercise to implement these algorithms from scratch using my favorite toolkit: Python, Pandas, and Numpy.

Special thanks to Dr. Ng and his course staff for preparing such high-quality course material. Many of these equations and explanations are based on notes from the course lectures, and I have paraphrased many of the explanations here in my own words as best as possible.

The Dataset

The dataset I decided to work with for this project was the was the Kaggle Titanic dataset, which was recommended as a beginner dataset to work with.

Titanic Dataset Data

Figure 1: Structure of the dataset

Based on the train.csv file included in the dataset download, it was clear that this was a binary classification problem. Since I only know how to implement linear and logistic regression models so far, I decided to use logistic regression for this problem.

Project Structure

For reference, my project directory was structured like this:

.
├── Makefile
├── data
│   ├── gender_submission.csv
│   ├── test.csv
│   └── train.csv
└── src
    └── Titanic.ipynb

The Makefile in the main directory has a fake target to make launching jupyter lab easier.

Preparing the Data

I first needed to clean the data before I could use it. In order to do that, I first brought the data into a pandas DataFrame:

import pandas as pd
df = pd.read_csv("../data/train.csv")

Then, I dropped the columns that I couldn’t convert into numbers:

df = df.drop(columns=['Name', 'PassengerId', 'Ticket', 'Cabin'])

Note: Having fewer features means the model was less accurate than it could be. Some clever feature engineering would have made use of these columns somehow, but as I was a beginner, I didn’t really know what to do with these features other than throw them out.

Next, I converted all of the textual features into numerical values roughly within the range $(-1, 1)$:

def sex_to_numerical(s):
    return int(g == "female")
def embarkation_to_numerical(e):
    if (e == "C"):
        return -1
    elif (e == "Q"):
        return 0
    else:
        return 1
df.Sex = df.Sex.apply(sex_to_numerical)
df.Embarked = df.Embarked.apply(embarkation_to_numerical)

I then normalized the data using a simple approach: $(\text{value} - \text{max})/\text{range}$ where $\text{range} = \text{max} - \text{min}$.

def normalize_ticket_class(cl):
    return (cl - 3) / 2
def normalize_fare(farecol):
    range_f = farecol.max() - farecol.min()
    max_f = farecol.max()
    return farecol.apply(lambda fare: (fare - max_f)/range_f)
df.Fare = normalize_fare(df.Fare)
df.Pclass = df.Pclass.apply(normalize_ticket_class)

I needed to convert all missing/NaN values in the Age column to numbers so that I didn’t need to throw that feature out:

def de_nan_age(a):
    if (np.isnan(a)):
        return 0.0
    else:
        return a
df.Age = df.Age.apply(de_nan_age)

After all this processing, the DataFrame looks like this:

IndexSurvivedPclassSexAgeSibSpParchFareEmbarked
000.0022.010-0.9858489424377921
11-1.0138.010-0.8608642646173593-1
210.0126.000-0.98453143018200021
31-1.0135.010-0.89635570254437961
400.0035.000-0.98428744643092761

Now, I’ll split up the DataFrame into two np.arrays, one with the features, the other with the labels:

features = df.drop(columns=['Survived']).values
labels = df.Survived.values

Add an $x_0 = 1$ column to the features array so everything looks clean:

x_zero_column = np.zeros((features.shape[0], 1)) + 1
features = np.append(x_zero_column, features, axis=1)

Using the features and labels arrays, create a design matrix $X$ and the label vector $y$:

X = np.asmatrix(features)
# y needs to be transposed from a row vector to a column vector
y = np.asmatrix(labels).T

The data is now ready to be used.

Define Variables

Using $X$ and $y$:, it’s time to define some variables that are important to the implmentation of the cost function $J$:

# m is the number of training samples
m = X.shape[0]
# n is the number of features, in this case we are not counting the x_0 column
n = X.shape[1] - 1

Let’s define the parameter $\theta$ as a $n+1$-dimensional vector, initially filled with zeros:

theta = np.asmatrix(np.zeros(n + 1)).T

And let’s set a learning rate $\alpha$, in this case I just chose an arbitrary value:

alpha = 0.001

It’s important to choose the right $\alpha$, because a smaller value may make the model slower to converge, while a larger value might make the model never converge.

Define Functions

The Sigmoid Function

One of the most important equations needed for logistic regression is the sigmoid function, which looks like this:

The equation for the sigmoid function is

$$ g(z) = \frac{1}{1+ e^{-z}} $$

Thanks to numpy, it’s very simple to define an element-wise vectorized sigmoid function, which will our code much more efficient:

def sigmoid(z):
    t = np.exp(-z) + 1
    return np.divide(1, t)

The Hypothesis Function

To generate a matrix of predictions using the parameter vector $\theta$, we define our hypothesis function $h_{\theta}(x)$ as follows:

$$ h_{\theta}(x) = g(X\theta) $$

where $X$ is the design matrix.

But what does the output of this function mean? $h_{\theta}(x)$ is actually outputting the probability that the output is 1. Formally, this function gives you the probablility that $y=1$, given $x$, parameterized by $\theta$ (mathematically, $P(y=1|x;\theta)$).

In Python, we can define the hypothesis function similarly:

def hypothesis(x):
    return sigmoid(x * theta)

The Cost Function

In order for the logistic regression algorithm to learn, it must have some function to minimize. This function is called the cost or loss function.

For logistic regression, the cost function is defined as:

$$ J(\theta) = \frac{1}{m} \bigg(-y^T \log{h(x)}- (1-y^T)\log{(1-h(x))}\bigg) $$

The Python code for this function came out to be too long to write out on one line, so the math is done on three separate lines.

Note: cost came out to be a 1x1 numpy matrix and was converted into a singular floating-point value at the end.

def cost():
    hypo = hypothesis(X)
    part1 = -(y.T) * np.log(hypo)
    part2 = (1-(y.T)) * np.log(1 - hypo)
    cost = (1/m) * (part1 - part2)
    return cost[0,0]

The Gradient Descent Function

The last (and probably most important) piece of this machine learning algorithm is the gradient_descent_step function, which calculates the gradient for $\theta$.

The update rule for $\theta$ is as follows:

$$ \theta := \theta - \frac{\alpha}{m}X^T(g(X\theta)-\vec{y}) $$

This is simple to implement in Python as a one-liner:

def gradient_descent_step(theta):
    return theta - (alpha/m)*(X.T)*(hypothesis(X) - y)

With all the necessary functions implemented, it’s time to actually learn the parameters $\theta$.

Run the Learning Algorithm

To run the learning algorithm, let’s create a loop in which gradient_descent_step is run every iteration:

for i in range(1630000):
    theta = gradient_descent_step(theta)

I found experimentally that after 1.63 million iterations, the cost delta after each iteration of Gradient Descent is negligible.

After the loop was run, the cost was approximately 0.45085533814736584.

Before we do anything else, let’s save the optimal $\theta$ in a file:

np.savetxt("theta.txt", theta)

Once saved, it can be loaded back like this:

theta = np.asmatrix(np.loadtxt("theta.txt"))
theta = theta.T

Run the Model on Test Data

Create the Prediction Function

Using the hypothesis(x) function, let’s define a function that outputs 1 if $P(y=1|x;\theta) ≥ 0.5$ or 0 otherwise:

def predict(x):
    h = hypothesis(x)
    comp = (h >= 0.5)
    return comp.astype(int)

predict(x) runs predictions on a matrix element-wise and converts them into numerical binary values.

Import the Test Data

Now, it’s time to bring in the test data, preparing it like we did for the training data:

test_df = pd.read_csv("../data/test.csv")
test_df = test_df.drop(columns=['Name', 'PassengerId', 'Ticket', 'Cabin'])
test_df.Pclass = test_df.Pclass.apply(normalize_ticket_class)
test_df.Sex = test_df.Sex.apply(sex_to_numerical)
test_df.Fare = normalize_fare(test_df.Fare)
test_df.Embarked = test_df.Embarked.apply(embarkation_to_numerical)
test_df.Age = test_df.Age.apply(de_nan_age)

Let’s add in the $x_0$ column like before:

test_X = test_df.values
x_zero_column_test = np.zeros((test_df.shape[0], 1)) + 1
test_X = np.append(x_zero_column_test, test_X, axis=1)
test_X = np.asmatrix(test_X)

Generate Predictions

Using the test_X matrix, let’s generate predictions and save them to a file:

predictions = predict(test_X)
np.savetxt("predictions.txt", predictions)

Prepare Submission For Kaggle

The predictions.txt file isn’t in the format that Kaggle needs for submission. Let’s put our predictions into a pandas DataFrame along with the PassengerId column:

# Function to flatten nested lists
flatten = lambda l: [item for sublist in l for item in sublist]
# Temp DataFrame to get PassengerIds
temp_df = pd.read_csv("../data/test.csv")
submission_df = pd.DataFrame()
submission_df["PassengerId"] = temp_df["PassengerId"]
submission_df["Survived"] = pd.Series(flatten(predictions.flatten().tolist()))

Let’s export to CSV for upload to Kaggle:

submission_df.to_csv("kaggle_submission.csv", index=False)

Results

According to Kaggle’s online judge, my model had an accuracy of 74.641% on the test dataset, which is not great by machine learning standards, but was nevertheless exciting for me, since I implemented the model (almost) from scratch.

This was an improvement over my submission from last year, which was a linear regression algorithm using scikit-learn.

Kaggle leaderboard screenshot

Figure 2: Pics or it didn't happen

Conclusion

This project was also a chance to dip my toes into creating Machine Learning models for use on real data, taking the theory and applying it to a real-world problem.

Based on my results, I can also conclude that feature engineering (such as using polynomial features) may be needed to achieve a higher accuracy. Additionally, using something like neural networks may be a better option to achieve higher accuracy, due to their ability to learn complex nonlinear hypotheses.

If you are a machine learning expert and have suggestions on how I could improve the accuracy of my toy model, I would love to hear them! Comment on the Hacker News thread linked below or email me at the address below.

Discuss on Hacker News