-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathode.js
More file actions
73 lines (61 loc) · 2.56 KB
/
ode.js
File metadata and controls
73 lines (61 loc) · 2.56 KB
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
class ODEsolver {
constructor(ode, y0, t0, t1) {
this.ode = ode
this.y0 = y0
this.t0 = t0
this.t1 = t1
if(isNaN(this.t0))
console.warn("invalid starting time");
}
euler(resolution) {
const h = (this.t1 - this.t0) / resolution
var ts = Array.from(Array(resolution + 1), (_, k) => k * h + this.t0) //time series datapoints
var ys = Array.from(Array(resolution + 1), () => Array(this.y0.length).fill(0))
ys[0] = this.y0
for (let i = 0; i < resolution; i++) {
ys[i + 1] = this.ode(ts[i], ys[i]).map((x, j) => ys[i][j] + x * h) //y_n+1 = y_n + dy/dx *h
}
return {
ts: ts,
ys: ys
}
}
rk4(resolution) {
const h = (this.t1 - this.t0) / resolution
var ts = Array.from(Array(resolution + 1), (_, k) => k * h + this.t0) //time series datapoints
var ys = Array.from(Array(resolution + 1), () => Array(this.y0.length).fill(0))
ys[0] = this.y0
if(this.y0.includes(NaN))
console.warn("y0 contains invalid starting value", this.y0)
for (let i = 0; i < resolution; i++) {
const k1 = this.ode(ts[i], ys[i]) // f(t, y_n)
const s1 = ys[i].map((y, j) => y + k1[j] * h / 2)
const k2 = this.ode(ts[i] + h / 2, s1) // f(t + h/2, y_n + k1*h/2)
const s2 = ys[i].map((y, j) => y + k2[j] * h / 2)
const k3 = this.ode(ts[i] + h / 2, s2) // f(t + h/2, y_n + k2*h/2)
const s3 = ys[i].map((y, j) => y + k3[j] * h)
const k4 = this.ode(ts[i] + h, s3) // f(t + h, y_n + k3*h)
ys[i + 1] = ys[i].map((x, j) => x + (k1[j] / 6 + k2[j] / 3 + k3[j] / 3 + k4[j] / 6) * h) //y_n+1 = y_n + (k1 +2*k2 + 2*k3 +k4)/6 *h
}
return {
ts: ts,
ys: ys
}
}
midpoint(resolution) {
const h = (this.t1 - this.t0) / resolution
var ts = Array.from(Array(resolution + 1), (_, k) => k * h + this.t0) //time series datapoints
var ys = Array.from(Array(resolution + 1), () => Array(this.y0.length).fill(0))
ys[0] = this.y0
for (let i = 0; i < resolution; i++) {
const k1 = this.ode(ts[i], ys[i]) // f(t, y_n)
const s1 = ys[i].map((y, j) => y + k1[j] * h / 2) // y_n + k1 * h/2
const k2 = this.ode(ts[i] + h / 2, s1) // f(t + h/2, y_n + k1*h/2)
ys[i + 1] = ys[i].map((x, j) => x + k2[j] * h) //y_n+1 = y_n + k2 *h
}
return {
ts: ts,
ys: ys
}
}
}