Learn R Programming

FFTrees 2.0.0

The R package FFTrees creates, visualizes and evaluates fast-and-frugal decision trees (FFTs) for solving binary classification tasks, using the algorithms and methods described in Phillips, Neth, Woike & Gaissmaier (2017, doi 10.1017/S1930297500006239 | html | PDF).

What are fast-and-frugal trees (FFTs)?

Fast-and-frugal trees (FFTs) are simple and transparent decision algorithms for solving binary classification problems. The key feature making FFTs faster and more frugal than other decision trees is that every node allows making a decision. When predicting novel cases, the performance of FFTs competes with more complex algorithms and machine learning techniques, such as logistic regression (LR), support-vector machines (SVM), and random forests (RF). Apart from being faster and requiring less information, FFTs tend to be robust against overfitting, and are easy to interpret, use, and communicate.

Installation

The latest release of FFTrees is available from CRAN at https://CRAN.R-project.org/package=FFTrees:

install.packages("FFTrees")

The current development version can be installed from its GitHub repository at https://github.com/ndphillips/FFTrees:

# install.packages("devtools")
devtools::install_github("ndphillips/FFTrees", build_vignettes = TRUE)

Getting started

As an example, let’s create a FFT predicting patients’ heart disease status (Healthy vs. Disease) based on the heartdisease dataset included in FFTrees:

library(FFTrees)  # load package

Using data

The heartdisease data provides medical information for 303 patients that were examined for heart disease. The full data contains a binary criterion variable describing the true state of each patient and were split into two subsets: A heart.train set for fitting decision trees, and heart.test set for a testing these trees. Here are the first rows and columns of both subsets of the heartdisease data:

  • heart.train (the training / fitting data) describes 150 patients:
diagnosisagesexcptrestbpscholfbsrestecgthalachexangoldpeakslopecathal
FALSE440np1081410normal17500.6flat0normal
FALSE510np1403080hypertrophy14201.5up1normal
FALSE521np1382230normal16900.0up1normal
TRUE481aa1102290normal16801.0down0rd
FALSE591aa1402210normal16410.0up0normal
FALSE581np1052400hypertrophy15410.6flat0rd

Table 1: Beginning of the heart.train subset (using the data of 150 patients for fitting/training FFTs).

  • heart.test (the testing / prediction data) describes 153 different patients on the same variables:
diagnosisagesexcptrestbpscholfbsrestecgthalachexangoldpeakslopecathal
FALSE510np1202950hypertrophy15700.6up0normal
TRUE451ta1102640normal13201.2flat0rd
TRUE531a1232820normal9512.0flat2rd
TRUE451a1423090hypertrophy14710.0flat3rd
FALSE661a1203020hypertrophy15100.4flat0normal
TRUE481a1302561hypertrophy15010.0up2rd

Table 2: Beginning of the heart.test subset (used to predict diagnosis for 153 new patients).

Our challenge is to predict each patient’s diagnosis — a column of logical values indicating the true state of each patient (i.e., TRUE or FALSE, based on the patient suffering or not suffering from heart disease) — from the values of potential predictors.

Questions answered by FFTs

To solve binary classification problems by FFTs, we must answer two key questions:

  • Which of the variables should we use to predict the criterion?
  • How should we use and combine predictor variables into FFTs?

Once we have created some FFTs, additional questions include:

  • How accurate are the predictions of a specific FFT?
  • How does its performance compare with alternative machine-learning algorithms?
  • How costly are the predictions of each algorithm?

The FFTrees package answers these questions by creating FFTs and allowing to evaluate, visualize, and compare them to alternative algorithms.

Creating fast-and-frugal trees (FFTs)

We use the main FFTrees() function to create FFTs for the heart.train data and evaluate their predictive performance on the heart.test data:

  • The main FFTrees() function allows creating an FFTrees object for the heartdisease data:
# Create an FFTrees object from the heartdisease data:
heart_fft <- FFTrees(formula = diagnosis ~.,
                     data = heart.train,
                     data.test = heart.test, 
                     decision.labels = c("Healthy", "Disease"))

Evaluating FFTrees() analyzes the training data, creates several FFTs, and applies them to the test data. The results are stored in an object heart_fft, which can be printed, plotted and summarized (with options for selecting specific data or trees).

  • Let’s plot our FFTrees object to visualize a tree and its predictive performance (on the test data):
# Plot the best tree applied to the test data: 
plot(heart_fft,
     data = "test",
     main = "Heart Disease")

