-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathFinal Report - Regression - DoA - Markdown.Rmd
796 lines (565 loc) · 38.6 KB
/
Final Report - Regression - DoA - Markdown.Rmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
---
title: "Final Report - DoA Regression Analysis"
author: "Christian Willig"
date: "`r Sys.Date()`"
output:
word_document:
reference_docx: word-styles-reference-01.docx
---
```{r setup, include=FALSE}
library(tidymodels)
library(readxl)
library(flextable)
library(DataExplorer)
library(dlookr)
library(ggExtra)
library(cowplot)
knitr::opts_chunk$set(echo = TRUE)
# loading data for tables
load(file = "procedures_data.RData")
```
# 1. Abstract
This report proposes two new depth of anaesthesia (DoA) indexes. We were given 15 cases of BIS index data separated in training and test datasets. The objective of this report is to create a new BIS index that could replace the current one being used at the Well Hospital by using supervised machine learning algorithms. After analyzing the data and feature selection, a screening process was performed in order to assess the performance of four potential models on the training data set. Two models were selected based on their performance: K-nearest neighbor and Random forest. Both models performed moderately well on the test dataset with an overall RMSE of 10.1 and 9.4 respectively. The results were evaluated using the correlation coefficient and Bland-Altman methods.
All code was written in R using several packages that helped on developing this report. The appendix contains all the pieces of code used to implement this work.
\pagebreak
# 2. Report body
The report is structured in the following way. We start with a summary of the studies that have used machine learning algorithms to create alternative BIS indexes. Next, we dive into the data analysis of the dataset which provides an understanding of the data we are using here. The rest of the sections present the methodology followed to build the new index and the results. Finally, all code used in the report are presented in the appendix section.
## 2.1 Research
The Depth of Anaesthesia (DoA) is a topic that has been researched for some time now and it's very relevant as it measures the level of consciousness a person has while on an anaesthetic drug.
It can be monitored via EEG signals because it reflects the electrical activity of the brain, so in an awake state, these signals show high-frequency activity and when the body interacts with anaesthesia, the signals slow down and become more synchronised. In a deep state the EEG signals show low-frequency activity. Controlling the amount of anaesthesia is the main challenge for anaesthetists due to the effect that a high/low dosage of anaesthesia has in each patient. These effects can lead to coma and postoperative complications or the patient could suffer from pain and waking up in the middle of surgery.
Many commercial models have been developed for monitoring DoA and are based on EEG signals.
Currently the BIS index is the standard device for prediction of DoA, but it has limitations:
* It doesn't work well with all drugs
* It's not accurate across patients
* It's being delayed
Studies have shown that EEG is a non-stationary (Elbert et al., 1994; Natarajan et al., 2004) and it exhibits non-linear behaviors (Lalitha et al., 2007, Elbert et al., 1994; Natarajan et al., 2004) which has derived in feature extractions of EEG signals based on non-linear dynamics.
Machine learning algorithms can be used to identify patterns in EEG signals that are associated with different DoA levels. For example,
* Gu et al., 2019 and Shalbaf et al., 2013 used EEG signals and an **artificial neural network** to monitor DoA.
* Peker et al, 2015 used **Random Forest** for determining the DoA level.
* Liu et al., 2019 used **CNN** to model patient's DoA.
* Yi et al., 2022 used **several machine learning models** to assess DoA (Robust Linear, Gaussian SVM, Squared Exponential Gaussian, etc).
As stated in the introduction of this piece of work, the problem to solve here is a regression problem. Some studies have taken a classification approach where instead of predicting the index they predict the state of DoA, and that converts the problem into a classification problem.
The three main characteristics of the problem that I have identified as relevant for the regression model are:
* **Accuracy**: The model has to be accurate enough in predicing the DoA state. This is important given the consequences of giving the wrong doses of anaesthesia to patients.
* **Speed**: The model has to be quick to make prediction as it is expected to work in real time, like during surgery for example. And also, it needs to be quick during training as it is expected to be retrained for different type of patients and type of drugs.
* **Robustness**: The model should be robust to noise.
There is no conclusion about which ML model is the best for DoA monitoring but there is one certainty about machine learning having the potential to help and improve the anaesthetist work by providing accuracy and robustness in predicting DoA.
## 2.2 Data Analysis
To begin with my analysis I imported the file "Project data set 1 (for reports 1 and 3) .xlsx" into my workspace. Then I read every sheet named "Train X" and appended the sets together into one. Following the construction of one dataset I checked the structure of the dataset. I have been provided a training dataset consisting of 27,452 observations. The first 10 rows of the dataset are shown in table 1.
```{r echo=FALSE, message=FALSE, warning=FALSE}
# load data
# read data set
# accessing all the sheets from dataset
sheet = excel_sheets("Project data set 1 (for reports 1 and 3) .xlsx")
# applying sheet names to dataframe names
data_frame = lapply(setNames(sheet[1:10], sheet[1:10]),
function(x) read_excel("Project data set 1 (for reports 1 and 3) .xlsx", sheet=x))
# attaching all dataframes together
eeg_data_train = bind_rows(data_frame, .id="Sheet") %>% select(-Sheet)
# applying sheet names to dataframe names
data_frame = lapply(setNames(sheet[11:15], sheet[11:15]),
function(x) read_excel("Project data set 1 (for reports 1 and 3) .xlsx", sheet=x))
eeg_data_test = bind_rows(data_frame, .id="Sheet") %>% select(-Sheet)
eeg_data_train %>%
round(4) %>%
head(5)%>%
flextable()%>%
autofit() %>%
set_caption('Table 1. First 10 Rows of Dataset.')
```
### 2.2.1 Statistics
After understanding the structure of the dataset we run statistical calculations as shown in table 2 in order to assess central tendency and dispersion of the data.
```{r echo=FALSE, message=FALSE, warning=FALSE}
library(papeR)
eeg_data_train %>%
describe(statistics = c("mean", "sd", "skewness", "kurtosis"))%>%
mutate(var = sd^2) %>%
relocate(var, .before = sd) %>%
rename(variables = described_variables) %>%
bind_cols(eeg_data_train %>% diagnose_numeric() %>% select(min, median, max, minus)) %>%
select(variables, n, na, minus, mean, median, var, sd, min, max, skewness, kurtosis) %>%
mutate_at(vars(-variables), funs(round(., 2))) %>%
rename(vars = variables, skew = skewness, kur = kurtosis) %>%
flextable(cwidth = 0.6)%>%
autofit() %>%
set_caption('Table 2. Univariate Profile for Simulated Data')
```
At a high level We can see that the dataset doesn't contain any missing values or negative values at all. This is a good scenario and has saved us to deal with missing values.
The mean value for the BIS index is 43.23 with a standard deviation of 17.01 which indicates the BIS observations are mostly around to the mean value.
Several variables have a variance close to zero which is interesting and deserve to have a closer look. Having variables with zero variance could indicate that there is not much information that is being provided, however, we need to have a look at other characteristics like correlation and relationship with the target variable.
Skewness and kurtosis in the variables are everywhere between positive and negative. One that stands out is variable x4 which has a very high negative skewness and kurtosis which will be reflected in its distribution.
### 2.2.2 Distributions
From the statistical calculations in the previous section we could see that several variables present skewness. Figure 1 corroborates this visually.
```{r echo=FALSE, fig.height=6, fig.width=10, message=FALSE, warning=FALSE, fig.cap = "Figure 1: Histogram of variables", fig.asp = 0.8, fig.width=9}
plot_histogram(eeg_data_train)
```
* BIS: this variable present a slight right skewness with two modes
* x1: this variable is a symmetric bimodal distribution
* x2: this variable is left skewed
* x3: this is variable present an uniform distribution
* x4: this variable is heavily skewed to the left
* x5: this variable is symmetric, close to normal
* x6: this variable is moderately skewed to the right
* x7: this variable is symmatric
* x8: this variable is skewed to the left
### 2.2.3 Relationships
Following the distributions we analysed the relationship between the variables, in terms of strength, direction and form.
We start with a scatter plot matrix between the BIS index and all the independent variables.
We can see in Figure 2 that:
* x1, x5, and x7 present a positive linear relationship with the BIS index.
* x2, x6 and x8 present a non-linear relationship with the BIS index.
* x3 and x4 present no linear relationship with the BIS index.
```{r echo=FALSE, message=FALSE, warning=FALSE, fig.cap="Figure 2: Variable scatter plots", fig.align='center', fig.asp = 0.8, fig.width=9}
plot_scatterplot(eeg_data_train, by = "BIS", sampled_rows = 1000L)
```
Figure 3 provides a visual summary of a pairwise correlation matrix between all the variables. Here I can see that the BIS variable:
* has a strong positive correlation with x2 and x7
* has a moderate positive correlation with x1, x5 and x8
* has a moderate negative correlation with x6
* has a weak negative correlation with x4
* and has a very weak positive correlation with x2
Another interesting insight here is that x1 with x5 present a very strong correlation, and x2 with x7 present a strong correlation. The first case indicate a multicollinearity problem.
```{r echo=FALSE, message=FALSE, warning=FALSE, fig.cap="Figure 3: Correlation matrix", fig.align='center', fig.asp = 0.8, fig.width=9}
plot_correlation(eeg_data_train)
```
### 2.2.4 Outliers
It is important to identify whether our dataset contains outliers or not as this can influence the performance of some machine learning algorithms. We'll assess for outliers visually and via statistics (standard-deviation-based threshold method).
```{r echo=FALSE, message=FALSE, warning=FALSE, fig.cap="Figure 4: Boxplot of independent variables againt BIS index", fig.asp = 0.8, fig.width=9}
plot_boxplot(eeg_data_train, by = "BIS")
```
Table 3 provides more insights around outliers by calculating the mean value of the outliers, the mean value of the variable with outliers and without outliers. This way I can assess how strong the influence of outlier for every variables is.
```{r echo=FALSE, message=FALSE, warning=FALSE}
diagnose_outlier(eeg_data_train) %>%
mutate_at(vars(-variables), funs(round(., 2))) %>%
flextable() %>%
autofit() %>%
set_caption("Table 3: Outlier statistics")
```
We can see that here are a number of outliers in pretty much all variables except x3 which makes sense as it is a uniform distribution. x4, x6 and x7 have more than 10% of their observations as outliers. In terms of strength of influence I can see that the mean values change only for the BIS index and x2. All the other variables maintain their mean values.
### 2.2.5 Normmality test
Checking for normal distribution is common test performed in data analysis. The following table 6 presents the result of Shapiro-Wilk normality test over all variables at once. We can see that based on this test none of the variable passed the normality test.
```{r echo=FALSE, message=FALSE, warning=FALSE}
normality(eeg_data_train) %>%
mutate(p_value = round(p_value,2)) %>%
flextable() %>%
autofit() %>%
set_caption("Table 4: Normality test")
```
### 2.2.6 Data analysis conclusion
From all the analysis performed above we can conclude that the training dataset contains:
* No missing values.
* No negative values: This makes sense as I wouldn't expect negative values for BIS index or EEG variables.
* Presence of outliers: This is an aspect that could introduce problems in machine learning algorithms as some methods are more sensitive to outliers than others. If an outlier is an error then it has to be filter out of the dataset, otherwise, it needs to be investigated. However, given that I don't have much more information about the patients and the variables I can't assume that they are errors coming from the collection process so I will keep the outliers in the dataset and will apply a transformation to generate a more balanced distribution.
* No normal distribution in any of the variables.
* Correlations between features: There is a high correlation between two independent variables which indicates the existence of multicollinearity.
* Linear correlation between target and features: There is a low to moderate correlation between the BIS index and the features
## 2.3 Methodology
Several models have been used in the studies reviewed for this report. Most of them are neural networks with combination of other transformations. At a high level, the approach proposed for this project is to perform:
![Process](process.png)
1. a feature selection step then
2. a modelling step and finally
3. a testing step where final models will run against the test data
In the first step we'll perform several procedures to ensure the input data has enough information to predict the new BIS index.
In the modelling step, instead of picking two models from the given list of options, I'm going to select from the list four potential models to evaluate on the training data set (screening). Then I'll pick the best two based on their performance via k-fold cross validation.
After the final selection, the two selected models will be fitted again and their parameters tuned in order to find the best model configuration.
The final outcome will be the prediction of each test dataset via the two tuned selected models.
## 2.4 Evaluation methods
During the screening process the five pre-selected models will be trained with default parameters using a 10-fold cross-validation training approach. This approach provides two benefits: it reduces the chances of overfitting and provides better estimates on average performance between the models.
They'll be evaluated using root mean square error (RMSE), R square ($R^2$), mean square error (MSE) and mean absolute error (MAE). $R^2$ values go from 0 to 1 where high values indicate higher correlation between the truth and the predicted value from training
After the best two models are selected and evaluated on the test datasets, I'll use the scatter plot diagram and Bland-Altman diagram method to assess the agreement between the new BIS index (prediction) and the BIS index. Also, the pearson correlation coefficient will be calculated to measure correlation.
## 2.5 Machine Learning methods application
### 2.5.1 Software
Due to previous experience, knowledge gained in other courses in my studies, personal interests and experience at work, I have used R during the implementation of this project. While R comes fully equipped with functions and features, working with RStudio as the IDE is a modular process based on different packages that need to be installed. Each package's objective covers different use cases and like this project involves data analysis, machine learning algorithms, visualization and report writing, I have used the following packages to assist me during the project. These packages are:
* Rmarkdowm, package to write reproducible reports where code and text and can be exported into word, html and pdf.
* Tidyverse, package to do all kind of data wrangling.
* Tidymodels, package that contains a collection of modelling packages that support all the machine learning activities like data pre-processing, training, cross-validation, testing, etc.
* Ggplot2, package that provides visualization capabilities in R.
* doParallel, package for managing CPU cores and perform parallel programming.
### 2.5.2 **Feature Selection**
In this step we will perform some model-based feature selection in order to understand how different models rank the feature and based on those output make an informed decision of which feature should be kept for the modelling step.
The following are the procedures that we will use:
**Boruta algorithm**: This is a model-based procedure that uses random forest to apply a feature importance measure and evaluates it in several iterations. The graph below present the result of this procedure and we can see the most important features are: x2, x8, x6, x5 and x4.
**Tree-based method**: This is a model-based feature importance procedure that uses a decision tree model's feature importance function. The graph below show the following features as the most important: x7, x6, x2, x1, x8 and x5.
**Stepwise selection**: This procedure assumes linear regression. It searches for the best possible linear regression model by iterating over different models with different features from the dataset until it minimizes the AIC measure. The result of both way steps here were: x2, x6, x8, x5, x7, x1 and x4.
In summary, the three procedures we run for feature selection didn't pick x3 as a relevant feature, x2 was picked as the most important feature followed by x8.
```{r echo=FALSE, message=FALSE, warning=FALSE}
df <- data.frame(ranking = c(1:7),
Barota = c("x2","x8","x6", "x5","x4","",""),
rPart = c("x7","x6","x2", "x1","x8","x5",""),
stepWise = c("x2","x6","x8", "x5","x7","x1","x4"))
df %>%
flextable() %>%
autofit() %>%
set_caption("Table 5: Summary Feature Importance")
```
Considering the insights from the data analysis section (correlations and skewness) and the first 5 most important features provided by the model-based feature importance procedures above, I have selected the following features for the models: **x2, x5, x6, x7 and x8**.
* x2 was pretty much the most important feature in all three models.
* x5 had a high correlation with x1, however, I selected it due to the fact that x1 was mostly classified as a low importance feature.
* x6 was pretty much the most important feature in all three models.
* x7 was selected high importance feature by one of the models only, however, it presented some good characteristics as a feature, like moderate linear correlation with the BIS and low skewness.
* x8 was selected relatively high in two of the models and presented some good characteristics as a feature, like moderate linear correlation with the BIS, low skewness and low number of outliers.
### 2.5.3 **Data Transformations**
The following data transformation steps will also be applied:
1. Standarisation of the features to prevent scaling issues.
2. Application of log normal transformation to the features with moderate to high skewness (x2 and x6).
These two steps were coded in a preprocessing function which is used by all the models.
### 2.5.4 **Screening process**
The four models we've initially selected belong to a mixture of algorithms that match the characteristics of this problem. They are also low bias models which means that they make very little assumptions about the target and features relationship. We have also considered one non-linear model given that this characteristic has been stated in several studies.
* K-Nearest Neighbor (instance-based algorithm)
* Support vector machines (non-linear algorithm)
* Neural Networks (non-parametric algorithm)
* Random Forest (tree-based algorithm)
* Multivariate Adaptive Regression Splines (non-linear algorithm)
We have left linear algorithms out due to the non-linear relationship nature between target and features (high bias models).
### 2.5.5 Model selection
Table 6 shows the results after running the screening process with the four pre-selected algorithms. The table list the best model at the top by RMSE metric. KNN and RF are the two models that performed the best during the cross validation procedure. These models are not tuned and were trained with default parameter values. As result, KNN and RF both showed a $R^2$ of 0.97 while the other three models ANN, MARS and SVR all obtained an average of 0.85, 0.82 and 0.68 respectively.
```{r echo=FALSE, message=FALSE, warning=FALSE}
screening_results %>%
arrange(rmse) %>%
mutate_at(c("mae","mape","rmse","rsq","smape"), round, 4) %>%
flextable() %>%
set_caption("Table 6: Screening process metric results") %>%
autofit()
```
The best two models that were selected are: **Random Forest and KNN**.
**Hyperparameter tunning**
The model parameters were adjusted by using a grid approach with 25 configuration options.
Each model configuration run is passed through a 10-fold cross validation process to ensure robustness in the metric results.
Random forest parameters:
* mtry: this is an integer number that indicates the number of predictors that will be randomly sampled at each split when creating the tree models.
* trees: this is the number of trees contained in the ensemble. we have used a fixed number of 500.
* min_n: this is an integer number that indicates the minimum number of data points in a node that are required for the node to be split further.
KNN parameters:
* neighbors: this is an integer that indicates the number of neighbors nearby to select.
* weight_func: this is a character value that indicates the distance weighting function to be used.
* dist_power: this is an integer number that indicates the Minkowski Distance Order.
## 2.6 Results
The following tables and figures present the training and testing results from the two selected final models. Table 7 summarises the metrics obtained from the runs. The hyperparameters that fitted the training dataset resulted in a RMSE of 1.24 for KNN with a $R^2$ of 0.99, and 1.09 for RF with a $R^2$ of 0.99 as well.The generalisation metrics on the test dataset were 10.1 for KNN with a $R^2$ of 0.68 and 9.4 for RF with a $R^2$ of 0.7.
```{r echo=FALSE, message=FALSE, warning=FALSE}
final_metrics %>%
flextable() %>%
autofit() %>%
add_header_row(
top = T,
values = c("", "KNN","","RF","")
) %>%
set_header_labels(metric = "metric",
knn.train = "Train",
knn.test = "Test",
rf.train = "Train",
rf.test = "Test") %>%
merge_at(i = 1, j = 2:3, part = "header") %>%
merge_at(i = 1, j = 4:5, part = "header") %>%
flextable::align(align = "center", j = c(1:5), part = "all") %>%
set_caption("Table 7: Training and testing metric results")
```
The predicted versus actual plot in figure x shows how well the regression model predicts the BIS index on the test dataset. The red line represents a perfect regression model.
```{r echo=FALSE, message=FALSE, warning=FALSE, fig.asp = 0.6, fig.width=9, fig.cap="Figure 5: Predicted vs actual scatter plot"}
p3 <- ggplot(pred_knn, aes(x = BIS, y = knn_pred)) +
geom_point() +
geom_abline(intercept = 0,
slope = 1,
color = "red",
linewidth = 2)
p4 <- ggplot(pred_rf, aes(x = BIS, y = rf_pred)) +
geom_point() +
geom_abline(intercept = 0,
slope = 1,
color = "red",
linewidth = 2)
plot_grid(p3, p4, labels=c("KNN", "RF"), ncol = 2, nrow = 1)
```
**BIS vs new index**
The following plot shows how the new two index aligned to the previous BIS index for all the 5 cases.
```{r echo=FALSE, message=FALSE, warning=FALSE, fig.asp = 0.8, fig.width=9, fig.cap="Figure 6: new indexes vs BIS index on test cases"}
test_new_index_bycase <- pred_rf_bycase %>%
dplyr::rename(rf_pred = .pred) %>%
bind_cols(pred_knn_bycase %>% select(.pred) %>% dplyr::rename(knn_pred = .pred)) %>%
select(Sheet, BIS, knn_pred, rf_pred) %>%
dplyr::group_by(Sheet) %>%
dplyr::mutate(time = row_number()) %>%
pivot_longer(cols = c(BIS, knn_pred, rf_pred), names_to = "index") %>%
ggplot(aes(x=time, y=value, color = index)) +
geom_line() +
labs(x="Time, seconds", y = "index") +
theme(legend.position = "bottom") +
facet_wrap(~Sheet, ncol = 1)
test_new_index_bycase
```
**Pearson coefficient by test case**
In figure 7 we can see that the pearson coefficient is over 0.8 for all cases except test case 4. This might be due to some patterns not being covered in the training dataset.
```{r echo=FALSE, fig.cap="Pearson", message=FALSE, warning=FALSE, fig.asp = 0.6, fig.width=9, fig.cap="Figure 7: Final models pearson coefficient by test case"}
knn_pearson_per_case %>%
inner_join(rf_pearson_per_case, by = "case") %>%
dplyr::rename(knn = knn.pearson, rf = rf.pearson) %>%
pivot_longer(cols = c(knn, rf), names_to = "model") %>%
ggplot(aes(x = case, y = value, color = model, group = model)) +
geom_point() +
geom_line() +
labs(y = "Correlation coefficient", x = "Cases") +
ylim(0,1)
```
**Bland-Altman plots**
The Bland-Altman plot tells the level of agreement between the new BIS index and the previous one. The estimated bias and 95% confidence intervals for both the knn-based and rf-based index are:
KNN
* bias: 0.03
* confidence interval: [-20.18, 20.23]
RF
* bias: 0.38
* confidence interval: [-18.41, 19.18]
We can see that the differences have a normal distribution which indicates that 95% of these differences lie between the intervals (red dotted lines)
```{r echo=FALSE, message=FALSE, warning=FALSE, fig.asp = 0.6, fig.width=9, fig.cap="Figure 8: Bland-Altman plots and distributions"}
mean_diff <- mean(pred_rf$diff)
lower <- mean_diff - 2*sd(pred_rf$diff)
upper <- mean_diff + 2*sd(pred_rf$diff)
p2 <- ggplot(pred_rf, aes(x = avg, y = diff)) +
geom_point(size=2, color = "#3492eb") +
geom_hline(yintercept = mean_diff) +
geom_hline(yintercept = lower, color = "red", linetype="dashed", linewidth = 1) +
geom_hline(yintercept = upper, color = "red", linetype="dashed", linewidth = 1) +
# annotate("text", label = paste0("upper = ",round(upper,2)), x = 4, y = 20.5) +
# annotate("text", label = paste0("bias = ",round(mean_diff,2)), x = 4, y = 1.5) +
# annotate("text", label = paste0("lower = ",round(lower,2)), x = 5, y = -16) +
# xlim(0,100) +
# ggtitle("Bland-Altman Plot") +
ylab("Difference Between Measurements") +
xlab("Average Measurement")
mean_diff <- mean(pred_knn$diff)
lower <- mean_diff - 2*sd(pred_knn$diff)
upper <- mean_diff + 2*sd(pred_knn$diff)
p1 <- ggplot(pred_knn, aes(x = avg, y = diff)) +
geom_point(size=2, color = "#3492eb") +
geom_hline(yintercept = mean_diff) +
geom_hline(yintercept = lower, color = "red", linetype="dashed", linewidth = 1) +
geom_hline(yintercept = upper, color = "red", linetype="dashed", linewidth = 1) +
# annotate("text", label = paste0("upper = ",round(upper,2)), x = 5, y = 22) +
# annotate("text", label = paste0("bias = ",round(mean_diff,2)), x = 3, y = 1.5) +
# annotate("text", label = paste0("lower = ",round(lower,2)), x = 5, y = -18) +
# xlim(0,100) +
# ggtitle("Bland-Altman Plot") +
ylab("Difference Between Measurements") +
xlab("Average Measurement")
p1.1 <- ggMarginal(p1, type = "histogram", size=5)
p1.2 <- ggMarginal(p2, type = "histogram", size=5)
plot_grid(p1.1, p1.2, labels=c("KNN", "RF"), ncol = 2, nrow = 1)
```
## 2.7 Discussion
In this report, machine learning algorithms were implemented and studied to create a new BIS index. Firstly we run a screening process to select two suitable models from a pre-selection of four models. This approach gives us a unbiased criteria for selecting the two models to train further. One opportunity of improvement could be that we run the screening process with hyperparameters tunning at the same time instead of default values. The final results were evaluated by comparing the correlation and agreement with the BIS index provided in the test cases.
Our two indexes use a k-nearest neighbor and random forest model to make the predictions based on five EEG features. Both models predicted BIS indexes for five cases with different levels of correlation, however, most of them were highly correlated (>0.8). There is one exception, We can see that test-4 case shows the lowest pearson coefficient. We think this could be due to some patterns in the test dataset not being covered in the training dataset. We can see that there is a section in test-4 where BIS values are lower than the minimum values available in the training dataset.
```{r echo=FALSE, message=FALSE, warning=FALSE}
knn_pearson_per_case %>%
inner_join(rf_pearson_per_case, by = "case") %>%
dplyr::rename(knn = knn.pearson, rf = rf.pearson) %>%
flextable() %>%
set_caption("Table 8: Pearson Coefficient by Test case") %>%
autofit()
```
The challenges faced during the implementation of this report were training time during the screening and hyperparameter tunning processes. This was solved in a way by introducing parallel processing.
Another challenge was the organisation of the steps between pre-processing the data, feature selection, training, testing and calculating metrics. This was solve by using the Tidymodels framework which helped in managing all this in a structured way.
\pagebreak
## 3. References
1. Gu, Y.; Liang, Z.; Hagihira, S. Use of Multiple EEG Features and Artificial Neural Network to Monitor the Depth of Anesthesia. Sensors 2019, 19, 2499
2. Shalbaf, R.; Behnam, H.; Sleigh, J.W.; Steyn-Ross, A.; Voss, L.J. Monitoring the depth of anesthesia using entropy features and an artificial neural network. J. Neurosci. Methods 2013, 218, 17–24.
3. Peker, M.; Arslan, A.; Sen, B.; Celebi, F.V.; But, A. A novel hybrid method for determining the depth of anesthesia level: Combining ReliefF feature selection and random forest algorithm (ReliefF+RF). In Proceedings of the 2015 International Symposium on Innovations in Intelligent SysTems and Applications (INISTA), Madrid, Spain, 2–4 September 2015; pp. 1–8.
4. Shalbaf, A.; Saffar, M.; Sleigh, J.W.; Shalbaf, R. Monitoring the Depth of Anesthesia Using a New Adaptive Neurofuzzy System. IEEE J. Biomed. Heal. Inform. 2017, 22, 671–677.
5. Liang, Z.; Huang, C.; Li, Y.; Hight, D.F.; Voss, L.J.; Sleigh, J.W.; Li, X.; Bai, Y. Emergence EEG pattern classification in sevoflurane anesthesia. Physiol. Meas. 2018, 39, 045006.
6. Liu, Q.; Cai, J.; Fan, S.-Z.; Abbod, M.F.; Shieh, J.-S.; Kung, Y.; Lin, L. Spectrum Analysis of EEG Signals Using CNN to Model Patient’s Consciousness Level Based on Anesthesiologists’ Experience. IEEE Access 2019, 7, 53731–53742
7. Huang, Yi, Wen, Peng, Song, Bo and Li, Yan. 2022. "Real-Time Depth of Anaesthesia Assessment Based on Hybrid Statistical Features of EEG." Sensors. 22 (16).
\pagebreak
# 4. Appendix
## 4.1 Reading file
```{r eval=F}
library(tidymodels) # package for data manipulation
library(readxl) # package for reading excel files
# read data set
# accessing all the sheets from dataset
sheet = excel_sheets("Project data set 1 (for reports 1 and 3) .xlsx")
# applying sheet names to dataframe names. Training data is contained within the first 10 tabs
data_frame = lapply(setNames(sheet[1:10], sheet[1:10]),
function(x) read_excel("Project data set 1 (for reports 1 and 3) .xlsx", sheet=x))
# attaching all dataframes together
eeg_data_train = bind_rows(data_frame, .id="Sheet") %>% select(-Sheet)
# applying sheet names to dataframe names. Test data is contained in the last 5 tabs
data_frame = lapply(setNames(sheet[11:15], sheet[11:15]),
function(x) read_excel("Project data set 1 (for reports 1 and 3) .xlsx", sheet=x))
eeg_data_test = bind_rows(data_frame, .id="Sheet") %>% select(-Sheet)
```
## 4.1 Feature selection code
**Boruta**
```{r eval=F}
library(Boruta)
# Decide if a variable is important or not using Boruta
boruta_output <- Boruta(BIS ~ ., data=na.omit(eeg_data_train), doTrace=2) # perform Boruta search
boruta_signif <- names(boruta_output$finalDecision[boruta_output$finalDecision %in% c("Confirmed", "Tentative")]) # collect Confirmed and Tentative variables
print(boruta_signif) # significant variables
plot(boruta_output, cex.axis=.7, las=2, xlab="", main="Variable Importance") # plot variable importance
```
![RF-based feature importance](boruta_feature_importance.png)
**rPart**
```{r eval=F}
library(caret)
set.seed(100)
rPartMod <- train(BIS ~ ., data = eeg_data_train, method="rpart")
rpartImp <- varImp(rPartMod)
print(rpartImp)
plot(rpartImp, top = 8, main='Variable Importance') # plot variable importance
```
![DT-based feature importance](rpart_feature_importance.png)
**stepWise both ways**
```{r eval=F}
base.mod <- lm(BIS ~ 1 , data= eeg_data_train) # base intercept only model
all.mod <- lm(BIS ~ . , data= eeg_data_train) # full model with all predictors
stepMod <- stats::step(base.mod, scope = list(lower = base.mod, upper = all.mod), direction = "both", trace = 0, steps = 1000) # perform
#step-wise algorithm
shortlistedVars <- names(unlist(stepMod[[1]])) # get the shortlisted variable.
shortlistedVars <- shortlistedVars[!shortlistedVars %in% "(Intercept)"] # remove intercept
print(shortlistedVars)
```
## 4.1 Screening process code
```{r eval=F}
library(tidymodels)
# Defining the pre-processing steps
data_recipe <- recipe(BIS ~ ., data = eeg_data_train) %>%
# select columns that will be used
step_select(BIS, x2, x5, x6, x7, x8, skip = T) %>%
# apply log transformation function to features
step_log(x2, x6) %>%
# normalise all predictors
step_normalize(all_numeric_predictors())
# Defining the models for the screening
# Multi-adaptive Regression Splines
mars_reg_spec <-
mars() %>%
set_mode("regression") %>%
set_engine("earth")
# knn
knn_mod_spec <-
nearest_neighbor() %>%
# This model can be used for classification or regression, so set mode
set_mode("regression") %>%
set_engine("kknn")
# SVR
svr_mod_spec <-
svm_linear() %>%
set_engine("kernlab") %>%
set_mode("regression")
# ANN
nn_mod_spec <-
mlp() %>%
# This model can be used for classification or regression, so set mode
set_mode("regression") %>%
set_engine("nnet")
# RF
rf_mod_spec <-
rand_forest() %>%
set_engine("ranger") %>%
set_mode("regression")
# Define the cross validation strategy, 10 folds
set.seed(345)
folds <- vfold_cv(eeg_data_train, v = 10)
# Define the workflow to be applied to all models: preprocess inputs and train with 10-fold CV.
spot_check_wf <-
workflow_set(
preproc = list(none = data_recipe),
models = list(knn = knn_mod_spec, svr = svr_mod_spec, nn = nn_mod_spec, rf = rf_mod_spec, mars = mars_reg_spec)
)
# To speed things up, let's run them in parallel using all my cores
library(doParallel)
all_cores <- parallel::detectCores(logical = FALSE)
cl <- makePSOCKcluster(all_cores)
registerDoParallel(cl)
# Fitting process
start <- Sys.time()
set.seed(456)
spot_check_rs <-
spot_check_wf %>%
workflow_map("fit_resamples", resamples = folds, metrics = metric_set(rmse, rsq, mae, mape, smape))
end <- Sys.time()
print(end - start)
```
## 4.2 Random Forest hyperparameter tunning
```{r eval=F}
# Data transformation
data_recipe <- recipe(BIS ~ ., data = eeg_data_train) %>%
step_select(BIS, x2, x5, x6, x7, x8, skip = T) %>%
step_log(x2, x6) %>%
step_normalize(all_numeric_predictors())
# Defining the model Random forest
rf_mod_spec <-
rand_forest(
mtry = tune(),
trees = 500,
min_n = tune()
) %>%
set_mode("regression") %>%
set_engine("ranger")
set.seed(345)
folds <- vfold_cv(eeg_data_train, v = 5)
rf_grid <- grid_regular(
mtry(range = c(1, 5)),
min_n(range = c(2, 8)),
levels = 5
)
ctrl <- control_grid(verbose = FALSE, save_pred = TRUE)
library(doParallel)
all_cores <- parallel::detectCores(logical = FALSE)
cl <- makePSOCKcluster(all_cores)
registerDoParallel(cl)
start <- Sys.time()
set.seed(0239)
rf_tune <- tune_grid(rf_mod_spec, data_recipe, folds, metrics = metric_set(rmse, rsq, mae, mape, smape), grid = 20, control = ctrl)
end <- Sys.time()
print(end - start)
show_best(rf_tune, metric = "rmse")
rf_best <- rf_tune %>% select_best(metric = "rmse")
# mtry min_n .metric .estimator mean n std_err .config
# <int> <int> <chr> <chr> <dbl> <int> <dbl> <chr>
# 1 4 5 rmse standard 2.70 5 0.0356 Preprocessor1_Model17
# 2 4 9 rmse standard 2.76 5 0.0327 Preprocessor1_Model09
# 3 3 12 rmse standard 2.82 5 0.0311 Preprocessor1_Model06
# 4 2 6 rmse standard 2.83 5 0.0285 Preprocessor1_Model07
# 5 2 10 rmse standard 2.91 5 0.0309 Preprocessor1_Model03
```
## 4.3 KNN hyperparameter tunning
```{r eval=F}
data_recipe <- recipe(BIS ~ ., data = eeg_data_train) %>%
step_select(BIS, x2, x5, x6, x7, x8, skip = T) %>%
step_log(x2, x6) %>%
step_normalize(all_numeric_predictors())
# Defining the models
# knn
knn_mod_spec <-
nearest_neighbor(
neighbors = tune(),
weight_func = tune(),
dist_power = tune()
) %>%
set_mode("regression") %>%
set_engine("kknn")
set.seed(345)
folds <- vfold_cv(eeg_data_train, v = 10)
knn_grid <- grid_regular(
neighbors(),
weight_func(),
dist_power(),
levels = list(neighbors = 4, weight_func = 4, dist_power = 1)
)
ctrl <- control_grid(verbose = FALSE, save_pred = TRUE)
library(doParallel)
all_cores <- parallel::detectCores(logical = FALSE)
cl <- makePSOCKcluster(all_cores)
registerDoParallel(cl)
start <- Sys.time()
set.seed(0239)
knn_tune <- tune_grid(knn_mod_spec, data_recipe, folds, metrics = metric_set(rmse, rsq, mae, mape, smape), grid = knn_grid, control = ctrl)
end <- Sys.time()
print(end - start)
show_best(knn_tune, metric = "rmse")
# neighbors weight_func dist_power .metric .estimator mean n std_err .config
# <int> <chr> <dbl> <chr> <chr> <dbl> <int> <dbl> <chr>
# 1 7 biweight 1 rmse standard 2.47 10 0.0431 Preprocessor1_Model03
# 2 10 biweight 1 rmse standard 2.49 10 0.0424 Preprocessor1_Model04
# 3 7 triangular 1 rmse standard 2.49 10 0.0419 Preprocessor1_Model15
# 4 4 triangular 1 rmse standard 2.50 10 0.0440 Preprocessor1_Model14
# 5 4 epanechnikov 1 rmse standard 2.51 10 0.0445 Preprocessor1_Model06
```