From 0c788e979d1e42b40e7b10dd08d10d3bd532825c Mon Sep 17 00:00:00 2001 From: Bart de Koning Date: Tue, 25 Jun 2024 20:30:35 +0200 Subject: [PATCH 1/5] Introduce idx_prev to use as iguess, consistently use idx --- src/derivatives.jl | 54 ++++++++++++++++++++------------- src/interpolation_caches.jl | 38 +++++++++++++++++------ src/interpolation_methods.jl | 59 +++++++++++++++++++----------------- src/interpolation_utils.jl | 22 ++++++++------ 4 files changed, 105 insertions(+), 68 deletions(-) diff --git a/src/derivatives.jl b/src/derivatives.jl index 3bd78824..bd40e2ef 100644 --- a/src/derivatives.jl +++ b/src/derivatives.jl @@ -1,8 +1,20 @@ function derivative(A, t, order = 1) - order > 2 && throw(DerivativeNotFoundError()) ((t < A.t[1] || t > A.t[end]) && !A.extrapolate) && throw(ExtrapolationError()) - order == 1 && return _derivative(A, t, firstindex(A.t) - 1)[1] - return ForwardDiff.derivative(t -> _derivative(A, t, firstindex(A.t) - 1)[1], t) + iguess = A.idx_prev[] + + return if order == 1 + val, idx = _derivative(A, t, iguess) + A.idx_prev[] = idx + val + elseif order == 2 + ForwardDiff.derivative(t -> begin + val, idx = _derivative(A, t, iguess) + A.idx_prev[] = idx + val + end, t) + else + throw(DerivativeNotFoundError()) + end end function _derivative(A::LinearInterpolation{<:AbstractVector}, t::Number, iguess) @@ -105,17 +117,17 @@ function _derivative(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number) der end -_derivative(A::LagrangeInterpolation{<:AbstractVector}, t::Number, i) = _derivative(A, t), i -_derivative(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number, i) = _derivative(A, t), i +_derivative(A::LagrangeInterpolation{<:AbstractVector}, t::Number, idx) = _derivative(A, t), idx +_derivative(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number, idx) = _derivative(A, t), idx function _derivative(A::AkimaInterpolation{<:AbstractVector}, t::Number, iguess) - i = searchsortedfirstcorrelated(A.t, t, iguess) - i > length(A.t) ? i -= 1 : nothing - i -= 1 - i == 0 ? i += 1 : nothing - j = min(i, length(A.c)) # for smooth derivative at A.t[end] - wj = t - A.t[i] - (@evalpoly wj A.b[i] 2A.c[j] 3A.d[j]), i + idx = searchsortedfirstcorrelated(A.t, t, iguess) + idx > length(A.t) ? idx -= 1 : nothing + idx -= 1 + idx == 0 ? idx += 1 : nothing + j = min(idx, length(A.c)) # for smooth derivative at A.t[end] + wj = t - A.t[idx] + (@evalpoly wj A.b[idx] 2A.c[j] 3A.d[j]), idx end function _derivative(A::ConstantInterpolation{<:AbstractVector}, t::Number) @@ -139,15 +151,15 @@ end # CubicSpline Interpolation function _derivative(A::CubicSpline{<:AbstractVector}, t::Number, iguess) - i = searchsortedfirstcorrelated(A.t, t, iguess) - i > length(A.t) ? i -= 1 : nothing - i -= 1 - i == 0 ? i += 1 : nothing - dI = -3A.z[i] * (A.t[i + 1] - t)^2 / (6A.h[i + 1]) + - 3A.z[i + 1] * (t - A.t[i])^2 / (6A.h[i + 1]) - dC = A.u[i + 1] / A.h[i + 1] - A.z[i + 1] * A.h[i + 1] / 6 - dD = -(A.u[i] / A.h[i + 1] - A.z[i] * A.h[i + 1] / 6) - dI + dC + dD, i + idx = searchsortedfirstcorrelated(A.t, t, iguess) + idx > length(A.t) ? idx -= 1 : nothing + idx -= 1 + idx == 0 ? idx += 1 : nothing + dI = -3A.z[idx] * (A.t[idx + 1] - t)^2 / (6A.h[idx + 1]) + + 3A.z[idx + 1] * (t - A.t[idx])^2 / (6A.h[idx + 1]) + dC = A.u[idx + 1] / A.h[idx + 1] - A.z[idx + 1] * A.h[idx + 1] / 6 + dD = -(A.u[idx] / A.h[idx + 1] - A.z[idx] * A.h[idx + 1] / 6) + dI + dC + dD, idx end function _derivative(A::BSplineInterpolation{<:AbstractVector{<:Number}}, t::Number, iguess) diff --git a/src/interpolation_caches.jl b/src/interpolation_caches.jl index 62352001..83047730 100644 --- a/src/interpolation_caches.jl +++ b/src/interpolation_caches.jl @@ -17,8 +17,9 @@ struct LinearInterpolation{uType, tType, T} <: AbstractInterpolation{T} u::uType t::tType extrapolate::Bool + idx_prev::Base.RefValue{Int} function LinearInterpolation(u, t, extrapolate) - new{typeof(u), typeof(t), eltype(u)}(u, t, extrapolate) + new{typeof(u), typeof(t), eltype(u)}(u, t, extrapolate, Ref(1)) end end @@ -48,10 +49,11 @@ struct QuadraticInterpolation{uType, tType, T} <: AbstractInterpolation{T} t::tType mode::Symbol extrapolate::Bool + idx_prev::Base.RefValue{Int} function QuadraticInterpolation(u, t, mode, extrapolate) mode ∈ (:Forward, :Backward) || error("mode should be :Forward or :Backward for QuadraticInterpolation") - new{typeof(u), typeof(t), eltype(u)}(u, t, mode, extrapolate) + new{typeof(u), typeof(t), eltype(u)}(u, t, mode, extrapolate, Ref(1)) end end @@ -86,6 +88,7 @@ struct LagrangeInterpolation{uType, tType, T, bcacheType} <: n::Int bcache::bcacheType extrapolate::Bool + idx_prev::Base.RefValue{Int} function LagrangeInterpolation(u, t, n, extrapolate) bcache = zeros(eltype(u[1]), n + 1) fill!(bcache, NaN) @@ -93,7 +96,8 @@ struct LagrangeInterpolation{uType, tType, T, bcacheType} <: t, n, bcache, - extrapolate) + extrapolate, + Ref(1)) end end @@ -128,6 +132,7 @@ struct AkimaInterpolation{uType, tType, bType, cType, dType, T} <: c::cType d::dType extrapolate::Bool + idx_prev::Base.RefValue{Int} function AkimaInterpolation(u, t, b, c, d, extrapolate) new{typeof(u), typeof(t), typeof(b), typeof(c), typeof(d), eltype(u)}(u, @@ -135,7 +140,9 @@ struct AkimaInterpolation{uType, tType, bType, cType, dType, T} <: b, c, d, - extrapolate) + extrapolate, + Ref(1), + ) end end @@ -186,8 +193,9 @@ struct ConstantInterpolation{uType, tType, dirType, T} <: AbstractInterpolation{ t::tType dir::Symbol # indicates if value to the $dir should be used for the interpolation extrapolate::Bool + idx_prev::Base.RefValue{Int} function ConstantInterpolation(u, t, dir, extrapolate) - new{typeof(u), typeof(t), typeof(dir), eltype(u)}(u, t, dir, extrapolate) + new{typeof(u), typeof(t), typeof(dir), eltype(u)}(u, t, dir, extrapolate, Ref(1)) end end @@ -219,6 +227,7 @@ struct QuadraticSpline{uType, tType, tAType, dType, zType, T} <: d::dType z::zType extrapolate::Bool + idx_prev::Base.RefValue{Int} function QuadraticSpline(u, t, tA, d, z, extrapolate) new{typeof(u), typeof(t), typeof(tA), typeof(d), typeof(z), eltype(u)}(u, @@ -226,7 +235,9 @@ struct QuadraticSpline{uType, tType, tAType, dType, zType, T} <: tA, d, z, - extrapolate) + extrapolate, + Ref(1), + ) end end @@ -286,12 +297,15 @@ struct CubicSpline{uType, tType, hType, zType, T} <: AbstractInterpolation{T} h::hType z::zType extrapolate::Bool + idx_prev::Base.RefValue{Int} function CubicSpline(u, t, h, z, extrapolate) new{typeof(u), typeof(t), typeof(h), typeof(z), eltype(u)}(u, t, h, z, - extrapolate) + extrapolate, + Ref(1), + ) end end @@ -365,6 +379,7 @@ struct BSplineInterpolation{uType, tType, pType, kType, cType, T} <: pVecType::Symbol knotVecType::Symbol extrapolate::Bool + idx_prev::Base.RefValue{Int} function BSplineInterpolation(u, t, d, @@ -382,7 +397,9 @@ struct BSplineInterpolation{uType, tType, pType, kType, cType, T} <: c, pVecType, knotVecType, - extrapolate) + extrapolate, + Ref(1), + ) end end @@ -482,6 +499,7 @@ struct BSplineApprox{uType, tType, pType, kType, cType, T} <: pVecType::Symbol knotVecType::Symbol extrapolate::Bool + idx_prev::Base.RefValue{Int} function BSplineApprox(u, t, d, @@ -501,7 +519,9 @@ struct BSplineApprox{uType, tType, pType, kType, cType, T} <: c, pVecType, knotVecType, - extrapolate) + extrapolate, + Ref(1), + ) end end diff --git a/src/interpolation_methods.jl b/src/interpolation_methods.jl index e00adc59..0b48963e 100644 --- a/src/interpolation_methods.jl +++ b/src/interpolation_methods.jl @@ -1,7 +1,9 @@ -function _interpolate(interp, t) - ((t < interp.t[1] || t > interp.t[end]) && !interp.extrapolate) && +function _interpolate(A, t) + ((t < A.t[1] || t > A.t[end]) && !A.extrapolate) && throw(ExtrapolationError()) - _interpolate(interp, t, firstindex(interp.t) - 1)[1] + val, idx_prev = _interpolate(A, t, A.idx_prev[]) + A.idx_prev[] = idx_prev + return val end # Linear Interpolation @@ -114,61 +116,62 @@ function _interpolate(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number) N / D end -function _interpolate(A::LagrangeInterpolation{<:AbstractVector}, t::Number, i) - _interpolate(A, t), i +function _interpolate(A::LagrangeInterpolation{<:AbstractVector}, t::Number, idx) + _interpolate(A, t), idx end -function _interpolate(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number, i) - _interpolate(A, t), i +function _interpolate(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number, idx) + _interpolate(A, t), idx end function _interpolate(A::AkimaInterpolation{<:AbstractVector}, t::Number, iguess) - i = max(1, min(searchsortedlastcorrelated(A.t, t, iguess), length(A.t) - 1)) - wj = t - A.t[i] - (@evalpoly wj A.u[i] A.b[i] A.c[i] A.d[i]), i + idx = max(1, min(searchsortedlastcorrelated(A.t, t, iguess), length(A.t) - 1)) + wj = t - A.t[idx] + (@evalpoly wj A.u[idx] A.b[idx] A.c[idx] A.d[idx]), idx end # ConstantInterpolation Interpolation function _interpolate(A::ConstantInterpolation{<:AbstractVector}, t::Number, iguess) if A.dir === :left # :left means that value to the left is used for interpolation - i = max(1, searchsortedlastcorrelated(A.t, t, iguess)) - return A.u[i], i + idx = max(1, searchsortedlastcorrelated(A.t, t, iguess)) + return A.u[idx], idx else # :right means that value to the right is used for interpolation - i = min(length(A.t), searchsortedfirstcorrelated(A.t, t, iguess)) - return A.u[i], i + idx = min(length(A.t), searchsortedfirstcorrelated(A.t, t, iguess)) + return A.u[idx], idx end end function _interpolate(A::ConstantInterpolation{<:AbstractMatrix}, t::Number, iguess) if A.dir === :left # :left means that value to the left is used for interpolation - i = max(1, searchsortedlastcorrelated(A.t, t, iguess)) - return A.u[:, i], i + idx = max(1, searchsortedlastcorrelated(A.t, t, iguess)) + return A.u[:, idx], idx else # :right means that value to the right is used for interpolation - i = min(length(A.t), searchsortedfirstcorrelated(A.t, t, iguess)) - return A.u[:, i], i + idx = min(length(A.t), searchsortedfirstcorrelated(A.t, t, iguess)) + return A.u[:, idx], idx end end # QuadraticSpline Interpolation function _interpolate(A::QuadraticSpline{<:AbstractVector}, t::Number, iguess) - i = min(max(2, searchsortedfirstcorrelated(A.t, t, iguess)), length(A.t)) - Cᵢ = A.u[i - 1] - σ = 1 // 2 * (A.z[i] - A.z[i - 1]) / (A.t[i] - A.t[i - 1]) - return A.z[i - 1] * (t - A.t[i - 1]) + σ * (t - A.t[i - 1])^2 + Cᵢ, i + idx = min(max(2, searchsortedfirstcorrelated(A.t, t, iguess)), length(A.t)) + Cᵢ = A.u[idx - 1] + σ = 1 // 2 * (A.z[idx] - A.z[idx - 1]) / (A.t[idx] - A.t[idx - 1]) + return A.z[idx - 1] * (t - A.t[idx - 1]) + σ * (t - A.t[idx - 1])^2 + Cᵢ, idx end # CubicSpline Interpolation function _interpolate(A::CubicSpline{<:AbstractVector}, t::Number, iguess) - i = max(1, min(searchsortedlastcorrelated(A.t, t, iguess), length(A.t) - 1)) - I = A.z[i] * (A.t[i + 1] - t)^3 / (6A.h[i + 1]) + - A.z[i + 1] * (t - A.t[i])^3 / (6A.h[i + 1]) - C = (A.u[i + 1] / A.h[i + 1] - A.z[i + 1] * A.h[i + 1] / 6) * (t - A.t[i]) - D = (A.u[i] / A.h[i + 1] - A.z[i] * A.h[i + 1] / 6) * (A.t[i + 1] - t) - I + C + D, i + idx = max(1, min(searchsortedlastcorrelated(A.t, t, iguess), length(A.t) - 1)) + A.idx_prev[] = idx + I = A.z[idx] * (A.t[idx + 1] - t)^3 / (6A.h[idx + 1]) + + A.z[idx + 1] * (t - A.t[idx])^3 / (6A.h[idx + 1]) + C = (A.u[idx + 1] / A.h[idx + 1] - A.z[idx + 1] * A.h[idx + 1] / 6) * (t - A.t[idx]) + D = (A.u[idx] / A.h[idx + 1] - A.z[idx] * A.h[idx + 1] / 6) * (A.t[idx + 1] - t) + I + C + D, idx end # BSpline Curve Interpolation diff --git a/src/interpolation_utils.jl b/src/interpolation_utils.jl index 1b416ab5..347a6af5 100644 --- a/src/interpolation_utils.jl +++ b/src/interpolation_utils.jl @@ -41,11 +41,12 @@ function munge_data(u::AbstractVector, t::AbstractVector) Tu = Base.nonmissingtype(eltype(u)) Tt = Base.nonmissingtype(eltype(t)) @assert length(t) == length(u) - non_missing_indices = collect(i - for i in 1:length(t) - if !ismissing(u[i]) && !ismissing(t[i])) - newu = Tu.([u[i] for i in non_missing_indices]) - newt = Tt.([t[i] for i in non_missing_indices]) + non_missing_indices = collect( + idx for idx in 1:length(t) + if !ismissing(u[idx]) && !ismissing(t[idx]) + ) + newu = Tu.([u[idx] for idx in non_missing_indices]) + newt = Tt.([t[idx] for idx in non_missing_indices]) return newu, newt end @@ -54,11 +55,12 @@ function munge_data(U::StridedMatrix, t::AbstractVector) TU = Base.nonmissingtype(eltype(U)) Tt = Base.nonmissingtype(eltype(t)) @assert length(t) == size(U, 2) - non_missing_indices = collect(i - for i in 1:length(t) - if !any(ismissing, U[:, i]) && !ismissing(t[i])) - newUs = [TU.(U[:, i]) for i in non_missing_indices] - newt = Tt.([t[i] for i in non_missing_indices]) + non_missing_indices = collect( + idx for idx in 1:length(t) + if !any(ismissing, U[:, idx]) && !ismissing(t[idx]) + ) + newUs = [TU.(U[:, idx]) for idx in non_missing_indices] + newt = Tt.([t[idx] for idx in non_missing_indices]) return hcat(newUs...), newt end From 8babf720428dbfee0af47af5f7281d3864fea501 Mon Sep 17 00:00:00 2001 From: Bart de Koning Date: Wed, 26 Jun 2024 08:01:46 +0200 Subject: [PATCH 2/5] Fix formatting --- src/derivatives.jl | 16 ++++++++++------ src/interpolation_caches.jl | 25 +++++++++++++------------ src/interpolation_utils.jl | 4 ++-- 3 files changed, 25 insertions(+), 20 deletions(-) diff --git a/src/derivatives.jl b/src/derivatives.jl index bd40e2ef..5e9599d4 100644 --- a/src/derivatives.jl +++ b/src/derivatives.jl @@ -8,10 +8,10 @@ function derivative(A, t, order = 1) val elseif order == 2 ForwardDiff.derivative(t -> begin - val, idx = _derivative(A, t, iguess) - A.idx_prev[] = idx - val - end, t) + val, idx = _derivative(A, t, iguess) + A.idx_prev[] = idx + val + end, t) else throw(DerivativeNotFoundError()) end @@ -117,8 +117,12 @@ function _derivative(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number) der end -_derivative(A::LagrangeInterpolation{<:AbstractVector}, t::Number, idx) = _derivative(A, t), idx -_derivative(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number, idx) = _derivative(A, t), idx +function _derivative(A::LagrangeInterpolation{<:AbstractVector}, t::Number, idx) + _derivative(A, t), idx +end +function _derivative(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number, idx) + _derivative(A, t), idx +end function _derivative(A::AkimaInterpolation{<:AbstractVector}, t::Number, iguess) idx = searchsortedfirstcorrelated(A.t, t, iguess) diff --git a/src/interpolation_caches.jl b/src/interpolation_caches.jl index 83047730..86c6be61 100644 --- a/src/interpolation_caches.jl +++ b/src/interpolation_caches.jl @@ -96,8 +96,9 @@ struct LagrangeInterpolation{uType, tType, T, bcacheType} <: t, n, bcache, - extrapolate, - Ref(1)) + extrapolate, + Ref(1) + ) end end @@ -141,8 +142,8 @@ struct AkimaInterpolation{uType, tType, bType, cType, dType, T} <: c, d, extrapolate, - Ref(1), - ) + Ref(1) + ) end end @@ -236,8 +237,8 @@ struct QuadraticSpline{uType, tType, tAType, dType, zType, T} <: d, z, extrapolate, - Ref(1), - ) + Ref(1) + ) end end @@ -304,8 +305,8 @@ struct CubicSpline{uType, tType, hType, zType, T} <: AbstractInterpolation{T} h, z, extrapolate, - Ref(1), - ) + Ref(1) + ) end end @@ -398,8 +399,8 @@ struct BSplineInterpolation{uType, tType, pType, kType, cType, T} <: pVecType, knotVecType, extrapolate, - Ref(1), - ) + Ref(1) + ) end end @@ -520,8 +521,8 @@ struct BSplineApprox{uType, tType, pType, kType, cType, T} <: pVecType, knotVecType, extrapolate, - Ref(1), - ) + Ref(1) + ) end end diff --git a/src/interpolation_utils.jl b/src/interpolation_utils.jl index 347a6af5..3dca3404 100644 --- a/src/interpolation_utils.jl +++ b/src/interpolation_utils.jl @@ -43,7 +43,7 @@ function munge_data(u::AbstractVector, t::AbstractVector) @assert length(t) == length(u) non_missing_indices = collect( idx for idx in 1:length(t) - if !ismissing(u[idx]) && !ismissing(t[idx]) + if !ismissing(u[idx]) && !ismissing(t[idx]) ) newu = Tu.([u[idx] for idx in non_missing_indices]) newt = Tt.([t[idx] for idx in non_missing_indices]) @@ -57,7 +57,7 @@ function munge_data(U::StridedMatrix, t::AbstractVector) @assert length(t) == size(U, 2) non_missing_indices = collect( idx for idx in 1:length(t) - if !any(ismissing, U[:, idx]) && !ismissing(t[idx]) + if !any(ismissing, U[:, idx]) && !ismissing(t[idx]) ) newUs = [TU.(U[:, idx]) for idx in non_missing_indices] newt = Tt.([t[idx] for idx in non_missing_indices]) From 6bd1984cc63ce37837185814e211f8bad866aab9 Mon Sep 17 00:00:00 2001 From: Bart de Koning Date: Thu, 27 Jun 2024 19:27:43 +0200 Subject: [PATCH 3/5] Add tests --- test/derivative_tests.jl | 6 ++++++ test/interpolation_tests.jl | 30 ++++++++++++++++++++++++++++++ 2 files changed, 36 insertions(+) diff --git a/test/derivative_tests.jl b/test/derivative_tests.jl index 484c0330..d3abbdd2 100644 --- a/test/derivative_tests.jl +++ b/test/derivative_tests.jl @@ -1,4 +1,5 @@ using DataInterpolations, Test +using FindFirstFunctions: searchsortedfirstcorrelated using FiniteDifferences using DataInterpolations: derivative using Symbolics @@ -35,6 +36,11 @@ function test_derivatives(method, u, t; args = [], kwargs = [], name::String) adiff2 = derivative(func, _t, 2) @test isapprox(fdiff, adiff, atol = 1e-8) @test isapprox(fdiff2, adiff2, atol = 1e-8) + # Cached index + if hasproperty(func, :idx_prev) + @test abs(func.idx_prev[] - + searchsortedfirstcorrelated(func.t, _t, func.idx_prev[])) <= 1 + end end # t = t0 diff --git a/test/interpolation_tests.jl b/test/interpolation_tests.jl index 24811680..99c849e3 100644 --- a/test/interpolation_tests.jl +++ b/test/interpolation_tests.jl @@ -1,7 +1,18 @@ using DataInterpolations +using FindFirstFunctions: searchsortedfirstcorrelated using StableRNGs using Optim, ForwardDiff +function test_cached_index(A) + for t in range(first(A.t), last(A.t); length = 2 * length(A.t) - 1) + A(t) + if hasproperty(A, :idx_prev) + @test abs(A.idx_prev[] - + searchsortedfirstcorrelated(A.t, t, A.idx_prev[])) <= 2 + end + end +end + @testset "Linear Interpolation" begin for t in (1.0:10.0, 1.0collect(1:10)) u = 2.0collect(1:10) @@ -45,6 +56,7 @@ using Optim, ForwardDiff @test A(0) == [-2.0 0.0; -3.0 0.0; -4.0 0.0] @test A(3) == [4.0 6.0; 6.0 9.0; 8.0 12.0] @test A(5) == [8.0 10.0; 12.0 15.0; 16.0 20.0] + test_cached_index(A) # with NaNs (#113) u = [NaN, 1.0, 2.0, 3.0] @@ -168,6 +180,7 @@ end @test A(2.5) == 6.25 @test A(3.5) == 12.25 @test A(5.0) == 25 + test_cached_index(A) # backward-looking interpolation u = [1.0, 4.0, 9.0, 16.0] @@ -182,6 +195,7 @@ end @test A(2.5) == 6.25 @test A(3.5) == 12.25 @test A(5.0) == 25 + test_cached_index(A) # Test both forward and backward-looking quadratic interpolation u = [1.0, 4.5, 6.0, 2.0] @@ -204,6 +218,9 @@ end @test A_f(3.5) ≈ l₂ * u[2] + l₁ * u[3] + l₀ * u[4] @test A_b(3.5) ≈ l₂ * u[2] + l₁ * u[3] + l₀ * u[4] + test_cached_index(A_f) + test_cached_index(A_b) + # Matrix interpolation test u = [1.0 4.0 9.0 16.0; 1.0 4.0 9.0 16.0] A = QuadraticInterpolation(u, t; extrapolate = true) @@ -321,6 +338,7 @@ end @test A(8.6) ≈ 4.1796554159017080820603951 @test A(9.9) ≈ 3.4110386597938129327189927 @test A(10.0) ≈ 3.0 + test_cached_index(A) # Test extrapolation A = AkimaInterpolation(u, t; extrapolate = true) @@ -345,6 +363,7 @@ end @test A(3.5) == u[1] @test A(4.0) == u[1] @test A(4.5) == u[1] + test_cached_index(A) A = ConstantInterpolation(u, t; extrapolate = true) # dir=:left is default @test A(0.5) == u[1] @@ -356,6 +375,7 @@ end @test A(3.5) == u[3] @test A(4.0) == u[1] @test A(4.5) == u[1] + test_cached_index(A) end @testset "Matrix case" for u in [ @@ -372,6 +392,7 @@ end @test A(3.5) == u[:, 1] @test A(4.0) == u[:, 1] @test A(4.5) == u[:, 1] + test_cached_index(A) A = ConstantInterpolation(u, t; extrapolate = true) # dir=:left is default @test A(0.5) == u[:, 1] @@ -383,6 +404,7 @@ end @test A(3.5) == u[:, 3] @test A(4.0) == u[:, 1] @test A(4.5) == u[:, 1] + test_cached_index(A) end @testset "Vector of Vectors case" for u in [ @@ -398,6 +420,7 @@ end @test A(3.5) == u[4] @test A(4.0) == u[4] @test A(4.5) == u[4] + test_cached_index(A) A = ConstantInterpolation(u, t; extrapolate = true) # dir=:left is default @test A(0.5) == u[1] @@ -409,6 +432,7 @@ end @test A(3.5) == u[3] @test A(4.0) == u[4] @test A(4.5) == u[4] + test_cached_index(A) end @testset "Vector of Matrices case" for u in [ @@ -424,6 +448,7 @@ end @test A(3.5) == u[4] @test A(4.0) == u[4] @test A(4.5) == u[4] + test_cached_index(A) A = ConstantInterpolation(u, t; extrapolate = true) # dir=:left is default @test A(0.5) == u[1] @@ -435,6 +460,7 @@ end @test A(3.5) == u[3] @test A(4.0) == u[4] @test A(4.5) == u[4] + test_cached_index(A) end # Test extrapolation @@ -464,6 +490,7 @@ end @test A(-0.5) == P₁(-0.5) @test A(0.7) == P₂(0.7) @test A(2.0) == P₂(2.0) + test_cached_index(A) u_ = [0.0, 1.0, 3.0]' .* ones(4) u = [u_[:, i] for i in 1:size(u_, 2)] @@ -496,6 +523,7 @@ end t = [-1.0, 0.0, 1.0] A = CubicSpline(u, t; extrapolate = true) + test_cached_index(A) # Solution P₁ = x -> 1 + 1.5x + 0.75 * x^2 + 0.25 * x^3 # for x ∈ [-1.0, 0.0] @@ -552,6 +580,7 @@ end @test [A(25.0), A(80.0)] == [13.454197730061425, 10.305633616059845] @test [A(190.0), A(225.0)] == [14.07428439395079, 11.057784141519251] @test [A(t[1]), A(t[end])] == [u[1], u[end]] + test_cached_index(A) # Test extrapolation A = BSplineInterpolation(u, t, 2, :Uniform, :Uniform; extrapolate = true) @@ -582,6 +611,7 @@ end @test [A(25.0), A(80.0)] ≈ [12.979802931218234, 10.914310609953178] @test [A(190.0), A(225.0)] ≈ [13.851245975109263, 12.963685868886575] @test [A(t[1]), A(t[end])] ≈ [u[1], u[end]] + test_cached_index(A) # Test extrapolation A = BSplineApprox(u, t, 2, 4, :Uniform, :Uniform; extrapolate = true) From d72c9fb78845c8642cebd76fb85d073317654e84 Mon Sep 17 00:00:00 2001 From: Bart de Koning Date: Fri, 28 Jun 2024 08:58:30 +0200 Subject: [PATCH 4/5] Comments adressed --- src/derivatives.jl | 30 +++++++----------------------- src/interpolation_methods.jl | 36 ++++++++++++++---------------------- src/interpolation_utils.jl | 9 +++++++++ test/interpolation_tests.jl | 7 +++---- 4 files changed, 33 insertions(+), 49 deletions(-) diff --git a/src/derivatives.jl b/src/derivatives.jl index 5e9599d4..3fb10356 100644 --- a/src/derivatives.jl +++ b/src/derivatives.jl @@ -18,18 +18,12 @@ function derivative(A, t, order = 1) end function _derivative(A::LinearInterpolation{<:AbstractVector}, t::Number, iguess) - idx = searchsortedfirstcorrelated(A.t, t, iguess) - idx > length(A.t) ? idx -= 1 : nothing - idx -= 1 - idx == 0 ? idx += 1 : nothing + idx = get_idx(A.t, t, iguess; idx_shift = -1, ub_shift = -2, side = :first) (A.u[idx + 1] - A.u[idx]) / (A.t[idx + 1] - A.t[idx]), idx end function _derivative(A::LinearInterpolation{<:AbstractMatrix}, t::Number, iguess) - idx = searchsortedfirstcorrelated(A.t, t, iguess) - idx > length(A.t) ? idx -= 1 : nothing - idx -= 1 - idx == 0 ? idx += 1 : nothing + idx = get_idx(A.t, t, iguess; idx_shift = -1, ub_shift = -2, side = :first) (@views @. (A.u[:, idx + 1] - A.u[:, idx]) / (A.t[idx + 1] - A.t[idx])), idx end @@ -125,10 +119,7 @@ function _derivative(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number, idx) end function _derivative(A::AkimaInterpolation{<:AbstractVector}, t::Number, iguess) - idx = searchsortedfirstcorrelated(A.t, t, iguess) - idx > length(A.t) ? idx -= 1 : nothing - idx -= 1 - idx == 0 ? idx += 1 : nothing + idx = get_idx(A.t, t, iguess; idx_shift = -1, side = :first) j = min(idx, length(A.c)) # for smooth derivative at A.t[end] wj = t - A.t[idx] (@evalpoly wj A.b[idx] 2A.c[j] 3A.d[j]), idx @@ -146,19 +137,14 @@ end # QuadraticSpline Interpolation function _derivative(A::QuadraticSpline{<:AbstractVector}, t::Number, iguess) - idx = searchsortedfirstcorrelated(A.t, t, iguess) - idx > length(A.t) ? idx -= 1 : nothing - idx == 1 ? idx += 1 : nothing + idx = get_idx(A.t, t, iguess; lb = 2, ub_shift = 0, side = :first) σ = 1 // 2 * (A.z[idx] - A.z[idx - 1]) / (A.t[idx] - A.t[idx - 1]) A.z[idx - 1] + 2σ * (t - A.t[idx - 1]), idx end # CubicSpline Interpolation function _derivative(A::CubicSpline{<:AbstractVector}, t::Number, iguess) - idx = searchsortedfirstcorrelated(A.t, t, iguess) - idx > length(A.t) ? idx -= 1 : nothing - idx -= 1 - idx == 0 ? idx += 1 : nothing + idx = get_idx(A.t, t, iguess) dI = -3A.z[idx] * (A.t[idx + 1] - t)^2 / (6A.h[idx + 1]) + 3A.z[idx + 1] * (t - A.t[idx])^2 / (6A.h[idx + 1]) dC = A.u[idx + 1] / A.h[idx + 1] - A.z[idx + 1] * A.h[idx + 1] / 6 @@ -170,8 +156,7 @@ function _derivative(A::BSplineInterpolation{<:AbstractVector{<:Number}}, t::Num # change t into param [0 1] t < A.t[1] && return zero(A.u[1]), 1 t > A.t[end] && return zero(A.u[end]), lastindex(t) - idx = searchsortedlastcorrelated(A.t, t, iguess) - idx == length(A.t) ? idx -= 1 : nothing + idx = get_idx(A.t, t, iguess) n = length(A.t) scale = (A.p[idx + 1] - A.p[idx]) / (A.t[idx + 1] - A.t[idx]) t_ = A.p[idx] + (t - A.t[idx]) * scale @@ -192,8 +177,7 @@ function _derivative(A::BSplineApprox{<:AbstractVector{<:Number}}, t::Number, ig # change t into param [0 1] t < A.t[1] && return zero(A.u[1]), 1 t > A.t[end] && return zero(A.u[end]), lastindex(t) - idx = searchsortedlastcorrelated(A.t, t, iguess) - idx == length(A.t) ? idx -= 1 : nothing + idx = get_idx(A.t, t, iguess) scale = (A.p[idx + 1] - A.p[idx]) / (A.t[idx + 1] - A.t[idx]) t_ = A.p[idx] + (t - A.t[idx]) * scale N = spline_coefficients(A.h, A.d - 1, A.k, t_) diff --git a/src/interpolation_methods.jl b/src/interpolation_methods.jl index 0b48963e..85344c93 100644 --- a/src/interpolation_methods.jl +++ b/src/interpolation_methods.jl @@ -14,7 +14,7 @@ function _interpolate(A::LinearInterpolation{<:AbstractVector}, t::Number, igues t1 = t2 = one(eltype(A.t)) u1 = u2 = one(eltype(A.u)) else - idx = max(1, min(searchsortedlastcorrelated(A.t, t, iguess), length(A.t) - 1)) + idx = get_idx(A.t, t, iguess) t1, t2 = A.t[idx], A.t[idx + 1] u1, u2 = A.u[idx], A.u[idx + 1] end @@ -27,7 +27,7 @@ function _interpolate(A::LinearInterpolation{<:AbstractVector}, t::Number, igues end function _interpolate(A::LinearInterpolation{<:AbstractMatrix}, t::Number, iguess) - idx = max(1, min(searchsortedlastcorrelated(A.t, t, iguess), length(A.t) - 1)) + idx = get_idx(A.t, t, iguess) θ = (t - A.t[idx]) / (A.t[idx + 1] - A.t[idx]) return (1 - θ) * A.u[:, idx] + θ * A.u[:, idx + 1], idx end @@ -35,9 +35,7 @@ end # Quadratic Interpolation _quad_interp_indices(A, t) = _quad_interp_indices(A, t, firstindex(A.t) - 1) function _quad_interp_indices(A::QuadraticInterpolation, t::Number, iguess) - inner_idx = searchsortedlastcorrelated(A.t, t, iguess) - A.mode == :Backward && (inner_idx -= 1) - idx = max(1, min(inner_idx, length(A.t) - 2)) + idx = get_idx(A.t, t, iguess; idx_shift = A.mode == :Backward ? -1 : 0, ub_shift = -2) idx, idx + 1, idx + 2 end @@ -125,7 +123,7 @@ function _interpolate(A::LagrangeInterpolation{<:AbstractMatrix}, t::Number, idx end function _interpolate(A::AkimaInterpolation{<:AbstractVector}, t::Number, iguess) - idx = max(1, min(searchsortedlastcorrelated(A.t, t, iguess), length(A.t) - 1)) + idx = get_idx(A.t, t, iguess) wj = t - A.t[idx] (@evalpoly wj A.u[idx] A.b[idx] A.c[idx] A.d[idx]), idx end @@ -134,30 +132,28 @@ end function _interpolate(A::ConstantInterpolation{<:AbstractVector}, t::Number, iguess) if A.dir === :left # :left means that value to the left is used for interpolation - idx = max(1, searchsortedlastcorrelated(A.t, t, iguess)) - return A.u[idx], idx + idx = get_idx(A.t, t, iguess; lb = 1, ub_shift = 0) else # :right means that value to the right is used for interpolation - idx = min(length(A.t), searchsortedfirstcorrelated(A.t, t, iguess)) - return A.u[idx], idx + idx = get_idx(A.t, t, iguess; side = :first, lb = 1, ub_shift = 0) end + A.u[idx], idx end function _interpolate(A::ConstantInterpolation{<:AbstractMatrix}, t::Number, iguess) if A.dir === :left # :left means that value to the left is used for interpolation - idx = max(1, searchsortedlastcorrelated(A.t, t, iguess)) - return A.u[:, idx], idx + idx = get_idx(A.t, t, iguess; lb = 1, ub_shift = 0) else # :right means that value to the right is used for interpolation - idx = min(length(A.t), searchsortedfirstcorrelated(A.t, t, iguess)) - return A.u[:, idx], idx + idx = get_idx(A.t, t, iguess; side = :first, lb = 1, ub_shift = 0) end + A.u[:, idx], idx end # QuadraticSpline Interpolation function _interpolate(A::QuadraticSpline{<:AbstractVector}, t::Number, iguess) - idx = min(max(2, searchsortedfirstcorrelated(A.t, t, iguess)), length(A.t)) + idx = get_idx(A.t, t, iguess; lb = 2, ub_shift = 0, side = :first) Cᵢ = A.u[idx - 1] σ = 1 // 2 * (A.z[idx] - A.z[idx - 1]) / (A.t[idx] - A.t[idx - 1]) return A.z[idx - 1] * (t - A.t[idx - 1]) + σ * (t - A.t[idx - 1])^2 + Cᵢ, idx @@ -165,8 +161,7 @@ end # CubicSpline Interpolation function _interpolate(A::CubicSpline{<:AbstractVector}, t::Number, iguess) - idx = max(1, min(searchsortedlastcorrelated(A.t, t, iguess), length(A.t) - 1)) - A.idx_prev[] = idx + idx = get_idx(A.t, t, iguess) I = A.z[idx] * (A.t[idx + 1] - t)^3 / (6A.h[idx + 1]) + A.z[idx + 1] * (t - A.t[idx])^3 / (6A.h[idx + 1]) C = (A.u[idx + 1] / A.h[idx + 1] - A.z[idx + 1] * A.h[idx + 1] / 6) * (t - A.t[idx]) @@ -181,8 +176,7 @@ function _interpolate(A::BSplineInterpolation{<:AbstractVector{<:Number}}, t < A.t[1] && return A.u[1], 1 t > A.t[end] && return A.u[end], lastindex(t) # change t into param [0 1] - idx = searchsortedlastcorrelated(A.t, t, iguess) - idx == length(A.t) ? idx -= 1 : nothing + idx = get_idx(A.t, t, iguess) t = A.p[idx] + (t - A.t[idx]) / (A.t[idx + 1] - A.t[idx]) * (A.p[idx + 1] - A.p[idx]) n = length(A.t) N = spline_coefficients(n, A.d, A.k, t) @@ -198,10 +192,8 @@ function _interpolate(A::BSplineApprox{<:AbstractVector{<:Number}}, t::Number, i t < A.t[1] && return A.u[1], 1 t > A.t[end] && return A.u[end], lastindex(t) # change t into param [0 1] - idx = searchsortedlastcorrelated(A.t, t, iguess) - idx == length(A.t) ? idx -= 1 : nothing + idx = get_idx(A.t, t, iguess) t = A.p[idx] + (t - A.t[idx]) / (A.t[idx + 1] - A.t[idx]) * (A.p[idx + 1] - A.p[idx]) - n = length(A.t) N = spline_coefficients(A.h, A.d, A.k, t) ucum = zero(eltype(A.u)) for i in 1:(A.h) diff --git a/src/interpolation_utils.jl b/src/interpolation_utils.jl index 3dca3404..3bab4257 100644 --- a/src/interpolation_utils.jl +++ b/src/interpolation_utils.jl @@ -64,3 +64,12 @@ function munge_data(U::StridedMatrix, t::AbstractVector) return hcat(newUs...), newt end + +function get_idx(tvec, t, iguess; lb = 1, ub_shift = -1, idx_shift = 0, side = :last) + ub = length(tvec) + ub_shift + return if side == :last + clamp(searchsortedlastcorrelated(tvec, t, iguess) + idx_shift, lb, ub) + else + clamp(searchsortedfirstcorrelated(tvec, t, iguess) + idx_shift, lb, ub) + end +end diff --git a/test/interpolation_tests.jl b/test/interpolation_tests.jl index 6118be06..a7c4eb2e 100644 --- a/test/interpolation_tests.jl +++ b/test/interpolation_tests.jl @@ -6,10 +6,9 @@ using Optim, ForwardDiff function test_cached_index(A) for t in range(first(A.t), last(A.t); length = 2 * length(A.t) - 1) A(t) - if hasproperty(A, :idx_prev) - @test abs(A.idx_prev[] - - searchsortedfirstcorrelated(A.t, t, A.idx_prev[])) <= 2 - end + idx = searchsortedfirstcorrelated(A.t, t, A.idx_prev[]) + @test abs(A.idx_prev[] - + searchsortedfirstcorrelated(A.t, t, A.idx_prev[])) <= 2 end end From 8da6f06cc962a450c0cbe03fe6c4961f4746db66 Mon Sep 17 00:00:00 2001 From: Bart de Koning Date: Fri, 28 Jun 2024 10:02:33 +0200 Subject: [PATCH 5/5] comment adressed --- src/interpolation_utils.jl | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/interpolation_utils.jl b/src/interpolation_utils.jl index 3bab4257..bb62e78a 100644 --- a/src/interpolation_utils.jl +++ b/src/interpolation_utils.jl @@ -69,7 +69,9 @@ function get_idx(tvec, t, iguess; lb = 1, ub_shift = -1, idx_shift = 0, side = : ub = length(tvec) + ub_shift return if side == :last clamp(searchsortedlastcorrelated(tvec, t, iguess) + idx_shift, lb, ub) - else + elseif side == :first clamp(searchsortedfirstcorrelated(tvec, t, iguess) + idx_shift, lb, ub) + else + error("side must be :first or :last") end end