<- read.csv("../../../data/01-modified-data/horses.clean.csv")
horses <- read.csv("../../../data/01-modified-data/perf.clean.csv") perf
Exploratory Data Analysis
In this section, we will be performing an exploratory data analysis of our horses and performances data sets. This analysis will include looking at univariate, bivariate, and multivariate summary statistics and visualizations of the data, analyzing the relationships between the features in our data sets, analyzing parsimonious models that can be developed with the data, and beginning to develop some of the foundational knowledge and early conjecture pertaining to next steps in developing more advanced classifiers and predictive models.
All analysis will be performed in R, with the ggplot2 package serving as a primary package in this section
Reading in our data.
library(ggplot2)
library(dplyr)
Attaching package: 'dplyr'
The following objects are masked from 'package:stats':
filter, lag
The following objects are masked from 'package:base':
intersect, setdiff, setequal, union
Horse Rating versus Race Placement
Let’s first take a look at one of our biggest overarching questions: what attributes of a horse have the most impact on their outcome in a race. Due to the fact that the performance data set contains information on races of 25 DIFFERENT TRACK LENGTHS, and the length of a race is likely very much so NOT independent of a horse’s attributes, the following data visualizations will be grouped by these race lengths. Because looking at 25 different track lengths on a singular visualization would be too messy to glean any information from, I decided to break the data into 3 distinct groups by creating a new variable in our data called “race_length”. The groups will be races below 1600m (1 mile), races between 1600m and 2000m, and races over 2000m.
for(i in 1:nrow(perf)){
if(perf$distance[i] < 1600){
$race_length[i] <- "Less than 1600m"
perfelse if(perf$distance[i] >= 1600 & perf$distance[i] <= 2000){
} $race_length[i] <- "1600m - 2000m"
perfelse if(perf$distance[i] > 2000){
} $race_length[i] <- "More than 2000m"
perf
}
}
Let’s take look at the distribution of horse race lengths.
table(perf$race_length)
1600m - 2000m Less than 1600m More than 2000m
108922 157310 7761
We can see that the majority of races were less than 1600m, followed relatively closely by races between 1600m and 2000m, and then lastly, races more than 2000m. The distribution of the data is fairly evenly distributed across race lengths, all things considered. We do not need to treat one of the race length groups as an outlier to the data.
Now we will start to build visualizations.
<- ggplot(perf,aes(x=rating,y=final_placing))+geom_jitter()
g1 g1
Warning: Removed 7615 rows containing missing values (`geom_point()`).
Upon the initial creation of this plot, we can see that we have some errors in our data where there are final placings of 47 and 99. We are going to delete these points. We also have anomalies where ratings are 333 and 951, which we will also remove.
<- perf%>%
perffilter(rating < 250 & final_placing != 47 & final_placing != 99)
Now we re-create our plot, along with cleaned formatting.
# Re-ordering factor levels for legend
$race_length <- factor(perf$race_length, levels = c("More than 2000m", "1600m - 2000m", "Less than 1600m"))
perf<- ggplot(perf,aes(x=rating,y=final_placing,color=race_length))+
g1geom_point()+
ggtitle("Horse Rating vs. Final Race Placing")+
xlab("Horse Rating")+
ylab("Final Placing")+
guides(color = guide_legend(title = "Race Length"))+
scale_y_continuous(breaks=seq(1,14,by=1))
g1
As we can see above, there is a fairly even distribution of placements when we are looking at horse rankings for each race length. This plot helps us understand that historically, horses of all rankings have come in last place, first place, and everywhere in between. We can see, however, that there might be a small pattern happening towards the end of the x-axis: higher ranking horses look like they do not place low frequently and vice versa for lower ranked horses. In order to examine this potential pattern, lets only looking at horses with ranking 125 and up.
<- perf %>%
perf.rating.filter filter(rating >= 125)
<- ggplot(perf.rating.filter,aes(x=rating,y=final_placing,color=race_length))+
g2geom_jitter()+
ggtitle("Horse Rating vs. Final Race Placing")+
xlab("Horse Rating")+
ylab("Final Placing")+
guides(color = guide_legend(title = "Race Length"))+
scale_y_continuous(breaks=seq(1,14,by=1))
g2
Here, we are able to get a better gauge on how rating impacts final placement in a race. We can see that if a horse is above about a 135 rating, it becomes exceedingly rare for them to place below 10th. As we move on to a rating of 140, it becomes rare for the horse to place below 5th. These phenomena apply to races of all lengths (with races between 1600m and 2000m being the most common race). This knowledge becomes helpful when we eventually use horse rating to help determine the probability a horse has a podium finish (placement in the top 3).
Most horses, however, do not exceed ratings of 130 at the HJKC. Even still, the density of the data points is far higher at better placements for horses between 125 and 130. Horse rating certainly plays a role in determining if it wins (especially since rating is a metric derived from a horse’s past performance and attributes).
Starting Draw versus Final Race Placement
For every horse race, horses are given a starting position in the starting gates based on a random draw. There is often debate over what draw is best for a horse given the different lengths of races. I will be creating a plot to explore this debate and visualize how starting draw has affected final race placement among horses at HKJC.
<- ggplot(perf.rating.filter,aes(x=draw,y=final_placing,color=race_length))+
g3geom_count(aes(size = stat(prop)))+
ggtitle("Starting Draw vs. Final Race Placing")+
xlab("Draw")+
ylab("Final Placing")+
guides(color = guide_legend(title = "Race Length"))+
scale_y_continuous(breaks=seq(1,14,by=1))+
scale_x_continuous(breaks=seq(1,14,by=1))
g3
Warning: `stat(prop)` was deprecated in ggplot2 3.4.0.
ℹ Please use `after_stat(prop)` instead.
In the plot above, we can notice a couple things. For one, ALL winners (and even top 2 finishers) in races more than 2000m, started in gates 1 through 7, the inside lanes. This phenomena shows a clear advantage for the inside lanes in longer races. For races less than 1600m, this starting draw seems to not make as much of a difference, however the very outside lanes (lanes 10-14), do show less podium finishes than those horses in more inward lanes. These findings help inform the general relationship between draw and race placment, especially in terms of how we can expect to weight draw when we build models to predict a horse’s racing performance over different track lengths.
Summarry of Finshing Times for All Race Distances
In order to get an idea of how each race distance in the performance data set is distributed in terms of finishing time (in seconds), I have produced the table down below. This table will set the stage for interpreting the results in subsequent plots.s
# Removing observations where finish time = 0
<- perf %>%
perf.no.zero filter(finish_time != 0)
tapply(perf.no.zero$finish_time, perf.no.zero$distance, summary)
$`1000`
Min. 1st Qu. Median Mean 3rd Qu. Max.
23.72 57.32 57.84 57.85 58.35 72.54
$`1200`
Min. 1st Qu. Median Mean 3rd Qu. Max.
67.40 70.06 70.64 70.68 71.21 108.63
$`1400`
Min. 1st Qu. Median Mean 3rd Qu. Max.
38.86 82.77 83.40 83.49 84.05 135.70
$`1600`
Min. 1st Qu. Median Mean 3rd Qu. Max.
28.21 95.61 96.35 96.40 97.10 119.12
$`1650`
Min. 1st Qu. Median Mean 3rd Qu. Max.
96.34 100.61 101.37 101.42 102.13 147.75
$`1800`
Min. 1st Qu. Median Mean 3rd Qu. Max.
15.23 109.20 110.24 110.30 111.27 136.12
$`2000`
Min. 1st Qu. Median Mean 3rd Qu. Max.
120.0 123.0 124.0 124.2 125.1 148.6
$`2200`
Min. 1st Qu. Median Mean 3rd Qu. Max.
132.4 137.5 138.6 138.8 139.9 153.4
$`2400`
Min. 1st Qu. Median Mean 3rd Qu. Max.
145.5 148.1 149.3 149.9 150.8 163.6
The Effect of the Type of Track (Turf vs Dirt) on Finishing Time
Another topic of interest is how the type of track, turf versus dirt (labeled as All Weather Track, aka AWT, in the data), might effect the finishing times of times of horses. The Sha Tin Racecourse contains both dirt and turf tracks, while Happy Valley only contains turf tracks. Analyzing track differences could more optimally predict horse performance for each respective racing grounds. We will take a look at this question in the box plots below.
<- ggplot(perf.no.zero,aes(x=track,y=finish_time,color=as.factor(distance)))+
g4 geom_boxplot()+
ggtitle("Type of Track vs. Finishing Time")+
xlab("Type of Track")+
ylab("Finishing Time (s)")+
guides(color = guide_legend(title = "Race Length (m)"))
g4
Upon first glance, we can see that any race on an AWT track was either 1200m, 1650m, or 1800m, while races of all distances (where a finishing time was recorded) happened on turf tracks. There does not appear to be any significant difference in median finishing times for the 1200, 1650, and 1800 meter races on AWT tracks versus Turf tracks, which is actually quite suprising. The spread of times on turf tracks, however, does appear to be higher.
One can only wonder how the conditions of the tracks might effect finishing time, especially with the influences of weather, seasonality, and how many races the track has already seen before a race. To analyze the effect of track condition, along with the type of track, a grid of plots can be found below.
<- ggplot(perf.no.zero,aes(x=track,y=finish_time,color=race_length))+
g5 geom_violin()+
ggtitle("Type of Track and Track Condition vs. Finishing Time")+
xlab("Type of Track")+
ylab("Finishing Time (s)")+
facet_wrap(vars(going))+
guides(color = guide_legend(title = "Race Length (m)"))
g5
As a reminder, the meanings behind each acronym for ‘going’ can be found below (note that WF means Wet Fast and WS means Wet Slow): While it may look like there is a lot going on in the above plot, overall, it does not appear that there are any significant differences in median finishing times when turf and AWT tracks have any different going ratings. This finding is VERY suprising. This observation is not to say that track going has NO impact on finishing time, but these differences might be very marginal and require more detailed hypothesis testing and analysis to make further inference. It SHOULD be noted, however, that when an AWT track is rated as fast, good, slow, or wet, there seems to be very little spread in finishing times. In other words, there is far less variability in how fast all the horses run (races might be closer in nature).
The Effect of Horse Weight on Race Performance
library(ggpubr)
<- ggplot(perf.no.zero,aes(x=on_date_weight,y=finish_time))+
g6 geom_point()+
ggtitle("Horse Weight vs. Finishing Time")+
xlab("Horse Weight (lbs)")+
ylab("Finishing Time (s)")+
facet_wrap(vars(distance))+
stat_cor(method="pearson",label.x = 200, label.y = 30)
g6
Warning: Removed 2 rows containing non-finite values (`stat_cor()`).
Warning: Removed 2 rows containing missing values (`geom_point()`).
The results from this plot are probably the most surprising out of any of the EDA results thus far. While the significance of a Horse’s weight is a popular topic of debate among horse racing handicappers, it is a common trope that the heavier horse usually wins. This trope is actually a phenomena I thought this data visualization would display, but clearly, no such phenomena is evident. In fact, all of our correlation coefficients are negative. It should be said that these correlation coefficients all convey very weak relationships. All of our coefficients, except for races above 2000m, are statistically significant, too.
With this finding, let’s look if added weight to a horse (i.e. jockey weight, gear, etc.) has any signifcant effect on finish time.
<- ggplot(perf.no.zero,aes(x=actual_weight,y=finish_time))+
g7 geom_point()+
ggtitle("Horse Weight vs. Finishing Time")+
xlab("Added Weight (lbs)")+
ylab("Finishing Time (s)")+
facet_wrap(vars(distance))+
stat_cor(method="pearson",label.x = 100, label.y = 30)
g7
The same findings hold true for added weight to the horse as the weight of the horse itself. Added weight is very weakly correlated with finishing time across all race distances. Again, this result is very surprising.
The Effect of Horse Sex and Import Type on Podium Finishes
$podium <- horses$first_place + horses$second_place + horses$third_place horses
<- ggplot(horses,aes(x=import_type,y=podium,color=sex))+
g8 geom_bar(stat = 'identity')+
ggtitle("Horse Import Type and Sex vs. Number of Podium Finishes")+
xlab("Import Type")+
ylab("Podium Finishes")
g8
The most common types of horses are privately purchased (PP) horses and privately purchased griffins (PPG). Horses from Singapore (SG) also make up a large portion of the data. International Sale Griffins (ISG) are rarer, but still make up a portion of the data. We can see that most (almost entirely) of the horses are Geldings which are castrated make horses. This graphic is able to paint the picture of what horse attributes are most common in our dataset.
For more information regarding horse import types, see the HKJC website.
The Effect of a Horse’s Past Performance on Race Performance
Now I am going to check see if there is any relationship between a horses’s average past placement before a race and the place the horse ends up finishing in its race.
library(tidyr)
# We only want to look at horses that have run a race before, so let's filter rows where average_placing is NA
<- perf %>% drop_na(c(final_placing, average_placement))
perf
# We also know the average_placement variable is going to have outliers since our final_placing data had outliers. Let's fix this issue.
<- perf %>%
perf filter(average_placement < 15)
<-ggplot(perf,aes(x=average_placement, y=final_placing))+
g9geom_count()+
ggtitle("Average Placement Before Race vs. Final Race Placing")+
xlab("Average Placement")+
ylab("Final Placing")+
scale_y_continuous(breaks=seq(1,14,by=1))+
scale_x_continuous(breaks=seq(1,14,by=1))
g9
We can see something very compelling this plot: a horse’s average placement before its races appears to be very determinant in what place it will finish in a race. A horse will most likely finish 1st if it has a historic placement within 1 to 7. If a horse historically has finished closer to the middle of the pack, it is most likely to continue to finish in the mid pack. The same inference can be applied to horses with poor placements.
Days of Rest Between Races
<-ggplot(perf,aes(x=average_placement, y=days_between))+
g10geom_count()+
ggtitle("Average Placement Before Race vs. Final Race Placing")+
xlab("Days of Rest Between Races")+
ylab("Final Placing")
g10
We can see that most of our data is between 0 and around 15 days of rest. Let’s just look at data over that interval.
<- perf[perf$days_between<=15,]
perf_rest <-ggplot(perf_rest,aes(x=average_placement, y=days_between))+
g11geom_count()+
ggtitle("Average Placement Before Race vs. Final Race Placing")+
xlab("Days of Rest Between Races")+
ylab("Final Placing")
g11
One of the largest observations we can take away from this visualization is that horses that are placing on the podium (top 3) are mostly getting at least 4+ days of rest. A well rested horse does not mean that it will win, but if a horse placed highly, it does appear it was likely well rested horses.
Exporting data without outliers/discovered errors
write.csv(perf,"../../../data/01-modified-data/perf.clean.no.error.csv",row.names = FALSE)
Next Steps
With this exploratory analysis, we are going to carry these findings right into our Naive Bayes Classification section. Findings from this section will be used and referenced throughout the rest of this project.