Callbacks and Events#
Docs:
https://docs.sciml.ai/DiffEqDocs/stable/features/callback_functions/
https://tutorials.sciml.ai/html/introduction/04-callbacks_and_events.html
Types#
ContinuousCallback
: applied when a given continuous condition function hits zero. SciML solvers are able to interpolate the integration step to meet the condition.DiscreteCallback
: applied when its condition function is true.CallbackSet(cb1, cb2, ...)
: Multiple callbacks can be chained together to form aCallbackSet
.VectorContinuousCallback
: an array ofContinuousCallback
s.
How to#
After you define a callback, send it to the solve()
function:
sol = solve(prob, alg, callback=cb)
DiscreteCallback : Interventions at Preset Times#
The drug concentration in the blood follows exponential decay.
using OrdinaryDiffEq
using DiffEqCallbacks
using Plots
Exponential decay model
function dosing!(du, u, p, t)
du[1] = -u[1]
end
u0 = [10.0]
tspan = (0.0, 10.0)
prob = ODEProblem(dosing!, u0, tspan)
sol = solve(prob, Tsit5())
plot(sol, lw=3)
data:image/s3,"s3://crabby-images/ae044/ae04499245decd2ea0ce8fa73e2a944e08925d47" alt="_images/15ea67bd0b9318597d784d2cf8f6391180cb9651390828c8a0bec5dcececb15f.png"
Add a dose of 10 at t = 4
. You may need to force the solver to stop at t==4
by using tstops=[4.0]
to properly apply the callback.
condition(u, t, integrator) = t == 4
affect!(integrator) = integrator.u[1] += 10
cb = DiscreteCallback(condition, affect!)
sol = solve(prob, Tsit5(), callback=cb, tstops=[4.0])
plot(sol)
data:image/s3,"s3://crabby-images/d84e4/d84e4acad65d241ecbd1999c4c13419a183f89ae" alt="_images/e6adfa4672cdea9d611433a12e4b923d7bb8fd982a7b12f001a791c9e1551592.png"
Applying multiple doses.
dosetimes = [4.0, 8.0]
condition(u, t, integrator) = t ∈ dosetimes
affect!(integrator) = integrator.u[1] += 10
cb = DiscreteCallback(condition, affect!)
sol = solve(prob, Tsit5(), callback=cb, tstops=dosetimes)
plot(sol)
data:image/s3,"s3://crabby-images/40713/4071353c35bb252ddde79072bb35c30e1ec49c8a" alt="_images/752e295f007c143e4ae72d110138da7975fab90f79910dcbc8c407351242ceb2.png"
Conditional dosing. Note that a dose was not given at t=6 because the concentration is not more than 1.
dosetimes = [4.0, 6.0, 8.0]
condition(u, t, integrator) = t ∈ dosetimes && (u[1] < 1.0)
affect!(integrator) = integrator.u[1] += 10integrator.t
cb = DiscreteCallback(condition, affect!)
sol = solve(prob, Tsit5(), callback=cb, tstops=dosetimes)
plot(sol)
data:image/s3,"s3://crabby-images/cc8e0/cc8e06d5bd66261036344bb0571184a383dda371" alt="_images/91421c20440741b85e00c23d772b7897d5d99e40b583a69e0c1c14f31304d053.png"
PresetTimeCallback#
PresetTimeCallback(tstops, affect!)
takes an array of times and an affect!
function to apply. The solver will stop at tstops
to ensure callbacks are applied.
dosetimes = [4.0, 8.0]
affect!(integrator) = integrator.u[1] += 10
cb = PresetTimeCallback(dosetimes, affect!)
sol = solve(prob, Tsit5(), callback=cb)
plot(sol)
data:image/s3,"s3://crabby-images/40713/4071353c35bb252ddde79072bb35c30e1ec49c8a" alt="_images/752e295f007c143e4ae72d110138da7975fab90f79910dcbc8c407351242ceb2.png"
Periodic callback#
PeriodicCallback
is a special case of DiscreteCallback
, used when a function should be called periodically in terms of integration time (as opposed to wall time).
prob = ODEProblem(dosing!, u0, (0.0, 25.0))
affect!(integrator) = integrator.u[1] += 10
cb = PeriodicCallback(affect!, 1.0)
sol = solve(prob, Tsit5(), callback=cb)
plot(sol)
data:image/s3,"s3://crabby-images/f7a97/f7a97c22e083751e2bdfeae2b9e80da64179ace0" alt="_images/03176b44c0ee09c60f8950d5a9f037ddf2cb3046b1e9710c8403592ed2ca0e4d.png"
Continuous Callback#
bouncing Ball example#
using OrdinaryDiffEq
using DiffEqCallbacks
using Plots
function ball!(du, u, p, t)
du[1] = u[2]
du[2] = -p
end
# When the ball touches the ground
# Telling when event_f(u,t) == 0
function condition(u, t, integrator)
u[1]
end
function affect!(integrator)
integrator.u[2] = -integrator.u[2]
end
cb = ContinuousCallback(condition, affect!)
u0 = [50.0, 0.0]
tspan = (0.0, 15.0)
p = 9.8
prob = ODEProblem(ball!, u0, tspan, p)
sol = solve(prob, Tsit5(), callback=cb)
plot(sol, label=["Position" "Velocity"])
data:image/s3,"s3://crabby-images/ac108/ac10878277657bb9fcda372535fdacb8d685b151" alt="_images/b08489b1d83579ab7342c40006a7a40e1594c3b3da75040bcc9a5d9f6ef6638e.png"
Vector Continuous Callback#
You can group multiple callbacks together into a vector. In this example, we will simulate bouncing ball with multiple walls.
using OrdinaryDiffEq
using DiffEqCallbacks
using Plots
function ball_2d!(du, u, p, t)
du[1] = u[2] ## x_pos
du[2] = -p ## x_vel
du[3] = u[4] ## y_pos
du[4] = 0.0 ## y_vel
end
ball_2d! (generic function with 1 method)
Vector conditions
function condition(out, u, t, integrator) ## Event when event_f(u,t) == 0
out[1] = u[1]
out[2] = (u[3] - 10.0)u[3]
end
condition (generic function with 2 methods)
Vector affects
function affect!(integrator, idx)
if idx == 1
integrator.u[2] = -0.9integrator.u[2]
elseif idx == 2
integrator.u[4] = -0.9integrator.u[4]
end
end
cb = VectorContinuousCallback(condition, affect!, 2)
u0 = [50.0, 0.0, 0.0, 2.0]
tspan = (0.0, 15.0)
p = 9.8
prob = ODEProblem(ball_2d!, u0, tspan, p)
sol = solve(prob, Tsit5(), callback=cb, dt=1e-3, adaptive=false)
plot(sol, idxs=(3, 1))
data:image/s3,"s3://crabby-images/a1af6/a1af6895f7b1edf236980384493c2995cf3120ca" alt="_images/9fac547ca337887146f872ef0feb02c10d2ea8384b08b568a7eecbb638867f93.png"
Other Callbacks#
https://diffeq.sciml.ai/stable/features/callback_library/
ManifoldProjection(g)
: keepg(u) = 0
for energy conservation.AutoAbstol()
: automatically adapt the absolute tolerance.PositiveDomain()
: enforce non-negative solution. The system should also be defined outside the positive domain. There is also a more general versionGeneralDomain()
.StepsizeLimiter((u,p,t) -> maxtimestep)
: restrict maximal allowed time step.FunctionCallingCallback((u, t, int) -> func_content; funcat=[t1, t2, ...])
: call a function at the time points (t1, t2, …) of interest.SavingCallback((u, t, int) -> data_to_be_saved, dataprototype)
: call a function and saves the result.IterativeCallback(time_choice(int) -> t2, user_affect!)
: apply an effect at the time of the next callback.
This notebook was generated using Literate.jl.