forked from diogro/ode_examples
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathAnalise qualitativa e pontos de bifurcacao em R.Rmd
303 lines (240 loc) · 11.7 KB
/
Analise qualitativa e pontos de bifurcacao em R.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
---
title: Análise qualitativa e Tutorial de diagrama de bifurcação em R
output: html_document
---
*Este tutorial pressupõe que você leu o [tutorial sobre integração numérica](http://pilaboratory.github.io/ode_examples/Integracao-numerica-em-R.html).*
# Explorando o espaço de parâmetros: diagramas de bifurcação
Diagramas de bifurcação representam as soluções (de longo prazo) de um
modelo em função de algum parâmetro, cujo efeito queremos
investigar. A ideia é que, à medida que esse parâmetro muda, as
soluções mudam de forma "bem comportada", e isso nos ajuda a entender
melhor o comportamento geral do modelo.
Neste tutorial, vamos estudar um modelo simples de predador-presa (o
Rosenzweig-MacArthur), e ver como a quantidade de recursos por presa
($K$) muda a dinâmica.
### O modelo de recursos do consumidor Rosenzweig-MacArthur
Este modelo é expresso como:
$$ \begin{aligned}
\frac{dR}{dt} &= rR \left( 1 - \frac{R}{K} \right) - \frac{a R C}{1+ahR} \\
\frac{dC}{dt} &= \frac{e a R C}{1+ahR} - d C
\end{aligned} $$
#### Soluções do modelo Rosenzweig–MacArthur
Usamos o mesmo método já explicado
[aqui](http://pilaboratory.github.io/ode_examples/Integracao-numerica-em-R.html)
para integrar este modelo numericamente:
```{r}
##Carregando bibliotecas
library(deSolve)
### Modelo Rosenzweig-MacArthur###
## Sistema de EDOs a ser integrado
RM <- function(t, y, parms){
with(as.list(c(y, parms)),
{
dR = r*R*(1-R/K) - a*R*C/(1+a*h*R)
dC = e*a*R*C/(1 + a*h*R) - d*C
return(list(c(R=dR,C=dC)))
})
}
##parâmetros
parms1 = c(r=1, K=10, a=1, h=0.1, e=0.1, d=0.1)
## condições iniciais
y0 = c(R=1,C=1)
## rodando a integração numérica do tempo 1 a 1000
out = ode(y=y0, times = seq(from=1, to=1000, by=0.5), func = RM, parms = parms1)
##plotando a solução
matplot(x=out[,1], y=out[,2:3], type = "l", lwd=2, lty=1,
col=c("blue", "darkgreen"), xlab="Time", ylab="Population size")
legend("topright", legend = c("Resource","Consumer"),
lty = 1, lwd=2, col=c("blue", "darkgreen"))
```
Para os parâmetros escolhidos acima, a solução de longo prazo
(assintótica) é um ponto fixo. Vamos ver isso no espaço de fase, ou
seja, o espaço de Predadores ($P$) vs. Presas ($V$). Notamos que as
setas estão "circulando", mas sempre apontam para dentro, e assim a
trajetória se move em direção ao meio, ao ponto fixo.
```{r}
## Espaço de fases e pontos fixos
##primeiro precisamos calcular as coordenadas para os vetores
xs = seq(from=0.9, to=1.3, length.out=7)
ys = seq(from=0.94, to=1.04, length.out=7)
coords = expand.grid(R=xs, C=ys)
## E, em seguida, faça um loop sobre todas as coordenadas para calcular as derivadas nesses pontos
dvs = matrix(NA, ncol=2, nrow=nrow(coords))
for(i in 1:nrow(coords))
dvs[i,] = unlist(RM(t=1, y = coords[i,], parms = parms1))
## agora plotamos a trajetória
plot(x=out[,2], y=out[,3], type="l", lwd=2, col="blue",
xlab="Resource",ylab="Consumer", xlim=c(0.9,1.3), ylim=c(0.94,1.04))
## e adicionamos o campo vetorial
arrows(x0=coords[,1], y0=coords[,2], x1=coords[,1]+dvs[,1]*0.5,
y1=coords[,2]+dvs[,2]*0.5, length=0.1, lwd=2)
```
#### Brincando um pouco com os parâmetros...
Aumentando a capacidade de carga $K$ de $10$ para $15$, agora vemos oscilações...
```{r}
# agora K = 15
parms2 = c(r=1, K=15, a=1, h=0.1, e=0.1, d=0.1)
out2 = ode(y=y0, times = seq(from = 1, to = 1000, by=0.5), func = RM, parms = parms2)
##plotando a solução
matplot(x=out2[,1], y=out2[,2:3], type = "l", lwd=2, lty=1,col=c("blue", "darkgreen"),
xlab="Time", ylab="Population size")
legend("topleft", legend = c("Resource","Consumer"), lty = 1, lwd=2,
col=c("blue", "darkgreen"))
```
Olhando novamente para o gráfico de espaço de fase, vemos agora que o
fluxo (as setas) de dentro apontam para fora, em direção a um ciclo
limite, e as setas de fora apontam para dentro. O ciclo limite
corresponde à solução periódica que acabamos de ver.
```{r}
## calculando os vetores novamente
xs = seq(from=0,to=6,length.out=8)
ys = seq(from=0,to=2,length.out=8)
coords = expand.grid(R=xs,C=ys)
dvs = matrix(NA, ncol=2, nrow=nrow(coords))
for(i in 1:nrow(coords))
dvs[i,] = unlist(RM(t=1, y = coords[i,], parms = parms2))
##A trajetoria
plot(x=out2[,2],y=out2[,3], type="l", lwd=2, col="blue",
xlab="Resource", ylab="Consumer", xlim=c(0,6),ylim=c(0,2))
##e os vetores
arrows(x0=coords[,1], y0=coords[,2], x1=coords[,1]+dvs[,1]*0.5,
y1=coords[,2]+dvs[,2]*0.5, length=0.1, lwd=2)
```
### O diagrama de bifurcação
Vimos as soluções para dois valores de $K$, $10$ e $15$, então
queremos plotá-los como uma função de $K$. No segundo caso, há
oscilações, então ao invés de pegar toda a solução, escolhemos apenas
o mínimo e o máximo da solução (depois de muito tempo). Quando a
solução é um ponto fixo, o mínimo e o máximo devem coincidir.
```{r}
##plotando tamanhos de população mínimo e máximo com diferentes valores de K
##objeto com os números de linha que usaremos para o gráfico a seguir
lines = (nrow(out)-500):nrow(out)
##criando um gráfico vazio
plot(0.1, type="n", xlim=c(0,20), ylim=range(out2[lines,2:3]), log="y",
xlab="K", ylab="Min and Max population")
##pontos para o K = 10
points(x = c(10,10), y = range(out[lines,3]), pch=21,bg="darkgreen", cex=3)
points(x = c(10,10), y = range(out[lines,2]), pch=21,bg="blue", cex=3)
##pontos para o K = 15
points(c(15,15), range(out2[lines,3]), pch=21,bg="darkgreen", cex=3)
points(c(15,15), range(out2[lines,2]), pch=21,bg="blue", cex=3)
```
Este é um diagrama de bifurcação pouco informativo: tem apenas dois
pontos em $K$! Vamos tentar com muitos valores de $K$.
O que acontece quando alteramos a capacidade de carga $K$ de valores
muito pequenos para valores muito grandes? Para valores muito
pequenos, o recurso não vai sustentar a população consumidora, mas
para valores maiores ok $K$, ambas as espécies devem ser
beneficiadas... certo?
```{r}
## este bloco calcula soluções para muitos K's, deve levar algum tempo
KK = seq(from = 0.5, to=25, by=0.5)
rminmax = matrix(NA, ncol=2, nrow=length(KK))
cminmax = matrix(NA, ncol=2, nrow=length(KK))
## Faça um loop sobre todos os valores de K e obtenha os tamanhos mínimos e máximos da população
for(i in 1:length(KK)){
parmsi = c(r=1, K=KK[i], a=1, h=0.1, e=0.1, d=0.1)
y0 = c(R=1,C=1)
out3 = ode(y=y0, times = seq(from = 1, to = 1000, by=0.5), func = RM, parms = parmsi)
rminmax[i,] = range(out3[(nrow(out3)-500):nrow(out3),2])
cminmax[i,] = range(out3[(nrow(out3)-500):nrow(out3),3])
}
plot(x=KK, y=rminmax[,1], type="l", lwd=2, col="blue",ylim=range(rminmax), log="y",
xlab="K", ylab="Min and Max population")
points(x=KK, y=rminmax[,2], type="l", lwd=2, col="blue")
points(x=KK, y=cminmax[,1], type="l", lwd=2, col="darkgreen",ylim=range(rminmax))
points(x=KK, y=cminmax[,2], type="l", lwd=2, col="darkgreen",ylim=range(rminmax))
```
Bem, a primeira previsão foi OK (observe que o gráfico acima usa uma
escala logarítmica), mas para $K$ altos, os mínimos da oscilação vão
para valores muito baixos, então as populações têm alto risco de
extinção. Esse fenômeno é o chamado **paradoxo do enriquecimento**.
### Dinâmica dos recursos do consumidor em um ambiente sazonal
Um tipo especial de diagrama de bifurcação pode ser usado quando temos
parâmetros que oscilam com o tempo, e queremos ver como isso interage
com o sistema. Vamos considerar novamente as equações de
Rosenzweig-MacArthur, mas agora fazemos $r$, a taxa de crescimento da
presa, oscilar senoidalmente no tempo:
$$ \begin{aligned}
\frac{dR}{dt} &= r(t) R \left( 1 - \frac{R}{K} \right) - \frac{a R C}{1+ahR} \\
\frac{dC}{dt} &= \frac{e a R C}{1+ahR} - d C \\
r(t) &= r_0 (1+\alpha \sin(2\pi t/T))
\end{aligned} $$
Integramos de forma usual:
```{r}
### Dinâmica dos recursos do consumidor em um ambiente sazonal ###
## sequência temporal
time <- seq(0, 2000, by = .5)
## paramêtros: vetor
parameters <- c(r0=1, alpha=0.1, T=80, K=10, a=1, h=0.1, e=0.1, d=0.1)
## condições iniciais: vetor
state <- c(R = 1, C = 1)
## Função R para calcular o valor das derivadas em cada valor de tempo
## Use os nomes das variáveis conforme definido nos vetores acima
RM2 <- function(t, state, parameters){
with(as.list(c(state, parameters)), {
r = r0 * (1 + alpha*sin(2*pi*t/T))
dR = R * ( r*(1 - R/K) - a*C / (1 + a*h*R) )
dC = e*a*R*C / (1 + a*h*R) - d*C
return(list(c(dR, dC)))
})
}
## Integrando com ode
out <- ode(y = state, times = time, func = RM2, parms = parameters)
## plotando
matplot(x = out[,1], y = out[,2:3], type="l", lwd=2, lty = 1,
col=c("blue", "darkgreen"), xlab = "Time", ylab = "Population size")
legend("topright", c("Resource", "Consumer"), lty=1, lwd=2, col=c("blue", "darkgreen"))
```
Observe que, mesmo com $K$ pequeno, as soluções oscilam devido à
oscilação de $r(t)$.
Agora usamos uma ferramenta que é a favorita de todos os tempos dos
físicos: o diagrama de ressonância. Funciona exatamente como um
diagrama de bifurcação, mas o parâmetro que é alterado é o período (ou
frequência) da oscilação externa.
```{r}
## Um diagrama de ressonância ##
## Nova sequência de tempo
time <- seq(0, 6000, by = 1)
## Sequência de valores de T
TT <- seq(1, 80, by = 2)
## Uma matriz para armazenar os resultados
results <- matrix(ncol=4, nrow=length(TT),
dimnames=list(NULL, c("R.min","R.max","C.min","C.max")))
## Loop sobre TT
for(i in 1:length(TT)){
parameters <- c(r0=1, alpha=0.1, T=TT[i], K=10, a=1, h=0.1, e=0.1, d=0.1)
tmp1 <- ode(y = state, times = time, func = RM2, parms = parameters)
results[i,1:2] <- range(tmp1[1001:nrow(tmp1), 2])
results[i,3:4] <- range(tmp1[1001:nrow(tmp1), 3])
}
## Plotando
plot(R.min ~ TT , data=results, type="l", lwd=2, lty = 1,
col="blue", xlab = "T", ylab = "Min / Max population size",
log="y", ylim = range(results))
lines(R.max ~ TT, data=results, type="l", lwd=2, lty = 1, col=c("blue"))
lines(C.min ~ TT, data=results, type="l", lwd=2, lty = 1, col=c("darkgreen"))
lines(C.max ~ TT, data=results, type="l", lwd=2, lty = 1, col=c("darkgreen"))
legend("topright", c("Resource", "Consumer"), lty=1, lwd=2, col=c("blue", "darkgreen"))
```
Vemos um pico forte! (lembre-se que esta é uma escala logarítmica). A
frequência em que esse pico ocorre é a **frequência de ressonância**
do sistema, e está relacionada com a frequência natural do sistema
(que existe mesmo quando vai para um ponto fixo com parâmetros
constantes!). A oscilação externa excita a frequência natural e aciona
ciclos de grande amplitude, como quando empurramos uma gangorra (ou
gangorra, o balancín).
## E se eu realmente quiser explorar um espaço de parâmetros de 10 dimensões?
Primeiro: boa sorte. Em segundo lugar, você provavelmente terá que
amostrar o espaço, em vez de passar por tudo. O método recomendado
para fazer isso são amostras de hipercubo latino ( [Latin Hypercube
samples])(http://en.wikipedia.org/wiki/Latin_hypercube_sampling), que
usa uma amostragem aleatória de combinaçãoes de parâmetros, garantindo
uma boa varredura do espaço de parâmetros. Observe, no entanto, que
esse método é uma maneira de amostrar o espaço de parâmetros e fazer
estatísticas úteis com ele, portanto, o resultado só fará sentido se
você souber interpretar adequadamente os resultados. Dito isso,
existem implementações para R e python:
* [R-Cran pse: Parameter space exploration](http://cran.r-project.org/web/packages/pse/)
* [PyDOE: design of experiments for Python](http://pythonhosted.org/pyDOE/randomized.html)