Getting Started

First, let's load the package along with the StaticArrays.jl package which we use for each data point

using RadialBasisFunctions
using StaticArrays

Interpolation

Suppose we have a set of data $\mathbf{x}$ where $\mathbf{x}_i \in \mathbb{R}^2$, and we want to interpolate a function $f:\mathbb{R}^2 \rightarrow \mathbb{R}$

f(x) = 2*x[1]^2 + 3*x[2]
x = [SVector{2}(rand(2)) for _ in 1:1000]
y = f.(x)
1000-element Vector{Float64}:
 3.414138724708506
 3.9682422219644424
 3.4452597205593056
 1.9637389321092669
 2.03906326935385
 1.7280227402207728
 2.923788394917044
 2.3967082943616553
 1.299960986325814
 2.6689316046364064
 ⋮
 1.9593324673312928
 2.1796196313571023
 2.037125583073049
 2.628399991299812
 3.132616322943325
 3.0870813042761838
 2.7725158394907217
 1.2862342363476673
 2.435657029871568

and now we can build the interpolator

interp = Interpolator(x, y)
Interpolator
├─Input type: StaticArraysCore.SVector{2, Float64}
├─Output type: Float64
├─Number of points: 1000
└─Basis: Polyharmonic spline (r³) with degree 2 Monomial

and evaluate it at a new point

x_new = [rand(2) for _ in 1:5]
y_new = interp(x_new)
y_true = f.(x_new)
5-element Vector{Float64}:
 2.8615727350768903
 2.700010271270799
 3.713269573263843
 3.2674119994354665
 1.4756499097656015

and compare the error

abs.(y_true .- y_new)
5-element Vector{Float64}:
 2.709683339929825e-7
 0.1069314665994634
 1.1996301907402085e-7
 4.040314487419039e-7
 4.042177481622389e-9

Wow! The error is numerically zero! Well... we set ourselves up for success here. Interpolator (along with RadialBasisOperator) has an optional argument to provide the type of radial basis including the degree of polynomial augmentation. The default basis is a cubic polyharmonic spline with 2nd degree polynomial augmentation (which the constructor is PHS(3, poly_deg=2)) and given the underlying function we are interpolating is a 2nd order polynomial itself, we are able to represent it exactly (up to machine precision). Let's see what happens when we only use 1st order polynomial augmentation

interp = Interpolator(x, y, PHS(3, poly_deg=1))
y_new = interp(x_new)
abs.(y_true .- y_new)
5-element Vector{Float64}:
 2.485863590528936e-7
 0.10693130954050645
 8.526584238666146e-8
 2.6863373170371574e-7
 1.1925893605990723e-9

Operators

This package also provides an API for operators. There is support for several built-in operators along with support for user-defined operators. Currently, we have implementations for

  • partial derivative (1st and 2nd order)
  • laplacian
  • gradient

but we plan to add more in the future. Please make and issue or pull request for additional operators.

Partial Derivative

We can take the same data as above and build a partial derivative operator with a similar construction as the interpolator. For the partial we need to specify the order of differentiation we want along with the dimension for which to take the partial. We can also supply some optional arguments such as the basis and number of points in the stencil. The function inputs order is partial(x, order, dim, basis; k=5)

df_x_rbf = partial(x, 1, 1)

# define exact
df_x(x) = 4*x[1]

# error
all(abs.(df_x.(x) .- df_x_rbf(y)) .< 1e-10)
true

Laplacian

Building a laplacian operator is as easy as

lap_rbf = laplacian(x)

# define exact
lap(x) = 4

# error
all(abs.(lap.(x) .- lap_rbf(y)) .< 1e-8)
true

Gradient

We can also retrieve the gradient. This is really just a convenience wrapper around Partial.

grad = gradient(x)

# define exacts
df_x(x) = 4*x[1]
df_y(x) = 3

# error
all(df_x.(x) .≈ grad(y)[1])
true
all(df_y.(x) .≈ grad(y)[2])
true