Mathematical and statistical software often relies on sequential
computations. ordinary differential equations where
numerical approximations are based on looping over the evolving time.
When using high-level languages such as R
calculations can
be very slow unless the algorithms can be vectorized
There are various excellent (O)DE solvers (R: deSolve) Here I will
illustrate the above techniques using the targeted
R
-package based on the target C++
library.
The ODE is specified using the specify_ode
function
args(targeted::specify_ode)
#> function (code, fname = NULL, pname = c("dy", "x", "y", "p"))
#> NULL
The differential equations are here specified as a string containing
the C++
code defining the differential equation via the
code
argument. The variable names are defined through the
pname
argument which defaults to
All variables are treated as armadillo vectors/matrices,
arma::mat
.
As an example, we can specify the simple differential equation y′(t) = y(t) − 1
This compiles the function and stores the pointer in the variable
dy
.
To solve the ODE we must then use the function
solve_ode
The first argument is the external pointer, the second argument
input
is the input matrix (x(t) above), and the
init
argument is the vector of initial boundary conditions
y(0). The argument
par
is the vector of parameters defining the ODE (p).
In this example the input variable does not depend on any exogenous variables so we only need to supply the time points, and the defined ODE does not depend on any parameters. To approximate the solution with initial condition y(0) = 0, we therefore run the following code
As a more interesting example consider the Lorenz Equations $$\frac{dx_{t}}{dt} = \sigma(y_{t}-x_{t})$$ $$\frac{dy_{t}}{dt} = x_{t}(\rho-z_{t})-y_{t}$$ $$\frac{dz_{t}}{dt} = x_{t}y_{t}-\beta z_{t}$$
we may define them as
library(targeted)
ode <- 'dy(0) = p(0)*(y(1)-y(0));
dy(1) = y(0)*(p(1)-y(2));
dy(2) = y(0)*y(1)-p(2)*y(2);'
f <- specify_ode(ode)
With the choice of parameters given by σ = 10, ρ = 28, β = 8/3 and initial conditions (x0, y0, z0) = (1, 1, 1), we can calculate the solution
tt <- seq(0, 100, length.out=2e4)
y <- solve_ode(f, input=tt, init=c(1, 1, 1), par=c(10, 28, 8/3))
head(y)
#> [,1] [,2] [,3]
#> [1,] 1.000000 1.000000 1.0000000
#> [2,] 1.003322 1.135177 0.9920639
#> [3,] 1.013094 1.271279 0.9849468
#> [4,] 1.029068 1.409156 0.9786954
#> [5,] 1.051048 1.549630 0.9733721
#> [6,] 1.078888 1.693496 0.9690541
colnames(y) <- c("x","y","z")
scatterplot3d::scatterplot3d(y, cex.symbols=0.1, type='b',
color=viridisLite::viridis(nrow(y)))
To illustrate the use of exogenous inputs, consider the following simulated data
n <- 1e4
tt <- seq(0, 10, length.out=n) # Time
xx <- rep(0, n); xx[(n/3):(2*n/3)] <- 1 # Exogenous input, x(t)
input <- cbind(tt, xx)
and the following ODE
y′(t) = β0 + β1y(t) + β2y(t)x(t) + β3x(t) ⋅ t
With y(0) = 100 and β0 = 0, β1 = 0.4, β2 = −0.5, β3 = −5 we obtain the following solution
sessionInfo()
#> R version 4.4.2 (2024-10-31)
#> Platform: x86_64-pc-linux-gnu
#> Running under: Ubuntu 24.04.1 LTS
#>
#> Matrix products: default
#> BLAS: /usr/lib/x86_64-linux-gnu/openblas-pthread/libblas.so.3
#> LAPACK: /usr/lib/x86_64-linux-gnu/openblas-pthread/libopenblasp-r0.3.26.so; LAPACK version 3.12.0
#>
#> locale:
#> [1] LC_CTYPE=en_US.UTF-8 LC_NUMERIC=C
#> [3] LC_TIME=en_US.UTF-8 LC_COLLATE=C
#> [5] LC_MONETARY=en_US.UTF-8 LC_MESSAGES=en_US.UTF-8
#> [7] LC_PAPER=en_US.UTF-8 LC_NAME=C
#> [9] LC_ADDRESS=C LC_TELEPHONE=C
#> [11] LC_MEASUREMENT=en_US.UTF-8 LC_IDENTIFICATION=C
#>
#> time zone: Etc/UTC
#> tzcode source: system (glibc)
#>
#> attached base packages:
#> [1] stats graphics grDevices utils datasets methods base
#>
#> other attached packages:
#> [1] targeted_0.5 lava_1.8.0 knitr_1.49
#>
#> loaded via a namespace (and not attached):
#> [1] Matrix_1.7-1 future.apply_1.11.3 jsonlite_1.8.9
#> [4] futile.logger_1.4.3 compiler_4.4.2 Rcpp_1.0.13-1
#> [7] parallel_4.4.2 jquerylib_0.1.4 globals_0.16.3
#> [10] splines_4.4.2 yaml_2.3.10 fastmap_1.2.0
#> [13] lattice_0.22-6 R6_2.5.1 future_1.34.0
#> [16] nloptr_2.1.1 maketools_1.3.1 bslib_0.8.0
#> [19] rlang_1.1.4 cachem_1.1.0 xfun_0.49
#> [22] sass_0.4.9 sys_3.4.3 viridisLite_0.4.2
#> [25] cli_3.6.3 formatR_1.14 futile.options_1.0.1
#> [28] digest_0.6.37 grid_4.4.2 mvtnorm_1.3-2
#> [31] lifecycle_1.0.4 RcppArmadillo_14.2.2-1 timereg_2.0.6
#> [34] scatterplot3d_0.3-44 pracma_2.4.4 evaluate_1.0.1
#> [37] data.table_1.16.4 lambda.r_1.2.4 numDeriv_2016.8-1.1
#> [40] listenv_0.9.1 codetools_0.2-20 buildtools_1.0.0
#> [43] survival_3.8-3 optimx_2024-12.2 parallelly_1.41.0
#> [46] rmarkdown_2.29 mets_1.3.4 tools_4.4.2
#> [49] htmltools_0.5.8.1