-
Notifications
You must be signed in to change notification settings - Fork 370
/
Copy pathParkingTicketsTo.Rmd
262 lines (190 loc) · 10.4 KB
/
ParkingTicketsTo.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
---
output:
html_document:
smart: false
---
Revealing Toronto's Parking Ticket Data
=====================================================================
By Susan Li
March 13 2017
Don't want to be nailed? The City of Toronto publishes their parking tickets data every year. Below is one year parking tickets data to show you when and where most tickets are issued.
I will be using R and some of its extended libraries for this project.
```{r global_options, include=FALSE}
knitr::opts_chunk$set(echo=FALSE, warning=FALSE, message=FALSE)
```
```{r}
data1 <- read.csv('Parking_Tags_Data_2016_1.csv', stringsAsFactors = F)
data2 <- read.csv('Parking_Tags_Data_2016_2.csv', stringsAsFactors = F)
data3 <- read.csv('Parking_Tags_Data_2016_3.csv', stringsAsFactors = F)
data4 <- read.csv('Parking_Tags_Data_2016_4.csv', stringsAsFactors = F)
parking_df <- rbind(data1, data2, data3, data4)
options("scipen"=100, "digits"=4)
```
```{r}
dim(parking_df)
```
```{r}
parking_df$date_of_infraction <- as.Date(as.character(parking_df$date_of_infraction), "%Y%m%d")
```
```{r}
summary(parking_df$date_of_infraction)
summary(parking_df$set_fine_amount)
```
The dataset can be accessed through [City of Toronto Open Data Portal](http://www1.toronto.ca/wps/portal/contentonly?vgnextoid=ca20256c54ea4310VgnVCM1000003dd60f89RCRD&vgnextchannel=7807e03bb8d1e310VgnVCM10000071d60f89RCRD). It contains 2256761 parking tickets, from January 1 2016 to December 31 2016, the amount from 0 to $450. I will omit missing values in the data, that means 1331 rows are omitted. Also I will need to do some cleaning such as change the date and time format to the appropriate format. I also added a new column - day of week.
```{r}
parking_df$time_of_infraction <- sprintf("%04d", parking_df$time_of_infraction)
parking_df$time_of_infraction <- format(strptime(parking_df$time_of_infraction, format="%H%M"), format = "%H:%M")
```
```{r}
parking_df <- parking_df[complete.cases(parking_df[,-1]),]
```
```{r}
library(ggplot2)
ggplot(aes(x = date_of_infraction), data = parking_df) + geom_histogram(bins = 48, color = 'black', fill = 'gold') +
ggtitle('Histogram of Infraction Date')
```
The number of parking tickets distributed almost evenly throughout the year. It seems slowing down towards the end of the year.
```{r}
parking_df$time_of_infraction <- as.POSIXlt(parking_df$time_of_infraction, format="%H:%M")$hour
```
```{r}
ggplot(aes(x = time_of_infraction), data = parking_df) + geom_histogram(bins = 24, color = 'black', fill = 'blue') +
ggtitle('Histogram of Infraction Time')
```
The time distribution appears bimodal with period peaking around 9am to 1pm and again after midnight. The safest time to park your car without being nailed is around 5 and 6am.
```{r}
sort(table(parking_df$set_fine_amount))
ggplot(aes(x = set_fine_amount), data = parking_df) + geom_histogram(bins = 100, color = 'black', fill = 'red') +
ggtitle('Histogram of Fine Amount')
```
The most common amount is $30, then $50, then $40. I believe the "0" value means the parking tickets were cancelled.
So which dates have the most number of infractions? and which dates have the least number of infractions?
```{r}
library(dplyr)
parking_df_day <- dplyr::summarise(date_group, count = n(),
total_day = sum(set_fine_amount),
na.rm = TRUE)
parking_df_day[order(parking_df_day$count),]
```
April 1st has the most number of infractions, and December 25 has the least number of parking tickets issued. I wonder why.
```{r}
parking_df$day_of_week <- weekdays(as.Date(parking_df$date_of_infraction))
```
```{r}
library(dplyr)
weekday_group <- group_by(parking_df, day_of_week)
parking_df_weekday <- dplyr::summarise(weekday_group, count = n(),
total_day = sum(set_fine_amount))
parking_df_weekday$day_of_week <- ordered(parking_df_weekday$day_of_week, levels = c('Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday', 'Sunday'))
ggplot(aes(x = day_of_week, y = count), data = parking_df_weekday) +
geom_bar(stat = 'identity') +
ylab('Number of Infractions') +
ggtitle('Infractions Day of Week')
```
Apparently, less infractions happened in the weekend than during the weekdays.
Now let's look at what the infractions are. Because there are more than 200 different infractions, it makes sense to only look at the top 10.
```{r}
library(dplyr)
infraction_group <- group_by(parking_df, infraction_description, infraction_code)
parking_df_infr <- dplyr::summarise(infraction_group, count = n())
parking_df_infr <- head(parking_df_infr[order(parking_df_infr$count, decreasing = TRUE),], n = 10)
parking_df_infr
```
```{r}
library(ggthemes)
ggplot(aes(x = reorder(infraction_description, count), y = count), data = parking_df_infr) +
geom_bar(stat = 'identity') +
theme_tufte() +
theme(axis.text = element_text(size = 10, face = 'bold')) +
coord_flip() +
xlab('') +
ylab('Total Number of Infractions') +
ggtitle("Top 10 Infractions") +
theme_fivethirtyeight()
```
Now, what are the top 10 locations that have the most infractions?
```{r}
library(dplyr)
location_group <- group_by(parking_df, location2)
parking_df_lo <- dplyr::summarise(location_group, total = sum(set_fine_amount),
count = n())
parking_df_lo <- head(parking_df_lo[order(parking_df_lo$count, decreasing = TRUE), ], n=10)
ggplot(aes(x = reorder(location2, count), y = count), data = parking_df_lo) +
geom_bar(stat = 'identity') +
theme_tufte() +
theme(axis.text = element_text(size = 10, face = 'bold')) +
coord_flip() +
xlab('') +
ylab('Total Number of Infractions') +
ggtitle("Top 10 Locations") +
theme_fivethirtyeight()
```
Here it is! Be aware when you are around thoes areas.
How about the trend? Is there any infraction type increase or decrease over time?
```{r}
parking_df_infr_1 <- parking_df %>%
filter(infraction_description %in% parking_df_infr$infraction_description)
```
```{r}
library(dplyr)
date_in_group <- group_by(parking_df_infr_1, infraction_description, date_of_infraction)
parking_df_infr_1 <- dplyr::summarise(date_in_group, total =
sum(set_fine_amount),
count = n())
```
```{r}
ggplot(aes(x = date_of_infraction, y = count, color = infraction_description), data = parking_df_infr_1) +
geom_jitter(alpha = 0.05) +
geom_smooth(method = 'loess') +
xlab('Date') +
ylab('Number of Infractions') +
ggtitle('Time Series of the Top Infractions') +
scale_y_log10() +
theme_solarized()
```
This is not a good looking graph. Most of the top infractions have been steady over time. only "PARK FAIL TO DISPLAY RECEIPT" had dropped since fall, and "PARK MACHINE-REQD FEE NOT PAID" had an increase since October. Does it have anything to do with the season? or weather? Or simply more park machines broken toward the end of the year?
```{r echo=FALSE, warning=FALSE, message=FALSE}
week_time_group <- group_by(parking_df, day_of_week, time_of_infraction)
parking_df_week_time <- dplyr::summarise(week_time_group, time_sum =
sum(set_fine_amount),
count = n())
```
```{r}
parking_df_week_time$day_of_week <- ordered(parking_df_week_time$day_of_week, levels = c('Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday', 'Sunday'))
ggplot(aes(x = time_of_infraction, y = count, color = day_of_week), data = parking_df_week_time) +
geom_line(size = 2.5, alpha = 0.7) +
geom_point(size = 0.5) +
xlab('Hour(24 hour clock)') +
ylab('Number of Infractions') +
ggtitle('Infractions Time of the Day') +
theme_economist()
```
This is a much better looking graph. the highest counts are around noon time during the weekday, this trend changed during the weekend.
Now let me drill down to the top 10 infractions.
```{r}
top_10 <- c('PARK-SIGNED HWY-PROHIBIT DY/TM', 'PARK ON PRIVATE PROPERTY', 'PARK PROHIBITED TIME NO PERMIT', 'PARK FAIL TO DISPLAY RECEIPT', 'PARK MACHINE-REQD FEE NOT PAID', 'PARK - LONGER THAN 3 HOURS ', 'STOP-SIGNED HWY-PROHIBIT TM/DY', 'STAND VEH.-PROHIBIT TIME/DAY', 'STOP-SIGNED HIGHWAY-RUSH HOUR', 'PARK-VEH. W/O VALID ONT PLATE')
top_10
```
```{r}
parking_df_top_10 <- parking_df %>%
filter(infraction_description %in% top_10)
top_10_groups <- group_by(parking_df_top_10, infraction_description, day_of_week, time_of_infraction)
parking_df_top_10 <- dplyr::summarise(top_10_groups, total =
sum(set_fine_amount),
count = n())
```
```{r}
parking_df_top_10$day_of_week <- ordered(parking_df_top_10$day_of_week, levels = c('Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday', 'Sunday'))
ggplot(aes(x = time_of_infraction, y = count, color = day_of_week), data = parking_df_top_10) +
geom_line(size = 1.5) +
geom_point(size = 0.5) +
xlab('Hour(24 hour clock)') +
ylab('Number of Infractions') +
ggtitle('(log_10)Infractions Time of the Day') +
scale_y_log10() +
facet_wrap(~infraction_description)
```
I found two sharp curved infractions interesting, one is "STOP-SIGNED HIGHWAY RUSH-HOUR", there are two peak infraction hours around 8am and 4pm during the weekdays, the weekend is very quiet. It makes sense as it labels as "RUSH-HOUR". Another is 'PARK-LONGER THAN 3 HOURS', this is the only infractions happened more during the early hours around 4am, this applies to weekends as well as weekdays.
It seems to me that residents live in an appartment building without a garage will be ticketed for overnight parking on their street if their street has no on-street permits. That's why most of the infractions happened in the early hours of the day.
The above analysis doesn't represent all the problems and areas since not every illegally parked car will be reported. It does give a general idea and might start a conversation on how and where the City of Toronto should intervene.
Source code used to created this post can be found [here](https://github.com/susanli2016/Data-Analysis-with-R/blob/master/ParkingTicketsTo.Rmd).