Figure 1: A fast-and-frugal tree (FFT) predicting heart disease for test data and its performance characteristics.

  • A summary of the trees in our FFTrees object and their key performance statistics can be obtained by summary(heart_fft).

  • Additionally, we can compare the predictive performance of different machine learning algorithms on a range of metrics:

# Compare predictive performance across algorithms: 
heart_fft$competition$test
#> # A tibble: 5 × 18
#>   algorithm     n    hi    fa    mi    cr  sens  spec    far   ppv   npv   acc
#>   <chr>     <int> <int> <int> <int> <int> <dbl> <dbl>  <dbl> <dbl> <dbl> <dbl>
#> 1 fftrees     153    64    19     9    61 0.877 0.762 0.238  0.771 0.871 0.817
#> 2 lr          153    55    13    18    67 0.753 0.838 0.162  0.809 0.788 0.797
#> 3 cart        153    50    19    23    61 0.685 0.762 0.238  0.725 0.726 0.725
#> 4 rf          153    59     8    14    72 0.808 0.9   0.1    0.881 0.837 0.856
#> 5 svm         153    55     7    18    73 0.753 0.912 0.0875 0.887 0.802 0.837
#> # ℹ 6 more variables: bacc <dbl>, wacc <dbl>, dprime <dbl>, cost_dec <dbl>,
#> #   cost_cue <dbl>, cost <dbl>

Building FFTs from verbal descriptions

FFTs are so simple that we even can create them ‘from words’ and then apply them to data.
For example, let’s create a tree with the following three nodes and evaluate its performance on the heart.test data:

  1. If sex = 1, predict Disease.
  2. If age < 45, predict Healthy.
  3. If thal = {fd, normal}, predict Healthy,
    otherwise, predict Disease.

These conditions can directly be supplied to the my.tree argument of FFTrees():

# Create custom FFT 'in words' and apply it to test data:

