1 Introduction

The RMS Titanic was a British luxury passenger liner that sank during its maiden voyage en route to New York City from Southampton, England, killing about 1,500 passengers and ship personnel. It is one of the most famous tragedies in modern history, it inspired numerous stories, several films, and a musical and has been the subject of much scholarship and scientific speculation.

This report will continue the scientific exploration of the Titanic. The estimated 2,224 passengers all have their own stories of escape or death in the wreck. It is believed, however, that studying the underlying information about these passengers can provide insight on why certain people survived.

2 Data Source

The data set used in this report is sourced from Kaggle.com. It was orignially created for a machine learning competition hosted by Kaggle. The link for this data set is https://www.kaggle.com/c/titanic. The data is originally split into seperate train and testing data sets for purposes of the competition. The train data is used in this report for more observations and the complete set of variables.

The data was uploaded to a GitHub repository for perpetual online access. This report will source the data from this repository (URL, https://raw.githubusercontent.com/ncbrechbill/STA321/refs/heads/main/STA321/titanic.csv)

This data contains 891 observations on 12 variables. Any observations of N/A in variables of interest will be removed. The variables are:

  1. PassengerID: A unique identifier
  2. Survived: The survival status of the passenger (0 = No, 1 = Yes)
  3. Pclass: The passenger’s ticket class (1st, 2nd, or 3rd)
  4. Name: The passenger’s name
  5. Sex: The passenger’s sex
  6. Age: The passenger’s age in years
  7. sibSp: Number of siblings/spouse aboard
  8. Parch: Number of parents/children aboard
  9. Ticket: Passenger’s ticket number
  10. Fare: Passenger’s boarding fare in USD
  11. Cabin: The passenger’s cabin
  12. Embarked: Location which passenger embarked (C = Cherbourg, Q = Queenstown, S = Southampton)
url = "https://raw.githubusercontent.com/ncbrechbill/STA321/refs/heads/main/STA321/titanic.csv"
titanic <- read.csv(url)
data <- na.omit(dplyr::select(titanic, Survived, Pclass, Age, Sex, SibSp, Parch, Fare))
data$Pclass <- as.factor(data$Pclass)

3 Research Question

This report investigates the relationship between a passenger’s age, ticket class, and sex with their likelihood of survival on the Titanic. To address this, a multiple logistic regression model is applied, with survival (survived vs. not survived) as the binary outcome variable and passenger age, sex, and ticket class as the predictors. This approach allows us to evaluate whether these factors significantly affected survival odds, estimate the direction and magnitude of the effects, and assess the model’s ability to explain variation in survival outcomes.

4 Exploratory Data Analysis

All observations with missing values in any of the variables were removed. This leaves 714 observations. Their distributions and potential correlations are provided in the figure below.

pairs.panels(data[,-9], 
             method = "pearson", # correlation method
             hist.col = "royalblue",
             density = TRUE,  # show density plots
             ellipses = TRUE # show correlation ellipses
             )

We can see that there are more deaths than survivals, more high class passengers than middle or low, and more men than women. Age is slightly skewed right, but no transformation will be applied. SibSP, Parch, and Fare are highly skewed. PassengerID, Name, Ticket, Cabin, and Embarked were not included for irrelevance.

5 Multiple Logistic Regression

The variables sex, Age, and Pclass will be used for logistic regression. Fare was not selected because it is already categorized by Pclass. Parch and SibSp were not selected because of a few confounding factors. For example a passenger could have at most 2 parents aboard, but any realistically possible amount of children could be aboard with a parent. Similarly, a passenger can have at most one spouse, but any number of siblings aboard. These variables can lead to any number of interpretations and their influence on a model would be obscured by these numerous possibilities.

5.1 Full Model

The first model contains all three factors.

full.model <- glm(Survived ~
                    Pclass+Age+Sex,
                  family = binomial(link = "logit"),
                  data = data)
pander(summary(full.model)$coef,
       caption="Summary of inferential statistics of the full model")
Summary of inferential statistics of the full model
  Estimate Std. Error z value Pr(>|z|)
(Intercept) 3.777 0.4011 9.416 4.682e-21
Pclass2 -1.31 0.2781 -4.71 2.472e-06
Pclass3 -2.581 0.2814 -9.169 4.761e-20
Age -0.03699 0.007656 -4.831 1.359e-06
Sexmale -2.523 0.2074 -12.16 4.811e-34

All values were statistically significant. Before creating a reduced model, we explore potential multicollinearity among the predictor variables. The following table shows the VIF for each individual variable.

pander(vif(full.model))
  GVIF Df GVIF^(1/(2*Df))
Pclass 1.417 2 1.091
Age 1.333 1 1.155
Sex 1.073 1 1.036

All GVIF values are under 1.5. This suggests no multicollinearity issues between these variables.

5.2 Reduced Model

The reduced model was constructed on the variable sex, determined by the correlation coefficients determined on the response variable during EDA. The coefficient remains similar for the sex variable, and in the same direction (negative towards male).

reduced.model <- glm(Survived~
                       Sex,
                     family = binomial(link = "logit"),
                     data = data)
pander(summary(reduced.model)$coef)
  Estimate Std. Error z value Pr(>|z|)
(Intercept) 1.124 0.1439 7.814 5.524e-15
Sexmale -2.478 0.185 -13.39 6.701e-41

5.3 Automatic Model Selection

The reduced model may use insufficient information, resulting in underfitting (missing important effects), while the full model can lead to overfitting (modeling noise and failing to generalize). To address this, we use automatic variable selection to find a simpler, more interpretable, and better-performing model by intelligently choosing a subset of predictors that includes the important variables from the reduced model while avoiding the noise of the full model.

final.model.forward = stepAIC(reduced.model, 
                      scope = list(lower=formula(reduced.model),upper=formula(full.model)),
                      direction = "forward",   # forward selection
                      trace = 0   # do not show the details
                      )
pander(summary(final.model.forward)$coef, 
      caption="Summary of inferential statistics of the final model")
Summary of inferential statistics of the final model
  Estimate Std. Error z value Pr(>|z|)
(Intercept) 3.777 0.4011 9.416 4.682e-21
Sexmale -2.523 0.2074 -12.16 4.811e-34
Pclass2 -1.31 0.2781 -4.71 2.472e-06
Pclass3 -2.581 0.2814 -9.169 4.761e-20
Age -0.03699 0.007656 -4.831 1.359e-06

5.4 Performance Comparison

The full model was automatically selected. Thus, their deviance is the same. With the full model utilizing only three predictors, the model remains relatively simple and parsimonious.

## Other global goodness-of-fit
global.measure=function(s.logit){
dev.resid = s.logit$deviance
dev.0.resid = s.logit$null.deviance
aic = s.logit$aic
goodness = cbind(Deviance.residual =dev.resid, Null.Deviance.Residual = dev.0.resid,
      AIC = aic)
goodness
}
goodness=rbind(full.model = global.measure(full.model),
      reduced.model=global.measure(reduced.model),
      final.model=global.measure(final.model.forward))
row.names(goodness) = c("full.model", "reduced.model", "auto.selected.model")
pander(goodness, caption ="Comparison of global goodness-of-fit statistics")
Comparison of global goodness-of-fit statistics
  Deviance.residual Null.Deviance.Residual AIC
full.model 647.3 964.5 657.3
reduced.model 750.7 964.5 754.7
auto.selected.model 647.3 964.5 657.3

6 Final Model

The final model is translated into an odds ratio for interpretation. Starting at a baseline 43.69 Odds ratio of survival, the ratio was reduced by a passenger being male and having a 2nd or 3rd class ticket. Every year of age also reduced survival odds.

# Odds ratio
model.coef.stats = summary(final.model.forward)$coef
odds.ratio = exp(coef(final.model.forward))
out.stats = cbind(model.coef.stats, odds.ratio = odds.ratio)                 
pander(out.stats,caption = "Summary Stats with Odds Ratios")
Summary Stats with Odds Ratios
  Estimate Std. Error z value Pr(>|z|) odds.ratio
(Intercept) 3.777 0.4011 9.416 4.682e-21 43.69
Sexmale -2.523 0.2074 -12.16 4.811e-34 0.08024
Pclass2 -1.31 0.2781 -4.71 2.472e-06 0.2699
Pclass3 -2.581 0.2814 -9.169 4.761e-20 0.07573
Age -0.03699 0.007656 -4.831 1.359e-06 0.9637

7 Analysis

The final model allows us to analyze the demographics that associate with survival the most. Of the passengers analyzed, the younger 1st class women had the best odds of survival. Older 3rd class men had the worst odds of survival. This provides insights into what the passengers found most important into factoring who’s survival was prioritized.

A popular excerpt from the Titanic’s wreck was “women and children first”. At a glance, the data supports this, as both women and younger passenger’s had better odds of survival. Additionally, we can see that higher class passengers had higher odds of survival. One can surmise that this narrative would be less popular than the relatively heroic “women and children first” rhetoric. In several instances, wealth helps people survive, and it seems the wreck of the titanic was no exception.

This model will be tweaked for prediction in a later report.

---
title: "Multiple Logistic Analysis of RMS Titanic Survivors"
author: "Noah Brechbill"
date: "`r Sys.Date()`"
output:
  html_document:
    toc: yes
    toc_float: yes
    toc_depth: 4
    fig_width: 6
    fig_height: 6
    fig_caption: yes
    number_sections: yes
    toc_collapsed: yes
    code_folding: hide
    code_download: yes
    smooth_scroll: yes
    theme: lumen
---

```{css, echo = FALSE}
/* Cascading Style Sheets (CSS) is a stylesheet language used to describe the presentation of a document written in HTML or XML. it is a simple mechanism for adding style (e.g., fonts, colors, spacing) to Web documents. */

h1.title {  /* Title - font specifications of the report title */
  font-size: 24px;
  color: DarkRed;
  text-align: center;
  font-weight: bold;
  font-family: "Gill Sans", sans-serif;
}
h4.author { /* Header 4 - font specifications for authors  */
  font-size: 20px;
  font-family: system-ui;
  font-weight: bold;
  color: DarkRed;
  text-align: center;
}
h4.date { /* Header 4 - font specifications for the date  */
  font-size: 18px;
  font-family: system-ui;
  font-weight: bold;
  color: DarkBlue;
  text-align: center;
}
h1 { /* Header 1 - font specifications for level 1 section title  */
    font-size: 22px;
    font-family: system-ui;
  font-weight: bold;
    color: navy;
    text-align: left;
}
h2 { /* Header 2 - font specifications for level 2 section title */
    font-size: 20px;
  font-weight: bold;
    font-family: "Times New Roman", Times, serif;
    color: navy;
    text-align: left;
}

h3 { /* Header 3 - font specifications of level 3 section title  */
    font-size: 18px;
  font-weight: bold;
    font-family: "Times New Roman", Times, serif;
    color: navy;
    text-align: left;
}

h4 { /* Header 4 - font specifications of level 4 section title  */
    font-size: 18px;
  font-weight: bold;
    font-family: "Times New Roman", Times, serif;
    color: darkred;
    text-align: left;
}

body { background-color:white; }

.highlightme { background-color:yellow; }

p { background-color:white; }

```

------------------------------------------------------------------------

```{r setup, include=FALSE}
load_packages <- function(pkg_list) {
  for (pkg in pkg_list) {
    if (!require(pkg, character.only = TRUE)) {
      install.packages(pkg, dependencies = TRUE)
      library(pkg, character.only = TRUE)
    }
  }
}

packages <- c("tidyverse", "GGally", "pander", "car", "MASS", "psych")
load_packages(packages)

knitr::opts_chunk$set(echo = TRUE,      
                      warning = FALSE,   
                      message = FALSE,  
                      results = TRUE,
                      comment = NA,
                      fig.align = "center"
                      )   
```

# Introduction

The RMS Titanic was a British luxury passenger liner that sank during its maiden voyage en route to New York City from Southampton, England, killing about 1,500 passengers and ship personnel. It is one of the most famous tragedies in modern history, it inspired numerous stories, several films, and a musical and has been the subject of much scholarship and scientific speculation.

This report will continue the scientific exploration of the Titanic. The estimated 2,224 passengers all have their own stories of escape or death in the wreck. It is believed, however, that studying the underlying information about these passengers can provide insight on why certain people survived.

# Data Source

The data set used in this report is sourced from Kaggle.com. It was orignially created for a machine learning competition hosted by Kaggle. The link for this data set is <https://www.kaggle.com/c/titanic>. The data is originally split into seperate train and testing data sets for purposes of the competition. The train data is used in this report for more observations and the complete set of variables.

The data was uploaded to a GitHub repository for perpetual online access. This report will source the data from this repository (URL, <https://raw.githubusercontent.com/ncbrechbill/STA321/refs/heads/main/STA321/titanic.csv>)

This data contains 891 observations on 12 variables. Any observations of N/A in variables of interest will be removed. The variables are:

1.  **PassengerID**: A unique identifier
2.  **Survived**: The survival status of the passenger (0 = No, 1 = Yes)
3.  **Pclass**: The passenger's ticket class (1st, 2nd, or 3rd)
4.  **Name**: The passenger's name
5.  **Sex**: The passenger's sex
6.  **Age**: The passenger's age in years
7.  **sibSp**: Number of siblings/spouse aboard
8.  **Parch**: Number of parents/children aboard
9.  **Ticket**: Passenger's ticket number
10. **Fare**: Passenger's boarding fare in USD
11. **Cabin**: The passenger's cabin
12. **Embarked**: Location which passenger embarked (C = Cherbourg, Q = Queenstown, S = Southampton)

```{r data loading}
url = "https://raw.githubusercontent.com/ncbrechbill/STA321/refs/heads/main/STA321/titanic.csv"
titanic <- read.csv(url)
data <- na.omit(dplyr::select(titanic, Survived, Pclass, Age, Sex, SibSp, Parch, Fare))
data$Pclass <- as.factor(data$Pclass)
```

# Research Question

This report investigates the relationship between a passenger’s age, ticket class, and sex with their likelihood of survival on the Titanic. To address this, a multiple logistic regression model is applied, with survival (survived vs. not survived) as the binary outcome variable and passenger age, sex, and ticket class as the predictors. This approach allows us to evaluate whether these factors significantly affected survival odds, estimate the direction and magnitude of the effects, and assess the model’s ability to explain variation in survival outcomes.

# Exploratory Data Analysis

All observations with missing values in any of the variables were removed. This leaves 714 observations. Their distributions and potential correlations are provided in the figure below.

```{r}
pairs.panels(data[,-9], 
             method = "pearson", # correlation method
             hist.col = "royalblue",
             density = TRUE,  # show density plots
             ellipses = TRUE # show correlation ellipses
             )
```

We can see that there are more deaths than survivals, more high class passengers than middle or low, and more men than women. Age is slightly skewed right, but no transformation will be applied. SibSP, Parch, and Fare are highly skewed. PassengerID, Name, Ticket, Cabin, and Embarked were not included for irrelevance.

# Multiple Logistic Regression

The variables sex, Age, and Pclass will be used for logistic regression. Fare was not selected because it is already categorized by Pclass. Parch and SibSp were not selected because of a few confounding factors. For example a passenger could have at most 2 parents aboard, but any realistically possible amount of children could be aboard with a parent. Similarly, a passenger can have at most one spouse, but any number of siblings aboard. These variables can lead to any number of interpretations and their influence on a model would be obscured by these numerous possibilities.

## Full Model

The first model contains all three factors.

```{r}
full.model <- glm(Survived ~
                    Pclass+Age+Sex,
                  family = binomial(link = "logit"),
                  data = data)
pander(summary(full.model)$coef,
       caption="Summary of inferential statistics of the full model")
```

All values were statistically significant. Before creating a reduced model, we explore potential multicollinearity among the predictor variables. The following table shows the VIF for each individual variable.

```{r}
pander(vif(full.model))
```

All GVIF values are under 1.5. This suggests no multicollinearity issues between these variables.

## Reduced Model

The reduced model was constructed on the variable sex, determined by the correlation coefficients determined on the response variable during EDA. The coefficient remains similar for the sex variable, and in the same direction (negative towards male).

```{r}
reduced.model <- glm(Survived~
                       Sex,
                     family = binomial(link = "logit"),
                     data = data)
pander(summary(reduced.model)$coef)
```

## Automatic Model Selection

The reduced model may use insufficient information, resulting in underfitting (missing important effects), while the full model can lead to overfitting (modeling noise and failing to generalize). To address this, we use automatic variable selection to find a simpler, more interpretable, and better-performing model by intelligently choosing a subset of predictors that includes the important variables from the reduced model while avoiding the noise of the full model.

```{r}
final.model.forward = stepAIC(reduced.model, 
                      scope = list(lower=formula(reduced.model),upper=formula(full.model)),
                      direction = "forward",   # forward selection
                      trace = 0   # do not show the details
                      )
pander(summary(final.model.forward)$coef, 
      caption="Summary of inferential statistics of the final model")
```

## Performance Comparison

The full model was automatically selected. Thus, their deviance is the same. With the full model utilizing only three predictors, the model remains relatively simple and parsimonious.

```{r}
## Other global goodness-of-fit
global.measure=function(s.logit){
dev.resid = s.logit$deviance
dev.0.resid = s.logit$null.deviance
aic = s.logit$aic
goodness = cbind(Deviance.residual =dev.resid, Null.Deviance.Residual = dev.0.resid,
      AIC = aic)
goodness
}
goodness=rbind(full.model = global.measure(full.model),
      reduced.model=global.measure(reduced.model),
      final.model=global.measure(final.model.forward))
row.names(goodness) = c("full.model", "reduced.model", "auto.selected.model")
pander(goodness, caption ="Comparison of global goodness-of-fit statistics")
```

# Final Model

The final model is translated into an odds ratio for interpretation. Starting at a baseline 43.69 Odds ratio of survival, the ratio was reduced by a passenger being male and having a 2nd or 3rd class ticket. Every year of age also reduced survival odds.

```{r}
# Odds ratio
model.coef.stats = summary(final.model.forward)$coef
odds.ratio = exp(coef(final.model.forward))
out.stats = cbind(model.coef.stats, odds.ratio = odds.ratio)                 
pander(out.stats,caption = "Summary Stats with Odds Ratios")
```

# Analysis

The final model allows us to analyze the demographics that associate with survival the most. Of the passengers analyzed, the younger 1st class women had the best odds of survival. Older 3rd class men had the worst odds of survival. This provides insights into what the passengers found most important into factoring who's survival was prioritized.

A popular excerpt from the Titanic's wreck was "women and children first". At a glance, the data supports this, as both women and younger passenger's had better odds of survival. Additionally, we can see that higher class passengers had higher odds of survival. One can surmise that this narrative would be less popular than the relatively heroic "women and children first" rhetoric. In several instances, wealth helps people survive, and it seems the wreck of the titanic was no exception.

This model will be tweaked for prediction in a later report.
