Data Analysis and Evaluation Methods Comparison
Data Analysis and Evaluation Methods Comparison
Data Analysis and Evaluation Methods Comparison
Task1
Analysis in R
First, we will load the data into R, and show the summary of all variables:
teeth <- read.csv('teeth.csv', header = TRUE, row.names = 1)
summary(teeth)
##
##
##
##
##
##
##
##
##
##
##
##
##
##
TopInc
Min.
:0.000
1st Qu.:1.000
Median :2.000
Mean
:2.097
3rd Qu.:3.000
Max.
:3.000
TopPre
Min.
:0.000
1st Qu.:2.000
Median :3.000
Mean
:2.806
3rd Qu.:4.000
Max.
:4.000
BotInc
Min.
:1.000
1st Qu.:1.500
Median :3.000
Mean
:2.419
3rd Qu.:3.000
Max.
:4.000
BotPre
Min.
:0.000
1st Qu.:2.000
Median :3.000
Mean
:2.677
3rd Qu.:3.500
Max.
:4.000
TopCan
Min.
:0.0000
1st Qu.:0.5000
Median :1.0000
Mean
:0.7419
3rd Qu.:1.0000
Max.
:1.0000
TopMol
Min.
:1.000
1st Qu.:1.000
Median :3.000
Mean
:2.194
3rd Qu.:3.000
Max.
:3.000
BotCan
Min.
:0.0000
1st Qu.:0.0000
Median :1.0000
Mean
:0.6452
3rd Qu.:1.0000
Max.
:1.0000
BotMol
Min.
:1.000
1st Qu.:2.000
Median :3.000
Mean
:2.419
3rd Qu.:3.000
Max.
:3.000
In this dataset there is 31 observations - animals. As we can see, 8 numerical variables is present for each
animal. All of the variables have integer values. Range for each of the variables is easily calculated as:
sapply(teeth, max) - sapply(teeth, min)
## TopInc BotInc TopCan BotCan TopPre BotPre TopMol BotMol
##
3
3
1
1
4
4
2
2
Histograms of each attributes:
library(ggplot2)
library(gridExtra)
p1 <- qplot(teeth$TopInc, geom = "histogram")
p2 <- qplot(teeth$BotInc, geom = "histogram")
p3<- qplot(teeth$TopCan, geom = "histogram")
p4 <- qplot(teeth$BotCan, geom = "histogram")
p5 <- qplot(teeth$TopPre, geom = "histogram")
p6 <- qplot(teeth$BotCan, geom = "histogram")
p7 <- qplot(teeth$TopMol, geom = "histogram")
p8 <- qplot(teeth$BotCan, geom = "histogram")
grid.arrange(p1, p2, p3, p4, ncol =2)
10
10
count
15
count
15
0
0
teeth$TopInc
teeth$BotInc
20
20
count
count
15
15
10
10
5
5
0
0
0.00
0.25
0.50
0.75
1.00
0.00
teeth$TopCan
0.25
0.50
0.75
teeth$BotCan
1.00
12.5
20
10.0
7.5
count
count
15
5.0
10
5
2.5
0.0
0
0
0.00
teeth$TopPre
0.25
0.50
0.75
1.00
teeth$BotCan
20
15
count
count
15
10
10
5
0
1.0
1.5
2.0
2.5
3.0
0.00
teeth$TopMol
0.25
0.50
0.75
1.00
teeth$BotCan
Histograms provide us a nice overview of the distributions of values inside of the variables.
In order to create clusters of observations, we used the function hclust in R. First we created a distance
matrix for the observations. The distances are calculated as Eucleadian distances. Distance matrix is then
used for clustering - the hclust algorithm uses the complete linkage method.
d <- dist(as.matrix(teeth))
hc <-hclust(d)
plot(hc)
HOUSE MOUSE
BEAVER
GROUNDHOG
GRAY SQUIRREL
PORCUPINE
PIKA
RABBIT
RED BAT
PIGMY BAT
HOUSE BAT
ELEPHANT SEAL
FUR SEAL
SEA LION
JAGUAR
COUGAR
RIVER OTTER
MARTEN
WOLVERINE
GREY SEAL
SEA OTTER
WEASEL
BADGER
DEER
REINDEER
ELK
RACCOON
WOLF
BEAR
MOLE
BROWN BAT
SILVER HAIR BAT
Height
Cluster Dendrogram
d
hclust (*, "complete")
Clustering in KNIME
Similar to what we presented in the previous part, where hierarchical clustering has been done in R, we will
show how we created clusters and dendogram in KNIME.
The workflow for hierarchical clustering in KNIME is quite simple - we have a node to read the data, one
node for clustering, and additional node to write the clustering results to .csv file.
The CSV Reader node was set up to read the teeth.csv file, with , as a delimiter and to count on column
and and row headers as shown on the screenshot below.
The Hierarchical Clustering node was set up to produce 3 clusters, using the Euclidean distance and complete
linkage, same as in the R analysis. Configuration window is shown below.
The third node was simply configured with the path to write the output results.
After running the program, in order to see the dendogram, we can right-click on the Hierarchical Clustering
node and choose View: Dendogram/Distance View which gives us the produced dendoogram shown below.
We can notice it is the same as the one we obtained in R.
Resusbstitution error In this part, we will use the whole dataset to train and test the model. This kind
of training and prediction is not usually good as the test data is already known to the model, hence accuracy
is not showing the true state.
library(rpart)
#Train and predict on the whole set with decision trees,
#calculate the accuracy - repeated 20 times
acc_resub = rep(NA,20)
for (i in 1:20){
wine_model_resub = rpart(class~., data=wine) #form the model
class_predict_resub = predict(wine_model_resub, wine, type = "class") #make the predictions
tab_resub <-table(class_predict_resub, wine$class) #table of true end predicted classes
acc_resub[i] = sum(diag(tab_resub))/sum(tab_resub) #itteration accuracy
}
wine_acc_resub_mean <- (mean(acc_resub)) #mean accuracy
wine_acc_resub_sd <- (sd(acc_resub)) #standard devation of the accuracy
Hold-out-10% Next, we will split our data in the way that 90% is used for the training and 10% for the
testing. This way, we are presenting our model with the new data, and the accuracy will better show how the
model is behaving with the unknown data.
library(caret)
## Loading required package: lattice
#Train and predict on the 90% of the set with decision trees,
#calculate the accuracy - repeated 20 times
acc_HO = rep(NA,20)
for (i in 1:20){
#create partition of 90% of the data based on the class
train_index_HO <- createDataPartition(wine$class, p = .9, list = FALSE)
}
wine_acc_HO_mean <- mean(acc_HO)
wine_acc_HO_sd <- sd(acc_HO)
10-fold-cross-validation In the third case, we will use cross validation, with 10 folds. This means we
will split our data in 10 parts, train the model on 9 of them and test on the one that has not been used in
training. This will be repeated 10 times, every time leaving different part of the data for testing and training
on the rest of the set.
#Preparing the CV - spliting data into blocks
n <- nrow(wine)
K <- 10 #number of folds
n_block <- n%/%K+1
acc_CV10 <- rep(NA, 20)
for (i in 1:20){
#creating new blocks
rand = runif(n)
rang <- rank(rand)
block <- (rang-1)%/%n_block
block<- as.factor(block)
acc <- numeric(0)
for (k in 1:K-1){
#training and prediction
model_k <- rpart(class~., data = wine[block!=k,])
pred_k <- predict(model_k, wine[block==k,], type = "class")
tab_k <- table(pred_k, wine[block==k,]$class)
acc_k <- sum(diag(tab_k))/sum(tab_k)
acc <- rbind(acc, acc_k)
}
#calculating CV10 one itteration accuracy
acc_CV10[i] <- mean(acc)
}
wine_acc_CV10_mean <- (mean(acc_CV10))
wine_acc_CV10_sd <- (sd(acc_CV10))
for (k in 1:n){
#train and predict with excluding one by one of row from training and predicting for it
model_k <- rpart(class~., data = wine[-k,])
pred_k[k] <- predict(model_k, wine[k,], type = "class")
}
tab_k <- table(pred_k, wine$class)
acc_LOOCV[i] <- sum(diag(tab_k))/sum(tab_k)
}
wine_acc_LOOCV_mean <- mean(acc_LOOCV)
wine_acc_LOOCV_sd <- sd(acc_LOOCV)
Comparison of the evaluation methods Now we can compare all four methods. In the table below, we
can see that the accuracy is highest for the Resubstitution error method, which was expected since it was
measured on the whole training set. LOOCV method has the lowest accuracy, which is probably closest to
the real situation since it was tested separately on all cases present in the dataset.
wine_means <- cbind(wine_acc_resub_mean, wine_acc_HO_mean
, wine_acc_CV10_mean, wine_acc_LOOCV_mean)
wine_sds <- cbind(wine_acc_resub_sd, wine_acc_HO_sd
, wine_acc_CV10_sd, wine_acc_LOOCV_sd)
wine_stats <-rbind(wine_means, wine_sds)
colnames(wine_stats) <- c("Resub", "Hold_Out_10", "10_fold_CV", "LOOCV")
row.names(wine_stats) <- c("Mean", "Std_dev")
wine_stats
##
Resub Hold_Out_10 10_fold_CV
LOOCV
## Mean
0.9378531 0.86875000 0.87505556 0.8305085
## Std_dev 0.0000000 0.04488655 0.01903956 0.0000000
Cancer data
In this part we will perform the same as for the wine data. It will not be explain again, since everything
holds as in that data.
Importing and preparing the data
cancer = read.csv("breast-cancer-wisconsin.data.txt")
names(cancer) <- c("Sample_number","Clump_Thickness", "Uniformity_of_Cell_Size"
, "Uniformity_of_Cell_Shape","Marginal_Adhesion"
, "Single_Epithelial_Cell_Size", "Bare_Nuclei"
, "Bland_Chromatin","Normal_Nucleoli","Mitoses", "class")
cancer$class <- factor(cancer$class)
Resubstitution error Similar to the wine data, we will train and test on the whole dataset to obtain the
resubstitution error accuracy.
#Train and predict on the whole set with decision trees,
#calculate the accuracy - repeated 20 times
acc_resub = rep(NA,20)
9
for (i in 1:20){
cancer_model_resub = rpart(class~., data=cancer) #form the model
class_predict_resub = predict(cancer_model_resub, cancer, type = "class") #make the predictions
tab_resub <-table(class_predict_resub, cancer$class) #table of true end predicted classes
acc_resub[i] = sum(diag(tab_resub))/sum(tab_resub) #itteration accuracy
}
cancer_acc_resub_mean <- (mean(acc_resub)) #mean accuracy
cancer_acc_resub_sd <- (sd(acc_resub)) #standard devation of the accuracy
Hold-out-10% Again, similar to the wine case, we will split the data for train (90%) and test (10%) at
random, 20 times and take the average.
#Train and predict on the 90% of the set with decision trees,
#calculate the accuracy - repeated 20 times
acc_HO = rep(NA,20)
for (i in 1:20){
#create partition of 90% of the data based on the class
train_index_HO <- createDataPartition(cancer$class, p = .9, list = FALSE)
#split the data on the train and test part
cancer_train_HO <- cancer[train_index_HO,]
cancer_test_HO <- cancer[-train_index_HO,]
cancer_model_HO <- rpart(class~., data = cancer_train_HO)
class_predict_HO = predict(cancer_model_HO, cancer_test_HO, type="class")
tab_HO <- table(class_predict_HO, cancer_test_HO$class)
acc_HO[i] <- sum(diag(tab_HO))/sum(tab_HO)
}
cancer_acc_HO_mean <- mean(acc_HO)
cancer_acc_HO_sd <- sd(acc_HO)
10-fold-cross-validation
Leave-one-out-cross-validation
pred_k <-rep(NA,n)
acc_LOOCV <- rep(NA,20)
for (i in 1:20){
for (k in 1:n){
#train and predict with excluding one by one of row from training and predicting for it
model_k <- rpart(class~., data = cancer[-k,])
pred_k[k] <- predict(model_k, cancer[k,], type = "class")
}
tab_k <- table(pred_k, cancer$class)
acc_LOOCV[i] <- sum(diag(tab_k))/sum(tab_k)
}
cancer_acc_LOOCV_mean <- mean(acc_LOOCV)
cancer_acc_LOOCV_sd <- sd(acc_LOOCV)
Comparison of the evaluation methods Again, here we have very optimistic results for the resubstitution error, while LOOCV again has the lowest accuracy of all methods. In both cases we can notice
Resubstitution and LOOCV have 0 standard devation, which is expected, since in both methods the whole
set was used both times, and there is no randomness. In other two methods, train and test parts are chosen
randomly every times, so results obtained are every time different.
cancer_means <- cbind(cancer_acc_resub_mean, cancer_acc_HO_mean, cancer_acc_CV10_mean
, cancer_acc_LOOCV_mean)
cancer_sds <- cbind(cancer_acc_resub_sd, cancer_acc_HO_sd
, cancer_acc_CV10_sd, cancer_acc_LOOCV_sd)
cancer_stats <-rbind(cancer_means, cancer_sds)
colnames(cancer_stats) <- c("Resub", "Hold_Out_10", "10_fold_CV", "LOOCV")
row.names(cancer_stats) <- c("Mean", "Std_dev")
cancer_stats
##
Resub Hold_Out_10 10_fold_CV
LOOCV
## Mean
0.965616 0.93260870 0.93586975 0.9111748
## Std_dev 0.000000 0.03093197 0.00484623 0.0000000
11