Skip to main content

A Code for Memory-Saving Dyadic Adaptivity in Optimization and Simulation

Project description

DyAda: A Code for Dyadic Adaptivity in Optimization, Simulation, and Machine Learning

PyPI version Supported Python version Python package CI Coverage Codacy Badge License: GPL v3

Installation

It's as simple as

pip install dyada[drawing,matplotlib,opengl]

Or, if you would like to change the source code, do

git clone https://github.com/freifrauvonbleifrei/DyAda.git
cd DyAda
# ... git checkout the required version ...
pip install -e .[drawing,matplotlib,opengl]

Dyadic Adaptivity

Dyadic adaptivity means: A given hypercube of 2 or more dimensions may or may not be subdivided into two parts in any number of dimensions. Of the resulting sub-boxes, each may again be subdivided into two in any dimension, and so forth.

Why Dyadic Adaptivity?

Currently, the most common approach to adaptivity are octrees, which are a special type of dyadic adaptivity: Each box is either refined in every dimension or not at all. For a three-d domain, the tree and the resulting partitioning could look like this:

The octree tree

The octree partitioning

But maybe you didn't need all this resolution?

Maybe, in the finely-resolved areas, you only needed only some of the dimensions resolved finely:

The dyadic partitioning

This is what DyAda provides.

The tree will then look like this:

The omnitree tree

And you can use only 14 degrees of freedom instead of 29! (Count the number of colorful tree nodes to check!) This reduction will be even stronger if you go to higher dimensions.

For details, refer to the preprint, with animations!

Using DyAda

For a quick overview, the following example sticks to only two-dimensional discretizations, but all algorithms work on (almost) arbitrary-dimensional omnitrees, though DyAda may become slow for too many dimensions. (Find the full tutorial code in dyada_tutorial.py, and more examples of usage in the extensive test suite in /test.)

You can start with a regular RefinementDescriptor:

import bitarray as ba
import dyada
from random import randint

# %%
descriptor = dyada.RefinementDescriptor(2, [2, 1])
# dyada.plot_tree_tikz(descriptor, filename="simple_tree")
num_dimensions = descriptor.get_num_dimensions()
print(descriptor)

Expected output:

RefinementDescriptor('11 01 00 00 ...0 00 01 00 00')

This one has four rectangles in the first dimension and two on the second, because the level [2, 1] is passed as base-2 exponents. If you uncomment the line with plot_tree_tikz and you have latexmk and some LaTeX tikz packages installed, the script will generate a simple_tree.pdf in the same folder.

You can use the descriptor and MortonOrderLinearization to build a Discretization:

discretization = dyada.Discretization(dyada.MortonOrderLinearization(), descriptor)
print("initial discretization:")
print(discretization)

->

initial discretization:
_________
|_|_|_|_|
|_|_|_|_|

If you want to refine a single rectangle at once, you can use apply_single_refinement:

new_discretization, index_mapping = dyada.apply_single_refinement(
    discretization, 0, track_mapping="boxes"
)
print("after refining box 0:")
print(new_discretization)

->

after refining box 0:
_________________
|   |   |   |   |
|___|___|___|___|
|_|_|   |   |   |
|_|_|___|___|___|

Of course, you can also refine only in a subset of the dimensions:

# select random index and refinement
random_index = randint(0, new_discretization.descriptor.get_num_boxes() - 1)
random_refinement = ba.bitarray("00")
while random_refinement.count() == 0:
    random_refinement = ba.bitarray(
        "".join(str(randint(0, 1)) for _ in range(num_dimensions))
    )
new_discretization, index_mapping = dyada.apply_single_refinement(
    new_discretization, random_index, random_refinement, track_mapping="boxes"
)
print("after refining random box:")
print(new_discretization)

->

after refining random box:
_________________
|   |___|   |   |
|___|___|___|___|
|_|_|   |   |   |
|_|_|___|___|___|

You can keep running the above and watch your discretization become finer and finer!

To refine many rectangles at once, you can collect the refinements as PlannedAdaptiveRefinement object:

refining = dyada.PlannedAdaptiveRefinement(discretization)
refining.plan_refinement(0, ba.bitarray("11"))
refining.plan_refinement(1, ba.bitarray("01"))
new_discretization, index_mapping = refining.apply_refinements(track_mapping="boxes")
# dyada.plot_all_boxes_2d(new_discretization, backend="matplotlib", labels="boxes")
print("after applying planned refinements:")
print(new_discretization)

->

after applying planned refinements:
_________________
|   |   |   |   |
|___|___|___|___|
|_|_| | |   |   |
|_|_|_|_|___|___|

If you uncomment the plot_all_boxes_2d, it will show you the discretization as matplotlib. Other backends are tikz, ascii(only 2d), and opengl (only 3d).

Note that dyada does not store your function data; you have to manage your own container (for example, a numpy array) to do that. But the index_mapping in the above snippets helps you figure out how your function data has moved: new_indices = index_mapping[old_index].

For a full workflow based on dyada, have a look at the project thingies_with_omnitrees.

Contributing

Feel free to request features or voice your intent to work on/with DyAda as an issue. Depending on what you are looking for, exciting features may be in preparation, or they may just be waiting for you to implement them!

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

dyada-0.0.12.tar.gz (130.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

dyada-0.0.12-py3-none-any.whl (50.3 kB view details)

Uploaded Python 3

File details

Details for the file dyada-0.0.12.tar.gz.

File metadata

  • Download URL: dyada-0.0.12.tar.gz
  • Upload date:
  • Size: 130.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.12.9

File hashes

Hashes for dyada-0.0.12.tar.gz
Algorithm Hash digest
SHA256 11cf3d0174068bbdc38c6500e8787aa8ce748f939e622e712af2dea9c2ee2365
MD5 aa77061110ba957d03960c8b3382eed2
BLAKE2b-256 a4d3a528580aa8d72d4cb5a0ed9a8dc36aeaab146664f0948faaa9e30ecfe6a4

See more details on using hashes here.

Provenance

The following attestation bundles were made for dyada-0.0.12.tar.gz:

Publisher: publish-to-pypi.yml on freifrauvonbleifrei/DyAda

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file dyada-0.0.12-py3-none-any.whl.

File metadata

  • Download URL: dyada-0.0.12-py3-none-any.whl
  • Upload date:
  • Size: 50.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.12.9

File hashes

Hashes for dyada-0.0.12-py3-none-any.whl
Algorithm Hash digest
SHA256 4d3e7aa37f7977d2b3c269a3ae5c25ec9198e32dcda16aa55f1bb96fd698f1f4
MD5 f2da4ccec719bec5af9e135ae36561ae
BLAKE2b-256 95ee5888e8cff85b379eea3c11b4e8dd5329aed3577554e0d5df723f61d17bb1

See more details on using hashes here.

Provenance

The following attestation bundles were made for dyada-0.0.12-py3-none-any.whl:

Publisher: publish-to-pypi.yml on freifrauvonbleifrei/DyAda

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page