Atherosclerosis - Disease lethality predictionĀ¶
With this notebook we give a brief introduction to feature engineering on relational data with many columns. We discuss why feature engineering on such data is particularly challenging and what we can do to overcome these problems.
Summary:
- Prediction type: Binary classification
- Domain: Health
- Prediction target: Mortality within one year
- Source data: 146 columns in 2 tables, 22 MB
- Population size: 28433
Feature engineering and the curse of dimensionalityĀ¶
The problemĀ¶
To illustrate the point, we give a simplified example based on the real data used in the analysis below. When we engineer features from relational data, we usually write something like this:
SELECT AVG(t2.HDL)
FROM population_training t1
LEFT JOIN contr t2
ON t1.ICO = t2.ICO
WHERE t1.AGE >= 60 AND t1.ALOKOHOL IN ('1', '2')
GROUP BY t1.ICO;
Think about that for a second. This feature aggregates high-density lipoprotein (HDL) cholesterol values recorded during control dates conditional on age and alcohol consumption. We arbitrarily chose both, the column to aggregate over (HDL) and the set of columns to construct conditions on (AGE and ALKOHOL) out of a greater set of 146 columns.
Every column that we have can either be aggregated (here HDL) or it can be used for our conditions (here AGE and ALKOHOL). That means if we have n columns to aggregate, we can potentially build conditions for $n$ other columns. In other words, the computational complexity is $n^2$ in the number of columns.
Note that this problem occurs regardless of whether you automate feature engineering or you do it by hand. The size of the search space is $n^2$ in the number of columns in either case, unless you can rule something out a-priori.
The solutionĀ¶
So when we have relational data sets with many columns, what do we do? The answer is to write different features. Specifically, suppose we had features like this:
SELECT AVG(
CASE WHEN t1.AGE >= THEN weight1
CASE WHEN t1.ALKOHOL IN ('1', '2') THEN weight2
END
)
FROM population_training t1
LEFT JOIN contr t2
ON t1.ICO = t2.ICO
GROUP BY t1.ICO;
weight1 and weight2 are learnable weights. An algorithm that generates features like this can only use columns for conditions, it is not allowed to aggregate columns ā and it doesn't need to do so.
That means the computational complexity is linear instead of quadratic. For data sets with a large number of columns this can make all the difference in the world. For instance, if you have 100 columns the size of the search space of the second approach is only 1% of the size of the search space of the first one.
BackgroundĀ¶
To illustrate the problem of dimensionality in predictive analytics on relational data, we use the STULONG 1 dataset. It is a longitudinal study of atherosclerosis patients.
One of its defining features is that it contains many columns, which makes it a good candidate to illustrate the problem discussed in this notebook.
The are some academic studies related to this dataset:
- https://www.researchgate.net/publication/228572841_Mining_episode_rules_in_STULONG_dataset
- https://citeseerx.ist.psu.edu/doc_view/pid/3a9cb05b77b631b6fcbe253eb93e053ba8c0719c
The way these studies handle the large number of columns in the data set is to divide the columns into subgroups and then handling each subgroup separately. Even though this is one way to overcome the curse of dimensionality, it is not a very satisfying approach. We would like to be able to handle a large number of columns at once.
The analysis is based on the STULONG 1 dataset. It is publicly available and can be downloaded the the CTU Prague Relational Learning Repository (Now residing at relational-data.org.).
AnalysisĀ¶
Let's get started with the analysis and set up your session:
import os
from pathlib import Path
import numpy as np
import pandas as pd
from IPython.display import Image, Markdown
import matplotlib.pyplot as plt
%matplotlib inline
import getml
getml.engine.launch(home_directory=Path.home(), allow_remote_ips=True, token="token")
getml.engine.set_project("atherosclerosis")
getML engine is already running. Connected to project 'atherosclerosis'
1. Loading dataĀ¶
1.1 Download from sourceĀ¶
Downloading the raw data and convert it into a prediction ready format takes time. To get to the getML model building as fast as possible, we prepared the data for you and excluded the code from this notebook. It is made available in the example notebook featuring the full analysis.
population, contr = getml.datasets.load_atherosclerosis()
Loading population... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] Loading contr... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00]
1.2 Prepare data for getMLĀ¶
The getml.datasets.load_atherosclerosis
method took care of the entire data lifting:
- Downloads csv's from our servers in python
- Converts csv's to getML DataFrames
- Sets roles to columns inside getML DataFrames
Data visualization
The original data (image below) model is condensed into 2 tables:
- A population table population_{train/test/validate}, based on
death
table - A peripheral table:
contr
.
Death: population table
- Reference Date: Period of time from 1976 to 1999
- Target: If the patient dies within one year after each reference date
population
name | REFERENCE_DATE | ENTRY_DATE | ICO | TARGET | KONSKUP | STAV | VZDELANI | ZODPOV | TELAKTZA | AKTPOZAM | DOPRAVA | DOPRATRV | ALKOHOL | BOLHR | BOLDK | DUSNOST | RARISK | OBEZRISK | KOURRISK | HTRISK | CHOLRISK | MOC | AGE | PARTICIPATION | VYSKA | VAHA | SYST1 | DIAST1 | SYST2 | DIAST2 | TRIC | SUBSC | CHLST | TRIGL | KOURENI | DOBAKOUR | BYVKURAK | PIVOMN | VINOMN | LIHMN | KAVA | CAJ | CUKR | ROKNAR | ROKVSTUP | MESVSTUP | IM | HT | ICT | DIABET | HYPLIP | DUMMY | YEAR | PIVO7 | PIVO10 | PIVO12 | VINO | LIHOV | IML | HTD | HTL | ICTL | DIABD | DIABL | HYPLD | HYPLL | IMTRV | HTTRV | ICTTRV | DIABTRV | HYPLTRV | DENUMR | MESUMR | ROKUMR | PRICUMR | DEATH_DATE |
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
role | time_stamp | time_stamp | join_key | target | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | numerical | unused_float | unused_float | unused_float | unused_float | unused_float | unused_float | unused_float | unused_float | unused_float | unused_float | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string |
unit | time stamp | time stamp | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
0 | 1977-01-01 | 1976-10-01 | 10001 | 0Ā | 4 | 1 | 3 | 1 | 1 | 2 | 3 | 6 | 2 | 1 | 1 | 1 | 0 | 0 | 1 | 0 | 0 | 1 | 48Ā | -1899Ā | 169Ā | 71Ā | 120Ā | 85Ā | 120Ā | 90Ā | 4Ā | 12Ā | 209Ā | 86Ā | 4Ā | 10Ā | nanĀ | 1Ā | 4Ā | 9Ā | 2Ā | 4Ā | 3Ā | 29Ā | 1976Ā | 10Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1977Ā | NULL | NULL | NULL | NULL | 12.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
1 | 1978-01-01 | 1976-10-01 | 10001 | 0Ā | 4 | 1 | 3 | 1 | 1 | 2 | 3 | 6 | 2 | 1 | 1 | 1 | 0 | 0 | 1 | 0 | 0 | 1 | 49Ā | -1898Ā | 169Ā | 71Ā | 120Ā | 85Ā | 120Ā | 90Ā | 4Ā | 12Ā | 209Ā | 86Ā | 4Ā | 10Ā | nanĀ | 1Ā | 4Ā | 9Ā | 2Ā | 4Ā | 3Ā | 29Ā | 1976Ā | 10Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1978Ā | NULL | NULL | NULL | NULL | 12.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
2 | 1979-01-01 | 1976-10-01 | 10001 | 0Ā | 4 | 1 | 3 | 1 | 1 | 2 | 3 | 6 | 2 | 1 | 1 | 1 | 0 | 0 | 1 | 0 | 0 | 1 | 50Ā | -1897Ā | 169Ā | 71Ā | 120Ā | 85Ā | 120Ā | 90Ā | 4Ā | 12Ā | 209Ā | 86Ā | 4Ā | 10Ā | nanĀ | 1Ā | 4Ā | 9Ā | 2Ā | 4Ā | 3Ā | 29Ā | 1976Ā | 10Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1979Ā | NULL | NULL | NULL | NULL | 12.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
3 | 1980-01-01 | 1976-10-01 | 10001 | 0Ā | 4 | 1 | 3 | 1 | 1 | 2 | 3 | 6 | 2 | 1 | 1 | 1 | 0 | 0 | 1 | 0 | 0 | 1 | 51Ā | -1896Ā | 169Ā | 71Ā | 120Ā | 85Ā | 120Ā | 90Ā | 4Ā | 12Ā | 209Ā | 86Ā | 4Ā | 10Ā | nanĀ | 1Ā | 4Ā | 9Ā | 2Ā | 4Ā | 3Ā | 29Ā | 1976Ā | 10Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1980Ā | NULL | NULL | NULL | NULL | 12.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
4 | 1981-01-01 | 1976-10-01 | 10001 | 0Ā | 4 | 1 | 3 | 1 | 1 | 2 | 3 | 6 | 2 | 1 | 1 | 1 | 0 | 0 | 1 | 0 | 0 | 1 | 52Ā | -1895Ā | 169Ā | 71Ā | 120Ā | 85Ā | 120Ā | 90Ā | 4Ā | 12Ā | 209Ā | 86Ā | 4Ā | 10Ā | nanĀ | 1Ā | 4Ā | 9Ā | 2Ā | 4Ā | 3Ā | 29Ā | 1976Ā | 10Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1981Ā | NULL | NULL | NULL | NULL | 12.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
... | ... | ... | ...Ā | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | |
28428 | 1995-01-01 | 1977-03-01 | 30065 | 0Ā | 5 | 2 | 2 | 3 | 1 | 2 | 1 | 5 | 2 | 3 | 2 | 2 | 0 | 0 | 1 | 0 | 0 | 1 | 67Ā | -1882Ā | 179Ā | 69Ā | 110Ā | 80Ā | 120Ā | 80Ā | 10Ā | 18Ā | 235Ā | 733Ā | 4Ā | 10Ā | nanĀ | 3Ā | 4Ā | 7Ā | 1Ā | 5Ā | 2Ā | 28Ā | 1977Ā | 3Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1995Ā | NULL | 9.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
28429 | 1996-01-01 | 1977-03-01 | 30065 | 0Ā | 5 | 2 | 2 | 3 | 1 | 2 | 1 | 5 | 2 | 3 | 2 | 2 | 0 | 0 | 1 | 0 | 0 | 1 | 68Ā | -1881Ā | 179Ā | 69Ā | 110Ā | 80Ā | 120Ā | 80Ā | 10Ā | 18Ā | 235Ā | 733Ā | 4Ā | 10Ā | nanĀ | 3Ā | 4Ā | 7Ā | 1Ā | 5Ā | 2Ā | 28Ā | 1977Ā | 3Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1996Ā | NULL | 9.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
28430 | 1997-01-01 | 1977-03-01 | 30065 | 0Ā | 5 | 2 | 2 | 3 | 1 | 2 | 1 | 5 | 2 | 3 | 2 | 2 | 0 | 0 | 1 | 0 | 0 | 1 | 69Ā | -1880Ā | 179Ā | 69Ā | 110Ā | 80Ā | 120Ā | 80Ā | 10Ā | 18Ā | 235Ā | 733Ā | 4Ā | 10Ā | nanĀ | 3Ā | 4Ā | 7Ā | 1Ā | 5Ā | 2Ā | 28Ā | 1977Ā | 3Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1997Ā | NULL | 9.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
28431 | 1998-01-01 | 1977-03-01 | 30065 | 0Ā | 5 | 2 | 2 | 3 | 1 | 2 | 1 | 5 | 2 | 3 | 2 | 2 | 0 | 0 | 1 | 0 | 0 | 1 | 70Ā | -1879Ā | 179Ā | 69Ā | 110Ā | 80Ā | 120Ā | 80Ā | 10Ā | 18Ā | 235Ā | 733Ā | 4Ā | 10Ā | nanĀ | 3Ā | 4Ā | 7Ā | 1Ā | 5Ā | 2Ā | 28Ā | 1977Ā | 3Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1998Ā | NULL | 9.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
28432 | 1999-01-01 | 1977-03-01 | 30065 | 0Ā | 5 | 2 | 2 | 3 | 1 | 2 | 1 | 5 | 2 | 3 | 2 | 2 | 0 | 0 | 1 | 0 | 0 | 1 | 71Ā | -1878Ā | 179Ā | 69Ā | 110Ā | 80Ā | 120Ā | 80Ā | 10Ā | 18Ā | 235Ā | 733Ā | 4Ā | 10Ā | nanĀ | 3Ā | 4Ā | 7Ā | 1Ā | 5Ā | 2Ā | 28Ā | 1977Ā | 3Ā | 2Ā | 2Ā | 2Ā | 2Ā | 2Ā | 1Ā | 1999Ā | NULL | 9.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
28433 rows x 76 columns
memory usage: 16.09 MB
name: population
type: getml.DataFrame
Contr: peripheral table
contr
name | CONTROL_DATE | ICO | ZMTELAKT | AKTPOZAM | ZMDIET | LEKTLAK | ZMKOUR | POCCIG | PRACNES | JINAONE | BOLHR | BOLDK | DUSN | HODNSK | HYPERD | HYPCHL | HYPTGL | HMOT | CHLST | HDL | HDLMG | LDL | ROKVYS | MESVYS | PORADK | ZMCHARZA | MOC | LEKCHOL | SRDCE | HYPERT | CEVMOZ | DIAB | HODN0 | ROK0 | HODN1 | ROK1 | HODN2 | ROK2 | HODN3 | ROK3 | HODN4 | ROK4 | HODN11 | ROK11 | HODN12 | ROK12 | HODN13 | ROK13 | HODN14 | ROK14 | HODN15 | ROK15 | HODN21 | ROK21 | HODN23 | ROK23 | SYST | DIAST | TRIC | SUBSC | HYPERSD | HYPERS | CHLSTMG | TRIGL | TRIGLMG | GLYKEMIE | KYSMOC |
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
role | time_stamp | join_key | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | categorical | numerical | numerical | numerical | numerical | numerical | unused_float | unused_float | unused_float | unused_float | unused_float | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string | unused_string |
unit | time stamp | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
0 | 1977-09-01 | 10001 | 3 | 2 | 1 | 2 | 2 | 0.0 | 1 | 2.0 | 1 | 1 | 1 | 2 | 2.0 | 2.0 | 2.0 | 71Ā | 5.61 | nanĀ | nanĀ | nanĀ | 1977Ā | 9Ā | 1Ā | 20Ā | 1Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 130.0 | 90.0 | 4.0 | 12.0 | 2.0 | 2.0 | 217.0 | 1.22 | 108.0 | NULL | NULL |
1 | 1979-01-01 | 10001 | 1 | 1 | 1 | 2 | 1 | 0.0 | 2 | 1 | 2 | 1 | 2 | 2.0 | 2.0 | 2.0 | 72Ā | 6Ā | nanĀ | nanĀ | nanĀ | 1979Ā | 1Ā | 2Ā | 20Ā | 3Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 140.0 | 90.0 | 4.0 | 11.0 | 2.0 | 2.0 | 232.0 | 4.4 | 389.0 | NULL | NULL | |
2 | 1980-04-01 | 10001 | 2 | 1 | 1 | 2 | 1 | 0.0 | 2 | 2 | 1 | 1 | 2 | 2.0 | 2.0 | 2.0 | 71Ā | 6.23 | nanĀ | nanĀ | nanĀ | 1980Ā | 4Ā | 3Ā | 20Ā | 1Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 130.0 | 90.0 | 5.0 | 22.0 | 2.0 | 2.0 | 241.0 | 1.51 | 134.0 | NULL | NULL | |
3 | 1982-01-01 | 10001 | 2 | 1 | 1 | 2 | 1 | 0.0 | 1 | 2.0 | 1 | 1 | 1 | 2 | 2.0 | 2.0 | 2.0 | 74Ā | 5.2 | nanĀ | nanĀ | nanĀ | 1982Ā | 1Ā | 4Ā | 20Ā | 1Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 150.0 | 100.0 | 9.0 | 18.0 | 2.0 | 2.0 | 201.0 | 1.42 | 126.0 | NULL | NULL |
4 | 1983-02-01 | 10001 | 2 | 2 | 1 | 2 | 1 | 0.0 | 2 | 1 | 2 | 1 | 2 | 2.0 | 2.0 | 2.0 | 73Ā | 6.08 | 1.47 | 57Ā | 4.15 | 1983Ā | 2Ā | 5Ā | 20Ā | 1Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 165.0 | 105.0 | 7.0 | 15.0 | 2.0 | 2.0 | 235.0 | 0.99 | 88.0 | NULL | NULL | |
... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ...Ā | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | ... | |
10567 | 1987-10-01 | 20358 | 2 | 2 | 1 | 2 | 1 | 0.0 | 2 | 4 | 1 | 1 | 3 | 2.0 | 2.0 | 2.0 | 85Ā | 4.89 | 0.98 | 38Ā | 3.11 | 1987Ā | 10Ā | 1Ā | 20Ā | 1Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 120.0 | 80.0 | NULL | NULL | 2.0 | 2.0 | 189.0 | 1.74 | 154.0 | NULL | NULL | |
10568 | 1987-11-01 | 20359 | 2 | 2 | 1 | 2 | 1 | 14.0 | 2 | 1 | 1 | 1 | 3 | 2.0 | 2.0 | 2.0 | 80Ā | 4.5 | 0.98 | 38Ā | 3.16 | 1987Ā | 11Ā | 1Ā | 20Ā | 1Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 100.0 | 70.0 | NULL | NULL | 2.0 | 2.0 | 174.0 | 0.79 | 70.0 | NULL | NULL | |
10569 | 1987-10-01 | 20360 | 2 | 2 | 5 | 2 | 1 | 0.0 | 2 | 2 | 1 | 2 | 3 | 2.0 | 2.0 | 2.0 | 95Ā | 5.51 | 1.29 | 50Ā | 3.68 | 1987Ā | 10Ā | 1Ā | 40Ā | 1Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 120.0 | 80.0 | NULL | NULL | 2.0 | 2.0 | 213.0 | 1.18 | 104.0 | NULL | NULL | |
10570 | 1987-10-01 | 20362 | 1 | 2 | 1 | 2 | 1 | 0.0 | 2 | 6 | 4 | 1 | 3 | 2.0 | 2.0 | 2.0 | 56Ā | 4.89 | 3Ā | 116Ā | 1.51 | 1987Ā | 10Ā | 1Ā | 40Ā | 1Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 140.0 | 85.0 | NULL | NULL | 2.0 | 2.0 | 189.0 | 0.82 | 73.0 | NULL | NULL | |
10571 | 1978-09-01 | 30037 | 2 | 1 | 1 | 2 | 3 | 15.0 | 2 | 4 | 1 | 2 | 4 | 2.0 | 2.0 | 1.0 | 72Ā | 5.61 | nanĀ | nanĀ | nanĀ | 1978Ā | 9Ā | 1Ā | 20Ā | 1Ā | NULL | NULL | NULL | NULL | NULL | 0.0 | 0.0 | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | NULL | 125.0 | 90.0 | 9.0 | 17.0 | 2.0 | 2.0 | 217.0 | NULL | NULL | NULL | NULL |
10572 rows x 67 columns
memory usage: 5.82 MB
name: contr
type: getml.DataFrame
1.3 Define relational modelĀ¶
To start with relational learning, we need to specify an abstract data model. Here, we use the high-level star schema API that allows us to define the abstract data model and construct a container with the concrete data at one-go. While a simple StarSchema
indeed works in many cases, it is not sufficient for more complex data models like schoflake schemas, where you would have to define the data model and construct the container in separate steps, by utilzing getML's full-fledged data model and container APIs respectively.
split = getml.data.split.random(train=0.7, test=0.3)
star_schema = getml.data.StarSchema(population, split=split)
star_schema.join(contr, on="ICO", time_stamps=("REFERENCE_DATE", "CONTROL_DATE"))
star_schema
data frames | staging table | |
---|---|---|
0 | population | POPULATION__STAGING_TABLE_1 |
1 | contr | CONTR__STAGING_TABLE_2 |
subset | name | rows | type | |
---|---|---|---|---|
0 | test | population | 8557 | View |
1 | train | population | 19876 | View |
name | rows | type | |
---|---|---|---|
0 | contr | 10572 | DataFrame |
2. Predictive modelingĀ¶
We loaded the data, defined the roles, units and the abstract data model. Next, we create a getML pipeline for relational learning.
2.1 Fitting the pipelineĀ¶
To illustrate the problem of computational complexity, we fit two getML pipelines using different feature learning algorithms:
The first pipeline uses RelMT
and the second pipeline uses Relboost
. To demonstrate the power of feature ensembling, we build a third pipeline that uses both algorithms. At the end, we compare the runtime and predictive accuracies of all three pipelines.
relmt = getml.feature_learning.RelMT(
num_features=30,
loss_function=getml.feature_learning.loss_functions.CrossEntropyLoss,
num_threads=1,
)
relboost = getml.feature_learning.Relboost(
num_features=60,
loss_function=getml.feature_learning.loss_functions.CrossEntropyLoss,
min_num_samples=500,
max_depth=2,
num_threads=1,
)
feature_selector = getml.predictors.XGBoostClassifier()
xgboost = getml.predictors.XGBoostClassifier(
max_depth=5, reg_lambda=100.0, learning_rate=0.1
)
pipe1 = getml.pipeline.Pipeline(
tags=["relmt"],
data_model=star_schema.data_model,
feature_learners=relmt,
feature_selectors=feature_selector,
share_selected_features=0.8,
predictors=xgboost,
include_categorical=True,
)
pipe1
Pipeline(data_model='population', feature_learners=['RelMT'], feature_selectors=['XGBoostClassifier'], include_categorical=True, loss_function='CrossEntropyLoss', peripheral=['contr'], predictors=['XGBoostClassifier'], preprocessors=[], share_selected_features=0.8, tags=['relmt'])
pipe2 = getml.pipeline.Pipeline(
tags=["relboost"],
data_model=star_schema.data_model,
feature_learners=relboost,
feature_selectors=feature_selector,
share_selected_features=0.8,
predictors=xgboost,
include_categorical=True,
)
pipe2
Pipeline(data_model='population', feature_learners=['Relboost'], feature_selectors=['XGBoostClassifier'], include_categorical=True, loss_function='CrossEntropyLoss', peripheral=['contr'], predictors=['XGBoostClassifier'], preprocessors=[], share_selected_features=0.8, tags=['relboost'])
We begin with RelMT. Features generated by RelMT suffer from quadratic complexity. Luckily, the number of columns is not too high, so it is still manageble.
It always a good idea to check before fitting. There is one minor warning: This warning means that about 12% of patients never appear in CONTR (presumably because they never showed up to their health check-ups).
pipe1.check(star_schema.train)
Checking data model... Staging... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] Checking... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] The pipeline check generated 1 issues labeled INFO and 0 issues labeled WARNING.
type | label | message | |
---|---|---|---|
0 | INFO | FOREIGN KEYS NOT FOUND | When joining POPULATION__STAGING_TABLE_1 and CONTR__STAGING_TABLE_2 over 'ICO' and 'ICO', there are no corresponding entries for 12.291205% of entries in 'ICO' in 'POPULATION__STAGING_TABLE_1'. You might want to double-check your join keys. |
pipe1.fit(star_schema.train)
Checking data model... Staging... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] The pipeline check generated 1 issues labeled INFO and 0 issues labeled WARNING. To see the issues in full, run .check() on the pipeline. Staging... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] RelMT: Training features... 100% |āāāāāāāāāā| [elapsed: 04:39, remaining: 00:00] RelMT: Building features... 100% |āāāāāāāāāā| [elapsed: 00:01, remaining: 00:00] XGBoost: Training as feature selector... 100% |āāāāāāāāāā| [elapsed: 00:03, remaining: 00:00] XGBoost: Training as predictor... 100% |āāāāāāāāāā| [elapsed: 00:04, remaining: 00:00] Trained pipeline. Time taken: 0h:4m:47.297086
Pipeline(data_model='population', feature_learners=['RelMT'], feature_selectors=['XGBoostClassifier'], include_categorical=True, loss_function='CrossEntropyLoss', peripheral=['contr'], predictors=['XGBoostClassifier'], preprocessors=[], share_selected_features=0.8, tags=['relmt', 'container-9DTKIA'])
Let's see how well Relboost does. This algorithm has linear complexity in the number of columns.
pipe2.fit(star_schema.train)
Checking data model... Staging... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] Checking... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] The pipeline check generated 1 issues labeled INFO and 0 issues labeled WARNING. To see the issues in full, run .check() on the pipeline. Staging... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] Relboost: Training features... 100% |āāāāāāāāāā| [elapsed: 00:19, remaining: 00:00] Relboost: Building features... 100% |āāāāāāāāāā| [elapsed: 00:02, remaining: 00:00] XGBoost: Training as feature selector... 100% |āāāāāāāāāā| [elapsed: 00:02, remaining: 00:00] XGBoost: Training as predictor... 100% |āāāāāāāāāā| [elapsed: 00:03, remaining: 00:00] Trained pipeline. Time taken: 0h:0m:25.559479
Pipeline(data_model='population', feature_learners=['Relboost'], feature_selectors=['XGBoostClassifier'], include_categorical=True, loss_function='CrossEntropyLoss', peripheral=['contr'], predictors=['XGBoostClassifier'], preprocessors=[], share_selected_features=0.8, tags=['relboost', 'container-9DTKIA'])
Note that this runs through in under a minute. This demonstrates the power of computational complexity theory. If we had more columns, the difference between these two algorithms would become even more noticable.
2.2 Model evaluationĀ¶
pipe1.score(star_schema.test)
Staging... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] Preprocessing... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] RelMT: Building features... 100% |āāāāāāāāāā| [elapsed: 00:01, remaining: 00:00]
date time | set used | target | accuracy | auc | cross entropy | |
---|---|---|---|---|---|---|
0 | 2024-02-21 14:59:13 | train | TARGET | 0.9887 | 0.8768 | 0.05199 |
1 | 2024-02-21 14:59:39 | test | TARGET | 0.9867 | 0.7068 | 0.06838 |
pipe2.score(star_schema.test)
Staging... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] Preprocessing... 100% |āāāāāāāāāā| [elapsed: 00:00, remaining: 00:00] Relboost: Building features... 100% |āāāāāāāāāā| [elapsed: 00:01, remaining: 00:00]
date time | set used | target | accuracy | auc | cross entropy | |
---|---|---|---|---|---|---|
0 | 2024-02-21 14:59:38 | train | TARGET | 0.9878 | 0.8881 | 0.05407 |
1 | 2024-02-21 14:59:40 | test | TARGET | 0.9867 | 0.7129 | 0.06743 |
2.3 Studying the featuresĀ¶
It is always a good idea to study the features generated by the algorithms.
names, correlations = pipe2.features.correlations()
plt.subplots(figsize=(20, 10))
plt.bar(names, correlations)
plt.title("feature correlations")
plt.grid(True)
plt.xlabel("features")
plt.ylabel("correlations")
plt.xticks(rotation="vertical")
plt.show()
names, importances = pipe2.features.importances()
plt.subplots(figsize=(20, 10))
plt.bar(names, importances)
plt.title("feature importances")
plt.grid(True)
plt.xlabel("features")
plt.ylabel("importances")
plt.xticks(rotation="vertical")
plt.show()
As we can see from these figures we need many features to get a good result. No single feature is very correlated with the target and the feature importance is not concentrated on a small number of features (as is often the case with other data sets).
This implies that the many columns in the data set are actually needed. The reason we emphasize that is that we sometimes see data sets with many columns, but after analyzing them we find and only a handful of these columns are actually needed. This is not one of these times.
The most important features look like this:
pipe1.features.to_sql()[pipe1.features.sort(by="importances")[0].name]
DROP TABLE IF EXISTS "FEATURE_1_2";
CREATE TABLE "FEATURE_1_2" AS
SELECT SUM(
CASE
WHEN ( t2."zmkour" IN ( '7' ) ) AND ( t1."reference_date" > 852038400.000000 ) THEN COALESCE( t1."age" - 58.55541532813217, 0.0 ) * 0.6992075915192288 + COALESCE( t1."participation" - -1887.414410279945, 0.0 ) * -0.3722162052975991 + COALESCE( t1."vyska" - 174.626548875631, 0.0 ) * 0.002072665325539906 + COALESCE( t1."vaha" - 80.15500229463056, 0.0 ) * 0.7135789936249097 + COALESCE( t1."syst1" - 132.0390087195962, 0.0 ) * 0.6204509957556447 + COALESCE( t1."diast1" - 83.58329508949059, 0.0 ) * -0.2852251406284317 + COALESCE( t1."syst2" - 129.4263423588802, 0.0 ) * 0.5416045970884422 + COALESCE( t1."diast2" - 83.2121385956861, 0.0 ) * -0.7691376230609942 + COALESCE( t1."tric" - 9.410853602569986, 0.0 ) * -0.4921979178739925 + COALESCE( t1."subsc" - 18.22670949977054, 0.0 ) * 0.2235547808836321 + COALESCE( t1."chlst" - 232.7238412115649, 0.0 ) * -0.2305456390285043 + COALESCE( t1."trigl" - 134.5320100963745, 0.0 ) * -0.08732404144815611 + COALESCE( t1."koureni" - 3.382973841211565, 0.0 ) * 3.578257813757872 + COALESCE( t1."dobakour" - 6.960188159706287, 0.0 ) * 3.420605423460954 + COALESCE( t1."byvkurak" - 1.689421753097751, 0.0 ) * 1.468999926745384 + COALESCE( t1."pivomn" - 1.800826067003213, 0.0 ) * 2.006077827378471 + COALESCE( t1."vinomn" - 4.298990362551629, 0.0 ) * 1.509295328270136 + COALESCE( t1."lihmn" - 6.948141349242772, 0.0 ) * 1.689453613870151 + COALESCE( t1."kava" - 1.919343735658559, 0.0 ) * 15.33082510657582 + COALESCE( t1."caj" - 4.731642955484167, 0.0 ) * 10.97404738286759 + COALESCE( t1."cukr" - 4.495869664983938, 0.0 ) * 1.130351262906143 + COALESCE( t1."reference_date" - 614493572.4644332, 0.0 ) * -1.551420482850177e-08 + COALESCE( t1."entry_date" - 230484053.9697109, 0.0 ) * -2.196562174142811e-07 + COALESCE( t2."hmot" - 81.27387640449439, 0.0 ) * 0.2535134048176608 + COALESCE( t2."hdlmg" - 35.46147672552167, 0.0 ) * -0.04134495275193414 + COALESCE( t2."chlst" - 5.892755818619529, 0.0 ) * -10.72183117055155 + COALESCE( t2."hdl" - 0.9149819422150931, 0.0 ) * -1.580592374193315 + COALESCE( t2."ldl" - 2.463498194221503, 0.0 ) * -1.709624186115308 + COALESCE( t2."control_date" - 533301113.6436597, 0.0 ) * -6.539468045584373e-08 + 6.5967578763439345e+00
WHEN ( t2."zmkour" IN ( '7' ) ) AND ( t1."reference_date" <= 852038400.000000 OR t1."reference_date" IS NULL ) THEN COALESCE( t1."age" - 58.55541532813217, 0.0 ) * 7.009110879137228 + COALESCE( t1."participation" - -1887.414410279945, 0.0 ) * 13.11693059229238 + COALESCE( t1."vyska" - 174.626548875631, 0.0 ) * 1.269893142479568 + COALESCE( t1."vaha" - 80.15500229463056, 0.0 ) * -3.162664037823355 + COALESCE( t1."syst1" - 132.0390087195962, 0.0 ) * 0.3787421899628546 + COALESCE( t1."diast1" - 83.58329508949059, 0.0 ) * 1.971210134455975 + COALESCE( t1."syst2" - 129.4263423588802, 0.0 ) * 0.6635424419434129 + COALESCE( t1."diast2" - 83.2121385956861, 0.0 ) * -2.939270936123821 + COALESCE( t1."tric" - 9.410853602569986, 0.0 ) * 2.383223854209837 + COALESCE( t1."subsc" - 18.22670949977054, 0.0 ) * 1.498266105418248 + COALESCE( t1."chlst" - 232.7238412115649, 0.0 ) * -0.0517718309107393 + COALESCE( t1."trigl" - 134.5320100963745, 0.0 ) * 0.2276783275568421 + COALESCE( t1."koureni" - 3.382973841211565, 0.0 ) * -4.23254041935678 + COALESCE( t1."dobakour" - 6.960188159706287, 0.0 ) * 5.719150573247517 + COALESCE( t1."byvkurak" - 1.689421753097751, 0.0 ) * 2.70196864001066 + COALESCE( t1."pivomn" - 1.800826067003213, 0.0 ) * 18.79720042999637 + COALESCE( t1."vinomn" - 4.298990362551629, 0.0 ) * -10.93785280575887 + COALESCE( t1."lihmn" - 6.948141349242772, 0.0 ) * -13.52764654128384 + COALESCE( t1."kava" - 1.919343735658559, 0.0 ) * 15.62103938182056 + COALESCE( t1."caj" - 4.731642955484167, 0.0 ) * -8.797933019568511 + COALESCE( t1."cukr" - 4.495869664983938, 0.0 ) * -3.757872625493459 + COALESCE( t1."reference_date" - 614493572.4644332, 0.0 ) * -6.356934996127425e-07 + COALESCE( t1."entry_date" - 230484053.9697109, 0.0 ) * 5.640187513549882e-07 + COALESCE( t2."hmot" - 81.27387640449439, 0.0 ) * 3.023356785615304 + COALESCE( t2."hdlmg" - 35.46147672552167, 0.0 ) * -0.7240192100118329 + COALESCE( t2."chlst" - 5.892755818619529, 0.0 ) * -4.024059582855235 + COALESCE( t2."hdl" - 0.9149819422150931, 0.0 ) * -29.58919968380793 + COALESCE( t2."ldl" - 2.463498194221503, 0.0 ) * 1.778411316422035 + COALESCE( t2."control_date" - 533301113.6436597, 0.0 ) * 2.163236433930246e-07 + -7.9015375976371462e+00
WHEN ( t2."zmkour" NOT IN ( '7' ) OR t2."zmkour" IS NULL ) AND ( t1."htrisk" IN ( '6' ) ) THEN COALESCE( t1."age" - 58.55541532813217, 0.0 ) * -3.397114537185319 + COALESCE( t1."participation" - -1887.414410279945, 0.0 ) * -8.217148782839798 + COALESCE( t1."vyska" - 174.626548875631, 0.0 ) * 0.8989975149085871 + COALESCE( t1."vaha" - 80.15500229463056, 0.0 ) * -1.237315019051493 + COALESCE( t1."syst1" - 132.0390087195962, 0.0 ) * -0.9928276153180301 + COALESCE( t1."diast1" - 83.58329508949059, 0.0 ) * -2.788494436812396 + COALESCE( t1."syst2" - 129.4263423588802, 0.0 ) * -0.4818397334334769 + COALESCE( t1."diast2" - 83.2121385956861, 0.0 ) * 6.971936650721264 + COALESCE( t1."tric" - 9.410853602569986,