-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathinYIpcA0K.rmd
266 lines (187 loc) · 7.18 KB
/
inYIpcA0K.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
---
title: "The Simulacrum"
output: html_notebook
---
queueing really.
CH13 of [RA]
Excercises: 13.13, 13.16
Cases: 13.1, 13.2
<h4>Variables of a q system:</h4>
lambda - arrival rate (units/time period) <br>
mu - service rate (units/time period, per server) <br>
s, or c in our examples - number of servers<br>
<h4>Parameters of a queueing system that we need:</h4>
<ul>
<li>U - utilization factor. denoted as rho below</li>
<li>P0 - probability that system is empty</li>
<li>L_q - average number in line</li>
<li>L - average number in system</li>
<li>W_q - average time in line</li>
<li>W average time in system</li>
<li>P_w - probability of having to wait in line</li>
<li>P_n - probability of n units in the system</li>
</ul>
<h4> Implementation with R </h4>
```{r}
#we use the simmer package, first install it.it also has a separate plotting package, get it too. :) (un-comment the commands first!)
#install.packages("simmer")
#install.packages("simmer.plot")
rm(list=ls())
```
<h2> A M/M/c system with unlimited queue / clients </h2>
```{r}
#setting up the environment
library(simmer)
#the basic parameters for a M/M/c system
lambda <- 11
mu <- 6
c<- 2
rho <- lambda/mu
#total utilization with multiple servers
rho_c <- lambda/(c*mu)
#set up the function that the servers use
mmc.trajectory <- trajectory() %>%
seize("resource", amount=1) %>%
timeout(function() rexp(1, mu)) %>%
release("resource", amount=1)
#set up the environment with the arrival / server fuctions
#as you see exponential distributions around mu and lambda are chosen
mmc.env <- simmer() %>%
#note that you can input the queue length here. right now set to infinite.
add_resource("resource", capacity=c, queue_size=Inf) %>%
add_generator("arrival", mmc.trajectory, function() rexp(1, lambda)) %>%
run(until=2000)
```
System stats (formulas from p.649, CH13.6.1 of Ragsdale):
```{r}
library(simmer.plot)
# Theoretical values
ind <- 0:(c-1)
#probability of empty
pi_0 <-1/(sum(rho**ind/factorial(ind)) + (rho**c/factorial(c))*(1/(1-rho/c)))
#average number in system
mmc.L <- rho + ((rho**c/factorial(c))*pi_0)*((rho/c)/((1-rho/c)**2))
#average number in queue
mmc.Lq <- mmc.L - rho
```
Average waiting time and avg time in system:
```{r}
mmc.arrivals <- get_mon_arrivals(mmc.env)
mmc.resources<- get_mon_resources(mmc.env)
#avg time in system
mmc.t_system <- mmc.arrivals$end_time - mmc.arrivals$start_time
lambda_eff<-lambda
#avg wait time
mmc.W <- mmc.L / lambda_eff
mmc.W ; mean(mmc.t_system)
```
We are still missing the probabilities (of n units in system etc.)
```{r}
#probability of having to wait in line (of line being non-zero?)
p_w <- 1/factorial(c)*(lambda/mu)^c*(c*mu/(c*mu-lambda))*pi_0
#probability of n units in the system. set nunits to your value
nunits <- 2
p_n <- ((lambda/mu)^nunits)/factorial(nunits)*pi_0
#if n>s, another formula applies. Mind not to run the whole code chunk, but the individual lines. otherwise you'll only get the second result ^_^
nunitsm <- 1:10
p_n <- ((lambda/mu)^nunitsm)/(c*factorial(c)^(nunitsm-c))*pi_0
#we can plot the probabilities of a number of units in the system (or in queue). We have two formulas, so we need a loop. Ugly solution below
probabilities_units <- as.data.frame(cbind(nunitsm,p_n <- ((lambda/mu)^nunitsm)/(c*factorial(c)^(nunitsm-c))*pi_0))
#now we need the probabilities for when its less than c,(amount of servers)
for (ci in 1:c) {
probabilities_units$V2[ci] <- ((lambda/mu)^ci)/factorial(ci)*pi_0
}
```
What about visualization?
```{r}
#a barplot of probabilities of n units in system
barplot(probabilities_units$V2,main="probability of n units in system")
# Evolution of the average number of customers in the system
plot(mmc.env, "resources", "usage", "resource", items="system") +
geom_hline(yintercept=mmc.L) #+ ylim(0, 15)
#same but in queue
plot(mmc.env, "resources", "usage", "resource", items="queue") +
geom_hline(yintercept=mmc.L) #+ ylim(0, 15)
#same but for how many servers are used
plot(mmc.env, "resources", "usage", "resource", items="server") +
geom_hline(yintercept=mmc.L) #+ ylim(0, 15)
#you can play with the Y-axis of the chart by removing the comment sign before + ylim(..). If your graph area becomes to small, the line will get broken + error messages. :)
#plot the average waiting time for arrivals!
#remember we put the arrival times into the mmc.arrivals object with the get_mon_arrivals function?
plot(mmc.arrivals,metric="waiting_time")
#plot activity time (served time?)
plot(mmc.arrivals,metric="activity_time")
#plot all resources (resources are in this case queue + servers)
plot(mmc.resources, metric="usage")
```
What can you plot with this?
Look here for examples:
https://r-simmer.org/extensions/plot/reference/plot.mon.html
<h2> An M/M/c/k </h2>
An MMck is implemented in much the same way, we only need to adjust the formulas.
first, purge the environment (we'll be reusing the variable names ಠ‿ಠ)
```{r}
rm(list=ls())
```
Now create a new simulation environment with simmer
```{r}
#setting up the environment
library(simmer)
#the basic parameters for a M/M/c system
lambda <- 11
mu <- 6
c<- 5
k<-3
rho<- lambda/mu
rho_c <- lambda/(c*mu)
#set up the function that the servers use
mmc.trajectory <- trajectory() %>%
seize("resource", amount=1) %>%
timeout(function() rexp(1, mu)) %>%
release("resource", amount=1)
#set up the environment with the arrival / server fuctions
#as you see exponential distributions around mu and lambda are chosen
mmc.env <- simmer() %>%
# --- -- -- - -- - - -- - --
#note that you can input the queue length here. right now set to a set length!.
add_resource("resource", capacity=c, queue_size=k) %>%
add_generator("arrival", mmc.trajectory, function() rexp(1, lambda)) %>%
run(until=2000)
```
Now just to adjust the pesky formulas ಥ_ಥ (find them on p.652, CH13.7 of Ragsdale)
```{r}
# index n
ind <- 1:(c)
ind2 <- (c+1):k
#index for length
ind3 <- 0:(c-1)
#units in system. just change the upper value in the vector to one you want
nunitsm <- 1:10
# probability of empty
pi_0 <- 1/(sum(rho**ind/factorial(ind)) + (rho**c/factorial(c))*sum((1/(1-rho/c)^(ind2-c))))
#probability of n units in system
#for s < n <= k
probabilities_units <- as.data.frame(cbind(nunitsm, ((lambda/mu)^nunitsm)/(c*factorial(c)^(nunitsm-c))*pi_0))
#for n <= s
for (ci in 1:c) {
probabilities_units$V2[ci] <- ((lambda/mu)^ci)/factorial(ci)*pi_0
}
#for n > k
probabilities_units$V2[probabilities_units$nunitsm > k] <- 0
#average lengths, of in system and in queue
#average number in queue
mmc.Lq <- (pi_0*rho^c*rho_c)/(factorial(c)*(1-rho_c)^2)*(1-rho_c^(k-c)-(k-c)*rho_c^(k-c)*(1-rho_c))
#average number in system
mmc.L <- sum(ind3*probabilities_units$V2[probabilities_units$nunitsm==ind3])+mmc.Lq+c*(1-sum(probabilities_units$V2[probabilities_units$nunitsm==ind3]))
#simulation outputs for avg waittime etc. not sure if they are correct (づ。◕‿‿◕。)づ
mmc.arrivals <- get_mon_arrivals(mmc.env)
mmc.resources<- get_mon_resources(mmc.env)
#avg time in system
mmc.t_system <- mmc.arrivals$end_time - mmc.arrivals$start_time
lambda_eff<-lambda
#avg wait time
mmc.W <- mmc.L / lambda_eff
mmc.W ; mean(mmc.t_system)
```
The graphs etc you can get with the functions in previous chunks.
thaat's it! q(❂‿❂)p