docs/source/tutorial/tutorial.custom_loss.rst
6fca822a
 Custom adaptive logic for 1D and 2D
 -----------------------------------
 
 .. note::
    Because this documentation consists of static html, the ``live_plot``
    and ``live_info`` widget is not live. Download the notebook
    in order to see the real behaviour.
 
 .. seealso::
     The complete source code of this tutorial can be found in
5e3edf47
     :jupyter-download:notebook:`tutorial.custom-loss`
6fca822a
 
d9aaa498
 .. jupyter-execute::
6fca822a
     :hide-code:
 
     import adaptive
6b170238
     adaptive.notebook_extension(_inline_js=False)
6fca822a
 
     # Import modules that are used in multiple cells
     import numpy as np
     from functools import partial
 
 
 `~adaptive.Learner1D` and `~adaptive.Learner2D` both work on the principle of
 subdividing their domain into subdomains, and assigning a property to
 each subdomain, which we call the *loss*. The algorithm for choosing the
 best place to evaluate our function is then simply *take the subdomain
 with the largest loss and add a point in the center, creating new
 subdomains around this point*.
 
 The *loss function* that defines the loss per subdomain is the canonical
 place to define what regions of the domain are “interesting”. The
 default loss function for `~adaptive.Learner1D` and `~adaptive.Learner2D` is sufficient
 for a wide range of common cases, but it is by no means a panacea. For
 example, the default loss function will tend to get stuck on
 divergences.
 
 Both the `~adaptive.Learner1D` and `~adaptive.Learner2D` allow you to specify a *custom
 loss function*. Below we illustrate how you would go about writing your
 own loss function. The documentation for `~adaptive.Learner1D` and `~adaptive.Learner2D`
 specifies the signature that your loss function needs to have in order
 for it to work with ``adaptive``.
 
 tl;dr, one can use the following *loss functions* that
 **we** already implemented:
 
 + `adaptive.learner.learner1D.default_loss`
 + `adaptive.learner.learner1D.uniform_loss`
b00f266f
 + `adaptive.learner.learner1D.curvature_loss_function`
6fca822a
 + `adaptive.learner.learner2D.default_loss`
 + `adaptive.learner.learner2D.uniform_loss`
 + `adaptive.learner.learner2D.minimize_triangle_surface_loss`
d5a63a3f
 + `adaptive.learner.learner2D.resolution_loss_function`
6fca822a
 
b00f266f
 Whenever a loss function has `_function` appended to its name, it is a factory function
 that returns the loss function with certain settings.
6fca822a
 
 Uniform sampling
 ~~~~~~~~~~~~~~~~
 
 Say we want to properly sample a function that contains divergences. A
 simple (but naive) strategy is to *uniformly* sample the domain:
 
d9aaa498
 .. jupyter-execute::
6fca822a
 
84cba2ee
     def uniform_sampling_1d(xs, ys):
         dx = xs[1] - xs[0]
6fca822a
         return dx
 
     def f_divergent_1d(x):
44c367f0
         if x == 0:
             return np.inf
6fca822a
         return 1 / x**2
 
     learner = adaptive.Learner1D(f_divergent_1d, (-1, 1), loss_per_interval=uniform_sampling_1d)
     runner = adaptive.BlockingRunner(learner, goal=lambda l: l.loss() < 0.01)
     learner.plot().select(y=(0, 10000))
 
d9aaa498
 .. jupyter-execute::
6fca822a
 
     %%opts EdgePaths (color='w') Image [logz=True colorbar=True]
 
     from adaptive.runner import SequentialExecutor
 
     def uniform_sampling_2d(ip):
         from adaptive.learner.learner2D import areas
         A = areas(ip)
         return np.sqrt(A)
 
     def f_divergent_2d(xy):
         x, y = xy
         return 1 / (x**2 + y**2)
 
     learner = adaptive.Learner2D(f_divergent_2d, [(-1, 1), (-1, 1)], loss_per_triangle=uniform_sampling_2d)
 
     # this takes a while, so use the async Runner so we know *something* is happening
     runner = adaptive.Runner(learner, goal=lambda l: l.loss() < 0.02)
 
d9aaa498
 .. jupyter-execute::
6fca822a
     :hide-code:
 
     await runner.task  # This is not needed in a notebook environment!
 
d9aaa498
 .. jupyter-execute::
6fca822a
 
     runner.live_info()
 
d9aaa498
 .. jupyter-execute::
6fca822a
 
     plotter = lambda l: l.plot(tri_alpha=0.3).relabel(
             '1 / (x^2 + y^2) in log scale')
     runner.live_plot(update_interval=0.2, plotter=plotter)
 
 The uniform sampling strategy is a common case to benchmark against, so
 the 1D and 2D versions are included in ``adaptive`` as
 `adaptive.learner.learner1D.uniform_loss` and
 `adaptive.learner.learner2D.uniform_loss`.
 
 Doing better
 ~~~~~~~~~~~~
 
 Of course, using ``adaptive`` for uniform sampling is a bit of a waste!
 
 Let’s see if we can do a bit better. Below we define a loss per
 subdomain that scales with the degree of nonlinearity of the function
 (this is very similar to the default loss function for `~adaptive.Learner2D`),
 but which is 0 for subdomains smaller than a certain area, and infinite
 for subdomains larger than a certain area.
 
 A loss defined in this way means that the adaptive algorithm will first
 prioritise subdomains that are too large (infinite loss). After all
 subdomains are appropriately small it will prioritise places where the
 function is very nonlinear, but will ignore subdomains that are too
 small (0 loss).
 
d9aaa498
 .. jupyter-execute::
6fca822a
 
     %%opts EdgePaths (color='w') Image [logz=True colorbar=True]
 
d5a63a3f
     def resolution_loss_function(min_distance=0, max_distance=1):
6fca822a
         """min_distance and max_distance should be in between 0 and 1
         because the total area is normalized to 1."""
d5a63a3f
         def resolution_loss(ip):
             from adaptive.learner.learner2D import default_loss, areas
             loss = default_loss(ip)
6fca822a
 
d5a63a3f
             A = areas(ip)
             # Setting areas with a small area to zero such that they won't be chosen again
             loss[A < min_distance**2] = 0
6fca822a
 
d5a63a3f
             # Setting triangles that have a size larger than max_distance to infinite loss
             loss[A > max_distance**2] = np.inf
6fca822a
 
d5a63a3f
             return loss
         return resolution_loss
     loss = resolution_loss_function(min_distance=0.01)
6fca822a
 
     learner = adaptive.Learner2D(f_divergent_2d, [(-1, 1), (-1, 1)], loss_per_triangle=loss)
     runner = adaptive.BlockingRunner(learner, goal=lambda l: l.loss() < 0.02)
     learner.plot(tri_alpha=0.3).relabel('1 / (x^2 + y^2) in log scale')
 
 Awesome! We zoom in on the singularity, but not at the expense of
 sampling the rest of the domain a reasonable amount.
 
 The above strategy is available as
d5a63a3f
 `adaptive.learner.learner2D.resolution_loss_function`.