/HeartAttackPrediction

In the final project we examine heart attack prediction factors.

Primary LanguageJupyter NotebookApache License 2.0Apache-2.0

Heart Attack Prediction with Machine Learning Models

We have examined heart attack prediction factors including, but not limited to:

  • If an individual was diagnosed with asthma
  • If an individual was ever overweight
  • If an individual ever had a stroke
  • If an individual was ever a smoker

Data Collecting (ETL)

We examined National Health and Nutrition Examination Survey (NHANES) data from the Centers for Disease Control and Prevention (CDC). The main data set that we used was Pre-Pandemic data from 2017 to March 2020. While there are a large number of surveys to use, we focused our attention primarily on two surveys: Medical Conditions and Demographics. A big challenge with this data and our project overall was the data cleaning process. The medical conditions table contained over 60 unique columns. Since there were so many columns, we first decided which columns were of most importance to our question. Once that was determined, we needed to clean all of the column names, since they were initially unique codes that provided no insight as to what the column was.

After the data table was in a readable format, our next focus turned to cleaning up the data since it had a lot of missing information. Binary columns were changed to 1 (had the diseases) and 0 (did not report having the disease). Cells in columns representing the age at which someone had a specified condition were converted to 0 if they either did not have the disease or if it fell after the age of the heart attack. A years column for each condition was added to indicate the difference between the condition age and the heart attack age. A max age column was also added to indicate the age of the condition that was closest to but still prior to the heart attack. This cleaned table was output to a csv file and used for further analysis. This process was also repeated for the 2015-2016 data set, which was used for further testing with some of the models.

An additional table was created that contained binary columns to indicate whether or not the patient had the condition prior to the heart attack. This table was also used in the later analyses.

Primary Component Analysis (PCA)

  • Since we are looking at many different factors to predict an outcome, it will be helpful to reduce the dimensions to provide simplicity. This will reduce accuracy but will make it easier to see the approximation of every variable.
  • There is a lot of variance within each of our variables, we want to simplify and create bigger, more meaningful features.
  • We hope to reduce our data to only two dimensions so we set n_components in our pca function to 2.
  • Using the explained_variance_ratio_ function we find the first primary component explains 33% of the variance, the second primary component explains 31%. Together they only explain 64% of the total variance. These values are relatively low because a large amount of the data is binary.
  • We are also able to determine which variables are the most important for each component using the components_ function. image
    • We found for the first primary component, total cholesterol is the most important variable.
    • For the second primary component, the number of drinks per day is the most important variable.
    • The importance of each variable is defined by how much each feature contributes to the primary component. image
  • With K-means clustering, we can identify that 3 is the optimal number of clustering.
  • We can then use these clusters to categorize individuals based on certain traits. Each group means those in the same group have similar characteristics. bokeh_plot
  • Gives an overall more simplistic view of our data

Logistic Regression

When focusing on logistic regression on dataframe, we found that it is nearly impossible to predict the accuracy using logistic regression. First of all, the dataset was not complete, there is a huge missing factor on heart_attack_age (y values), which the value '0' was used to subsitute. Logistic Regression is good with relationship if the features, and target are not complex.

For the model, the outcome came out with the accuracy of 96% which is relatively high for predicting the whether or not if a person has a chance of having a heart attack with existing conditions. But the macro average for the set was 0.03, which was lower than expected, meaning that the model cannot be used for all (among population).

Screen Shot 2023-06-12 at 6 18 59 PM

Random Forests

We also used a Random Forest Classifier on the binary data, which is 1 if they had the condition before the heart attack and 0 if they did not. We first ran our model on the baseline hyper-parameters to see where the model would be at. Below, we show the confusion matrix, accuracy score, and classification report.

Screen Shot 2023-06-11 at 4 37 24 PM

While we do have a high accuracy score, you can see from the confusion matrix that this model is really bad at predicting people with heart attacks, with it missing on 107 people. This can be explained by the fact that the data has a big class imbalance, as people with heart attacks only make up about 5% of our data. With this class imbalance, our model was not learning from the features and treating heart attacks as a rare occurrence. To solve this problem, we decided to down sample our non-heart attack class, so that the heart attack people make up more of the data. We also tuned our hyper-parameters with RandomizedSearchCV() and GridSearchCV() to get the best model for the data.