# 1. Create my own FFT (from verbal description):
my_fft <- FFTrees(formula = diagnosis ~., 
                  data = heart.train,
                  data.test = heart.test, 
                  decision.labels = c("Healthy", "Disease"),
                  my.tree = "If sex = 1, predict Disease.
                             If age < 45, predict Healthy.
                             If thal = {fd, normal}, predict Healthy,  
                             Otherwise, predict Disease.")

# 2. Plot and evaluate my custom FFT (for test data):
plot(my_fft,
     data = "test",
     main = "My custom FFT")

Figure 2: An FFT predicting heart disease created from a verbal description.

The performance measures (in the bottom panel of Figure 2) show that this particular tree is somewhat biased: It has nearly perfect sensitivity (i.e., is good at identifying cases of Disease) but suffers from low specificity (i.e., performs poorly in identifying Healthy cases). Expressed in terms of its errors, my_fft incurs few misses at the expense of many false alarms. Although the accuracy of our custom tree still exceeds the data’s baseline by a fair amount, the FFTs in heart_fft (created above) strike a better balance.

Overall, what counts as the “best” tree for a particular problem depends on many factors (e.g., the goal of fitting vs. predicting data and the trade-offs between maximizing accuracy vs. incorporating the costs of cues or errors). To explore this range of options, the FFTrees package enables us to design and evaluate a range of FFTs.

References

We had fun creating the FFTrees package and hope you like it too! As a comprehensive, yet accessible introduction to FFTs, we recommend our article in the journal Judgment and Decision Making (2017), entitled FFTrees: A toolbox to create, visualize,and evaluate fast-and-frugal decision trees (available in html | PDF ).

Citation (in APA format):

  • Phillips, N. D., Neth, H., Woike, J. K. & Gaissmaier, W. (2017). FFTrees: A toolbox to create, visualize, and evaluate fast-and-frugal decision trees. Judgment and Decision Making, 12 (4), 344–368. doi 10.1017/S1930297500006239

We encourage you to read the article to learn more about the history of FFTs and how the FFTrees package creates, visualizes, and evaluates them. When using FFTrees in your own work, please cite us and share your experiences (e.g., on GitHub) so we can continue developing the package.

By 2023, over 100 scientific publications have used or cited FFTrees (see Google Scholar for the full list).
Examples include:

  • Lötsch, J., Haehner, A., & Hummel, T. (2020). Machine-learning-derived rules set excludes risk of Parkinson’s disease in patients with olfactory or gustatory symptoms with high accuracy. Journal of Neurology, 267(2), 469–478. doi 10.1007/s00415-019-09604-6

  • Kagan, R., Parlee, L., Beckett, B., Hayden, J. B., Gundle, K. R., & Doung, Y. C. (2020). Radiographic parameter-driven decision tree reliably predicts aseptic mechanical failure of compressive osseointegration fixation. Acta Orthopaedica, 91(2), 171–176. doi 10.1080/17453674.2020.1716295

  • Klement, R. J., Sonke, J. J., Allgäuer, M., Andratschke, N., Appold, S., Belderbos, J., … & Mantel, F. (2020). Correlating dose variables with local tumor control in stereotactic body radiotherapy for early stage non-small cell lung cancer: A modeling study on 1500 individual treatments. International Journal of Radiation Oncology * Biology * Physics. doi 10.1016/j.ijrobp.2020.03.005

  • Nobre, G. G., Hunink, J. E., Baruth, B., Aerts, J. C., & Ward, P. J. (2019). Translating large-scale climate variability into crop production forecast in Europe. Scientific Reports, 9(1), 1–13. doi 10.1038/s41598-018-38091-4

  • Buchinsky, F. J., Valentino, W. L., Ruszkay, N., Powell, E., Derkay, C. S., Seedat, R. Y., … & Mortelliti, A. J. (2019). Age at diagnosis, but not HPV type, is strongly associated with clinical course in recurrent respiratory papillomatosis. PloS One, 14(6). doi 10.1371/journal.pone.0216697


[File README.Rmd last updated on 2023-06-03.]

Copy Link

Version

Install

install.packages('FFTrees')

Monthly Downloads

438

Version

2.0.0

License

CC0

Issues

Pull Requests

Stars

Forks

Maintainer

Last Published

June 5th, 2023

Functions in FFTrees (2.0.0)

fertility

Fertility data
fact_clean

Clean factor variables in prediction data
fftrees_cuerank

Calculate thresholds that optimize some statistic (goal) for cues in data
edit_nodes

Edit nodes in an FFT definition
drop_nodes

Drop a node from an FFT definition
describe_data

Describe data
fftrees_apply

Apply an FFT to data and generate accuracy statistics
fftrees_create

Create an object of class FFTrees
contraceptive

Contraceptive use data
creditapproval

Credit approval data
flip_exits

Flip exits in an FFT definition
fftrees_fitcomp

Fit competitive algorithms
fftrees_wordstofftrees

Convert a verbal description of an FFT into an FFTrees object
fftrees_threshold_factor_grid

Perform a grid search over factor and return accuracy statistics for a given factor cue
fftrees_ffttowords

Describe a fast-and-frugal tree (FFT) in words
fftrees_threshold_numeric_grid

Perform a grid search over thresholds and return accuracy statistics for a given numeric cue
forestfires

Forest fires data
fftrees_grow_fan

Grow fast-and-frugal trees (FFTs) using the fan algorithms
fftrees_ranktrees

Rank FFTs by current goal
fftrees_define

Create FFT definitions
get_fft_df

Get FFT definitions (from an FFTrees object x)
heartdisease

Heart disease data
get_best_tree

Select the best tree (from current set of FFTs)
get_exit_type

Get exit type (from a vector x of FFT exit descriptions)
iris.v

Iris data
heart.test

Heart disease testing data
heart.train

Heart disease training data
showcues

Visualize cue accuracies (as points in ROC space)
sonar

Sonar data
inwords

Provide a verbal description of an FFT
print.FFTrees

Print basic information of fast-and-frugal trees (FFTs)
read_fft_df

Read an FFT definition from tree definitions
wine

Wine tasting data
voting

Voting data
heart.cost

Cue costs for the heartdisease data
plot.FFTrees

Plot an FFTrees object
predict.FFTrees

Predict classification outcomes or probabilities from data
mushrooms

Mushrooms data
summary.FFTrees

Summarize an FFTrees object
write_fft_df

Write an FFT definition to tree definitions
titanic

Titanic survival data
select_nodes

Select nodes from an FFT definition
reorder_nodes

Reorder nodes in an FFT definition
classtable

Compute classification statistics for binary prediction and criterion (e.g.; truth) vectors
add_stats

Add decision statistics to data (based on frequency counts of a 2x2 classification outcomes)
breastcancer

Physiological data of patients tested for breast cancer
add_nodes

Add nodes to an FFT definition
comp_pred

Fit and predict competing classification algorithms
FFTrees

Main function to create and apply fast-and-frugal trees (FFTs)
add_fft_df

Add an FFT definition to tree definitions
FFTrees.guide

Open the FFTrees package guide
blood

Blood donation data
car

Car acceptability data