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.10.tar.gz (124.7 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.10-py3-none-any.whl (47.2 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: dyada-0.0.10.tar.gz
  • Upload date:
  • Size: 124.7 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.10.tar.gz
Algorithm Hash digest
SHA256 8df7d16c9bf237f87de12af6413f3ca8d3a463f621a43f73a21fa4555fe55e5d
MD5 354ab56660b83082a1e65b3b1bc7c60f
BLAKE2b-256 2aacc7abfc26919cda9006ae269edc05669e979a92de3bb32bd7da9a8cfc26e4

See more details on using hashes here.

Provenance

The following attestation bundles were made for dyada-0.0.10.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.10-py3-none-any.whl.

File metadata

  • Download URL: dyada-0.0.10-py3-none-any.whl
  • Upload date:
  • Size: 47.2 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.10-py3-none-any.whl
Algorithm Hash digest
SHA256 2e71369f3ef1aaec4f9460a3a18830dcf3bc7031aa951a505d8dd1e8a5598fd6
MD5 488206942fa947f3cf6a31498b8c437a
BLAKE2b-256 8f4b759aaee5cec9008d2422a7ce7d016d668f446b92c0d8c02d646f1e8fb0c2

See more details on using hashes here.

Provenance

The following attestation bundles were made for dyada-0.0.10-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