First, we downsized the data so that the non-heart attack group and heart attack group had a ratio of 2:1. We, then, split the data into training and testing sets and applied RandomizedSearchCV() with a range of different hyper-parameters. After that, we predicted the testing set, and its results are shown below.

Screen Shot 2023-06-11 at 4 56 07 PM

With the parameters from the randomized search, we did a GridSearchCV() with values close to the hyper-parameters of the randomized search to see if we can get a better model. The results from that model is shown below.

Screen Shot 2023-06-11 at 5 05 21 PM

We applied the same process to a downsized data set where the ratio was 3:1 non-heart attack to heart attack, but we found that the smaller down sampled model was better at predicting heart attacks and continue with that model.

We wanted to check if there was any biases from the model, so we decided to use it to predict the whole data set. The goal is to check if we get the same results from the testing, then we can say that there is no biases in the smaller data set that would skew the model into one direction. The results from predicting the whole data set is shown below.

Screen Shot 2023-06-11 at 5 06 52 PM

The total results shown are some what similar to the testing results, and we see that the the precision for heart attack is lower than the down sample. Thus, there might be a little bias within the model. From here, we also wanted to test this model with data from a different year, so we applied the model to data from the years 2015-2016. The results from applying that model is shown below.

Screen Shot 2023-06-11 at 5 07 57 PM

To check for the conditions most used to predict heart attack, we got the feature importances from our model and made a bar graph. From the graph, we can see that heart disease, heart failure, stroke, and angina pectoris are the main conditions used to predict heart attacks.

Screen Shot 2023-06-11 at 5 05 37 PM

PCA + Random Forests

Screenshot 2023-06-12 at 6 36 56 PM

Neural Networks

For the Neural Network we used the Medical Conditions data, a table containing dozens of conditions ranging from heart disease, stroke, and cancer to asthma, arthrithis, and hay fever. This also proved to be a challenge. As mentioned before, the NaN values were changed to 0 for simplicity purposes. This eased the process of both cleaning and analysis. since most of our values were NaN, 0 was effectively the mode of the data and better number to use than the mean or median.

The data was first further cleaned, with removal columns that have no value in the analysis, such as the ID number. The get_dummies function was used to turn existing non-binary columns into binary. Then the X and y values were chosen, y being the target, in this case Heart Attack, and X being a set of variables to test for in the model, or features, to try to predict whether or not heart attacks could be predicted.

The data was then scaled.

After scaling we were ready to set up the model. The amount of layers, neurons in each layer, and activation functions were chosen.

We then compiled the data.

Finally, the models were run by choosing the amount of epochs and training the model.

Upon starting the analysis, excessive accuracy was recorded, with values of up to 100% on the second or third epoch. There seemed to be either data leaking or overfitting, or perhaps there was an error in the model.

We first tried to analyze the actual inputs by creating a csv and checking the data. We cut additional inputs from the features, to ensure that we were not leaking the data. We inserted a function for early stopping. Still, high levels of accuracy remained.

Here we have the loss on each epoch when running the first model 75 times:

loss

Here we have the accuracy on each epoch when running this same model:

accuracy

The problem was an imbalanced data set. We ended up splitting the data into x and y training and testing sets, splitting heart attacks over the max age (age when first health event was diagnosed) and heart attacks where they were the first health event. We removed more categories. We scaled the data and finally, put all of the sets back together before compilation. After the training it was time to analyze the model's results. A rough accuracy score of 99% was recorded using the model. Precision, which is the fraction of relevant instances among the retrieved instances, is .84. Recall, which is is the fraction of relevant instances that were retrieved, was 1.0. This doesn not imply that for a model with a much larger set could predict heart attacks 100% of the time, but does imply that this model may be worth pursuing in the future, and we look forward to possibly doing that.

Here we have both graphs after then optimization:

loss2

accuracy2

In conclusion, the neural network seems to show some promise in predicting heart attacks. More analysis would be required, along with tuning for hyperparameters and adding additional data, to effectively replicate this model on bigger datasets.