Algorithms related with Mandelbrot set based on []the programs implemented in C by Claude Heiland-Allen](https://code.mathr.co.uk/mandelbrot-book).
Programs are minimal, complete and verifiable examples" (MCVE)
- independent
- 1-file
- short
- educational ( speed is not a main target)
- console ( without GUI)
- written in C programing language
TOC
- Grapical algorithms = methods of drawing/computing = representation finctions = coloring algorithms ( for all points of the choosen part of the plane)
- Numerical algorithms - how to compute points, curves ( = set of points) or numerical features ( like period) using numerical methods
- Symbolic algorithms - how to compute features using symbolic methods
- Technical notes
Grapical algorithms = methods of drawing/computing = representation finctions = coloring algorithms ( for all points of the choosen part of the plane)
- membership
- Escape Time
- Binary Decomposition
- Field lines
- Continuous Dwell
- Final angle
- Final radius
- Final radius
- Grid
- Improved grid
- Noise
- Gray grid
- Exterior coordinate
- Dwell gradient
- Exterior distance
- Atom domains
- Misiurewicz domains
- Interior coordinate
- Interior distance
- mu-atom mapping
- adaptive supersampling
Numerical algorithms - how to compute points, curves ( = set of points) or numerical features ( like period) using numerical methods
- Period
- Nucleus
- Attractor
- Interior point
- Misiurewicz point
- Child sizes
- Size estimate
- Shape estimate
- Domain size
- External ray in
- External ray out
Symbolic algorithms - how to compute features using symbolic methods
Colour according to whether the iterations diverge to infinity or not, within a fixed maximum number of iterations N and escape radius R
R >= 2
The disconnected specks visible in the image turn out to be copies of the whole, and despite appearances the Mandelbrot set is connected.
C99 Code
#include <complex.h>
#include <stdbool.h>
bool m_membership(int N, double R, double _Complex c)
{
double _Complex z = 0;
for (int n = 0; n < N; ++n)
{
if (cabs(z) > R)
return false;
z = z * z + c;
}
return true;
}
Code:
- membership_old.c almost original code,
- membership_new.c, new code, different method for computing
names and abreviations
- Escape Time Algorithm (ETA)
- Boolean/binary Escape Time Method ( bETM/M )
- Dwell is a colloquial name for the Representation Function called Escape-Iterations
See also *
Colour according to the integer number of iterations n at which
|zn| > R >= |z(n-1)|
where escape radius
R >= 2
C99 Code
#include <complex.h>
int m_escape_time(int N, double R, double _Complex c)
{
double _Complex z = 0;
for (int n = 0; n < N; ++n)
{
if (cabs(z) > R)
return n;
z = z * z + c;
}
return -1;
}
Code:
Binary decomposition of the Mandelbrot set exterior
Colour according to the sign of Im(z) (the first escaped iterate). Increasing escape radius is necessary to align edges between layers. Escape radius around ER = 25 makes the decomposed cells roughly square.
C99 Code = binary-decomposition.c
color of the pixel depends on:
- the sign of the imaginary part of final z ( cimag(z) )
- position of final z with repect to the horizontal axis ( x-axis)
img[j * w + i] = (k < kMax && cimag(z) < 0) ? 0 : 255;
External rays of angles (measured in turns)
- Let's remove the iteration bands and change the relative sizes of black and white checkerboard areas.
- The patterns (lines) were done by looking at the final value of the iterated Z_n, just after it escapes. The usual checkerboarded "binary decomposition" looks at just the sign of the imaginary part. But you really can choose any axis through the origin and color based on what side of the axis you end up on. Or you can choose a sector smaller than 180 degrees like I did: return (fabs(z.x)*0.1 < fabs(z.y));
- The bailout radius needs to be reasonably large for those field lines to align nicely between iteration bands. Because this analogy with field lines is only strictly true for an infinite bailout. So with a minimal bailout radius of 2.0, the binary decomposition ends up being very visibly distorted and misaligned.
color of the pixel depends on the position of final z
img[j * w + i] = (k < kMax && fabs(creal(z))*0.1 < fabs(cimag(z))) ? 255 : 0;
The image shows lines that aproximate external rays of angles (measured in turns)
Linas Vepstas derives the renormalized continuous escape time:
mu = n + 1 - log(log (|z|))/log(2)
In the above formula, the value of mu is almost completely independent of the iteration count, and of the escape radius, despite the fact that the right-hand-side of the equation is explicitly dependent on both of these quantities. The renormalized iteration count mu depends only on c, and is a piecewise-continuous, differentiable function thereof. By using a different analysis, it can be seen that the renormalized iteration count mu is in fact the residue remaining when a pole (due to the infinite sum) is removed. That is, the value of mu closely approximates the result of having iterated to infinity, that is, of having an infinite escape radius, and an infinite maximum iteration count.
C99 Code
#include <complex.h>
#include <math.h>
double m_continuous_dwell(int N, double R, double _Complex c)
{
double _Complex z = 0;
for (int n = 0; n < N; ++n)
{
if (cabs(z) > R)
return n + 1 - log2(log(cabs(z)));
z = z * z + c;
}
return -1;
}
colour function uses both sources of information now:
- hue coming from the escape time
- value from the final angle
With the escape radius at 512, the cells look almost square, and there are no visible gltiches.
c99 code: fa.c
colour function uses both sources of information now:
- hue coming from the escape time
- value from the final angle
- sat from final radius
c99 code: fr.c
if (n < nMax) // exterior
{
/* the name final_z_abs is confusing indeed, sorry.
It is the radial component of "exterior tiling" colouring, slightly different from Linas' renormalization.
why denominator = 64.0 ?
arbitrary, chosen to make the colouring look ok. try changing it as you like
*/
double final_z_abs = log(cabs(z)) / log(er) - 1.0;
// double final_z_abs = cabs(z);
int final_n = n;
double continuous_escape_time = final_n - log2(final_z_abs + 1.0);
hue = continuous_escape_time/64.0;
sat = 0.7;
val = 1.0;
}
c99 code: s.c
grid showing:
- how the escape time bands get closer together towards the boundary of the Mandelbrot set
- how the angle cells double-up within each successive band
c99 code: g.c
if (n < nMax) // exterior
{
double final_z_abs = log(cabs(z)) / log(er) - 1.0;
int final_n = n;
double continuous_escape_time = final_n - log2(final_z_abs + 1.0);
double final_z_arg = fmod( carg(z)/TwoPi + 1.0, 1.0);
int grid =
0.05 < final_z_abs &&
final_z_abs < 0.95 &&
0.05 < final_z_arg &&
final_z_arg < 0.95;
//
hue = continuous_escape_time/64.0;
sat = grid * 0.7;
val = 1.0;
}
The grid has noticeable step changes in the width of the rays heading towards the Mandelbrot set, because each successive escape time band has double the number of cells, so at the boundary between bands there is a discontinuity where the width halves. We can compensate for this by making the rays wider (in local coordinate terms) further out and thinner further in. The inside end should be twice as narrow as the outside end, to match up with the halving of the width when crossing band boundaries. We keep the middle the same width as before by ensuring the width factor is 1 there, which means the power should be 0 at the fractional radius 0.5. Now the rays of the grid seem to continue along a smooth path towards the Mandelbrot set.
c99 code: ig.c
if (n < nMax) // exterior
{
double final_z_abs = log(cabs(z)) / log(er) - 1.0;
int final_n = n;
double continuous_escape_time = final_n - log2(final_z_abs + 1.0);
double final_z_arg = fmod( carg(z)/TwoPi + 1.0, 1.0);
// improved grid
double k = pow ( 0.5 , 0.5 - final_z_abs ) ;
double grid_weight = 0.05 ;
int grid =
grid_weight < final_z_abs &&
final_z_abs < 1.0 - grid_weight &&
grid_weight * k < final_z_arg &&
final_z_arg < 1.0 - grid_weight * k;
//
hue = continuous_escape_time/64.0;
sat = grid * 0.7;
val = 1.0;
}
In the book it is : 1.31 Distance estimation.
Our images look noisy and grainy near the boundary of the Mandelbrot set. The escape time bands get closer and closer, while the pixel spacing is fixed. The pixel grid samples isolated points of a mathematically abstract image defined on the continuous plane. The Nyquist-Shannon sampling theorem shows that sampling isolated points from a continuum is a valid approximation only so long as the values don’t change too quickly between the points. Aliasing occurs when the values do change too quickly compared to the sampling rate, with the grainy noisy visual effects as we have seen. Because the escape time bands increase in number without bound as we approach the boundary of the Mandelbrot set, no sampling rate can be high enough.
We know the escape time bands get closer together, and we know this causes problems related to the pixel spacing. So, multiplying the derivative by the pixel spacing is a first step (the pixel spacing is in fact the derivative with respect to our image device coordinates). We know the problems occur when the bands are closer, meaning the variation is more rapid, so we divide something by this product. What we divide by will remain somewhat magical for now, involving the final z iterate, but our final result is a distance estimate for the Mandelbrot set scaled to image device coordinates.
The distance estimate has the useful property (proved in the Koebe 1/4 theorem) that it tells us roughly how far our point is from the boundary of the Mandelbrot set. We can use it to:
- mask the grainy noise from aliasing
- highlighting the intricate shape of the boundary of the Mandelbrot set with all its filaments and shrunken copies. It is the pleasing side-effect.
To contrast with the dark boundary, we fill the interior with white
c99 code: n.c
In the book it is : 1.32 Removing distracting colour.
The bold colours are now a distraction: they don’t show us anything we can’t see from the grid or the distance estimation. Desaturating the image and making the exterior grid a light grey gives us a clean style that is easy on the eyes.
c99 code: gg.c
Escape radius around R = 25^2 makes the exterior grid cells roughly square. The imaginary part of e is linearized to avoid shape distortion.
Re(e) = (arg(z)/2pi) mod 1
Im(e) = 2 - log(z) / log R
The exterior coordinate
dancing angels has an example of animating image texture during a zoom video.
#include <complex.h>
#include <math.h>
const double pi = 3.141592653589793;
double _Complex m_exterior_coordinates(int N, double R, double _Complex c)
{
double _Complex z = 0;
for (int n = 0; n < N; ++n)
{
if (cabs(z) > R)
return fmod(1 + carg(z) / (2 * pi), 1) + I * (2 - log(cabs(z)) / log(R));
z = z * z + c;
}
return 0;
}
v = (mu - mu_x, mu - mu_y, s)
g = v3 / |v|
where mu_x and mu_y are the continuous dwells of neighbouring pixels to
the above and left of the pixel under consideration, and
0 <= g <= 1
colours the boundary of the set where iteration count goes up increasingly quickly. Define mu to be large and negative for pixels that didn't escape.
Fake DE also describes using
tan^{-1} v_2/v_1
as hue for a rainbow colouring of dwell slope.
#include <complex.h>
#include <math.h>
const double pi = 3.141592653589793;
double _Complex m_dwell_gradient(int N, double R, double s, double d, double _Complex c)
{
double m = m_continuous_dwell(N, R, c);
double mx = m_continuous_dwell(N, R, c + d);
double my = m_continuous_dwell(N, R, c + d * I);
double vx = m - mx;
double vy = m - my;
double vz = s;
double vm = sqrt(vx * vx + vy * vy + vz * vz);
return vz / vm;
}
DEM/M = Distance Estimation Method for Mandelbrot set
Given c outside the Mandelbrot set, the exterior distance estimate satisfies by
d = 2*abs(z)*log|z|/|dz|
where dz is first derivative of zn with respect to c
Formalizes the dwell gradient idea, in that closeness of dwell bands is related to closeness to the boundary of the set.
the Koebe
- No point in the Mandelbrot set is within d/4
- There is a point in the Mandelbrot set within 4 d
Compare with pixel spacing to know if the Mandelbrot set might intersect a pixel.
Colouring (shading) using
A complex-valued distance estimate (with directional information) can be calculated by:
d = 2*z*log|z|/dz
Distance estimation and Circular wavefronts use distance estimates to give progressively finer approximations to the exterior, while adaptive super sampling describes an adaptive supersampling method.
C99 Code
#include <complex.h>
#include <math.h>
const double pi = 3.141592653589793;
double _Complex m_exterior_distance(int N, double R, double _Complex c)
{
double _Complex z = 0;
double _Complex dc = 0;
for (int n = 0; n < N; ++n)
{
if (cabs(z) > R)
return 2 * z * log(cabs(z)) / dc;
dc = 2 * z * dc + 1;
z = z * z + c;
}
return -1;
}
Code
Parameter Plain ( Mandelbrot) Atom Domains
Atom domains in the Mandelbrot set are defined as the index
C99 Code : atom-domains.c
#include <complex.h>
#include <math.h>
int m_atom_domains(int N, double R, double _Complex c)
{
double _Complex z = c;
double mp = 1.0 / 0.0;
int p = 0;
for (int n = 1; n < N; ++n)
{
double zp = cabs(z);
if (zp < mp)
{
mp = zp;
p = n;
}
z = z * z + c;
}
return p;
}
see also :
- gif and code at commons
- wikibooks
- @CHAModified describes a modification of atom domains, which makes smaller domains more visible
- filtered_atom_domains
One can apply atom domain algorithm only to the exterior and add grid:
C99 Code : ad2.c
Preperiodic points in the Mandelbrot set are called Misiurewicz points, and
they have repelling orbits (periodic hyperbolic components have attracting
orbits). Atom domains show the period with the preperiod fixed at
MisiurewiczDomains
defines the Misiurewicz domain for period
C99 Code = misiurewicz-domains.c
#include <complex.h>
int m_misiurewicz_domains(int N, int p, double _Complex c)
{
int q = 0;
double _Complex z = c;
double _Complex zp = c;
double mq = 1.0 / 0.0;
for (int n = 0; n < p; ++n)
{
z = z * z + c;
}
for (int n = 0; n < N - p; ++n)
{
double zq = cabs(z - zp);
if (zq < mq)
{
mq = zq;
q = n;
}
z = z * z + c;
zp = zp * zp + c;
}
return q;
}
Interior coordinates in the Mandelbrot set
There is an algorithm to find points on the boundary of the Mandelbrot set, given a particular hyperbolic component and the desired internal angle. It involves Newton's method in two complex variables to solve
where
$F^0(z,c) = z$ $F^{q+1}(z,c) = F^q(F(z,c)^2+c)$ -
$p$ is the period of the target component -
$b$ is interior coordinates:$b=e^{2 \pi i \theta}$ with the$\theta$ the desired internal angle. - the resulting
$c$ is the coordinates of the point on the boundary.
It can also be modified to find points in the interior, simply set
It is also possible to invert the algorithm, namely find b given c, and use b as polar texture coordinates to place copies of an image inside the Mandelbrot fractal. The answer is yes, and in some sense it's actually easier than the other direction.
The algorithm I developed works like this:
- when c is outside the Mandelbrot set, give up now
- For each period p, starting from 1 and increasing:
- Find
$z_0$ such that$F^p(z_0,c) = z_0$ using Newton's method in one complex variable (see "attractor"); - Find b by evaluating
$\frac{\partial}{\partial z} F^p(z,c)$ at$z_0$ - If
$\left|b\right| \le 1$ then return b, otherwise continue with the next p
- Find
Algorithm for calculating interior coordinates
- For each period p, starting from 1 and increasing:
- If |F^p(0, c)| reaches a new minimum:
- Find
$z_0$ such that$F^p(z_0,c)=z_0$ using Newton's method in one complex variable (see "attractor"); - Find
$b =dF^p(z_0,c)/dz$ - If
$|b| <= 1$ then return b, otherwise continue with the next p.
- Find
C99 Code = interior-coordinates.c
#include <complex.h>
double _Complex m_interior_coordinates(int N, int M, double _Complex c)
{
double _Complex z = 0;
double mz = 1.0 / 0.0;
for (int n = 0; n < N; ++n)
{
z = z * z + c;
double zp = cabs(z);
if (zp < mz)
{
mz = zp;
double _Complex w = m_attractor(z, n, c, M);
double _Complex dw = 1;
for (int m = 0; m < n; ++m)
{
dw = 2 * w * dw;
w = w * w + c;
}
if (cabs(dw) <= 1)
return dw;
}
}
return 0;
}
a simple test for interior-hood is (this follows directly from the interior distance formula, which gives a non-negative distance if the point is interior):
but one have to know period p and periodic point z_0
The general formula for interior distance estimation d is:
- where the derivatives are evaluated at
$z_0$ satisfying$F^p(z_0, c) = z_0$ - Obtaining z0 by iteration of fc is impractical, requiring a huge number of iterations, moreover that leaves p still to be determined.
Assuming p is known, it's possible to find z0 more directly by using Newton's method to solve the equation:
which can be implemented in C99 like this:
#include <complex.h>
int attractor
( complex double *z_out
, complex double *dz_out
, complex double z_in
, complex double c
, int period
) {
double epsilon = 1e-10;
complex double zz = z_in;
for (int j = 0; j < 64; ++j) {
complex double z = zz;
complex double dz = 1;
for (int i = 0; i < period; ++i) {
dz = 2 * z * dz;
z = z * z + c;
}
complex double zz1 = zz - (z - zz) / (dz - 1);
if (cabs(zz1 - zz) < epsilon) {
*z_out = z;
*dz_out = dz;
return 1;
}
zz = zz1;
}
return 0;
}
Following page Practical interior distance rendering, simple code for distance estimate rendering now looks something as follows; more involved algorithms that provide a significant speed-up can be found at the reference.
Pseudo-code for interior distance estimate rendering now looks something like:
dc := 0
z := 0
m := infinity
p := 0
for (n := 1; n <= maxiters; ++n) {
dc := 2 * z * dc + 1
z := z^2 + c
if (|z| < m) {
m := |z|
p := n
if (attractor(&z0, &dz0, z, c, p)) {
if (|dz0| <= 1) {
// point is interior with period p and known z0
// compute interior distance estimate
break
}
}
}
if (|z| > R) {
// point is exterior
// compute exterior distance estimate from z and dc
break
}
}
C99 Code
#include <complex.h>
#include <math.h>
double cnorm(double _Complex z)
{
return creal(z) * creal(z) + cimag(z) * cimag(z);
}
// Computing the interior distance estimate with known p and z0
double m_interior_distance(double _Complex z0, double _Complex c, int p) {
double _Complex z = z0;
double _Complex dz = 1;
double _Complex dzdz = 0;
double _Complex dc = 0;
double _Complex dcdz = 0;
for (int m = 0; m < p; ++m)
{
dcdz = 2 * (z * dcdz + dz * dc);
dc = 2 * z * dc + 1;
dzdz = 2 * (dz * dz + z * dzdz);
dz = 2 * z * dz;
z = z * z + c;
}
return (1 - cnorm(dz)) / cabs(dcdz + dzdz * dc / (1 - dz));
}
double m_distance(int N, double R, double _Complex c)
{
double _Complex dc = 0;
double _Complex z = 0;
double m = 1.0 / 0.0;
int p = 0;
for (int n = 1; n <= N; ++n)
{
dc = 2 * z * dc + 1;
z = z * z + c;
if (cabs(z) > R)
// point is exterior
// compute exterior distance estimate from z and dc
return 2 * cabs(z) * log(cabs(z)) / cabs(dc);
if (cabs(z) < m)
{
m = cabs(z);
p = n;
double _Complex z0 = m_attractor(z, c, p);
double _Complex w = z0;
double _Complex dw = 1;
for (int k = 0; k < p; ++k)
{
dw = 2 * w * dw;
w = w * w + c;
}
if (cabs(dw) <= 1)
// point is interior with period p and known z0
// compute interior distance estimate
return m_interior_distance(z0, c, p);
}
}
return 0;
}
Code
- interior-distance.c it uses both exterior ( DEM) and interior distance estimation method (IDM)
There is also test for 3 versions of above algorithm
- plain: just computes exterior distance estimates with no interior checking
- unbiased: checks for interior-hood every time a partial is reached.
- biased: uses local connectedness properties to postpone interior checking in exterior regions
One would think the extra interior checks would slow down the rendering, but unbiased is actually faster than plain for the default view because lots of pixels are interior with low periods, reducing the total number of iterations required (with plain all interior points are iterated up to the maximum iteration limit). However, zooming in to a region with no interior pixels visible shows a different result entirely: the interior checks are all useless, and do indeed slow down the rendering dramatically.
The solution is a modified rendering algorithm biased, which uses local connectedness properties to postpone interior checking in exterior regions. We keep track of the outcome of the previous pixel (whether it was interior or exterior) and use that to adjust the calculations - if the previous pixel was interior, perform interior checking as in unbiased, but if the previous pixel was exterior, instead of performing interior checking for each period as it is encountered, store the inputs to the interior checking and carry on. Postponing the expensive interior checks until the maximum iteration count has been reached means that for exterior points that escape before the maximum iteration count has been reached, we don't need to perform the interior checking at all. In the benchmark graphs, biased is blue, and you can see that it improves performance to near that of plain for the exterior-dominated embedded julia set view.
However, there is a flaw - for deep images (close to the resolution of double) where exterior distance estimates have no problems, the interior distance technique can fail in spectacularly ugly fashion:
Extra code from test page: 2014-11-02_practical_interior_distance_rendering
- Makefile
- gnuplot.txt - gnuplot commands for drawing diagram
- mandelbrot.c
- mandelbrot.sh
mu-atom mapping: period p hyperbolic components of the Mandelbrot set can each be mapped conformally to the unit disc, by the derivative d/dz of the periodic limit cycle where f_c^p(z_0) = z_0.
Code
One can run:
- make ( to make 1-10 png files)
- ./a.out 1 > 1.ppm ( to make one file)
Adaptive super-sampling using distance estimate By the Koebe 1/4 theorem, the distance estimate for the Mandelbrot set can be used for a colouring where every dark pixel is near the boundary and every pixel near the boundary is dark. Plain super-sampling (calculating several points for each output pixel) can be used to show finer details, but can be slow. Super-sampling for distance estimate colouring can be accelerated by only subdividing dark pixels, because light pixels are far from the boundary and would still be far from the boundary when subdivided.
Here's the core function of a small C99 renderer that uses this technique:
double cabs2(complex double z) {
return creal(z) * creal(z) + cimag(z) * cimag(z);
}
double colour(complex double c, double pixel_spacing, int maxiters, int depth) {
double er2 = 512 * 512;
complex double z = 0;
complex double dc = 0;
for (int i = 0; i < maxiters; ++i) {
dc = 2 * z * dc + 1;
z = z * z + c;
if (cabs2(z) > er2) {
double de = 2 * cabs(z) * log(cabs(z)) / (cabs(dc) * pixel_spacing);
if (de <= 4 && depth > 0) {
double p2 = pixel_spacing / 2;
double p4 = pixel_spacing / 4;
return 0.25 *
( colour(c + p4 + I * p4, p2, maxiters, depth - 1)
+ colour(c - p4 + I * p4, p2, maxiters, depth - 1)
+ colour(c + p4 - I * p4, p2, maxiters, depth - 1)
+ colour(c - p4 - I * p4, p2, maxiters, depth - 1)
);
} else {
return tanh(de);
}
}
}
return 0;
}
If the distance estimate is small compared to the pixel size, the pixel is subdivided into 4 smaller subpixels, up to a maximum subdivision depth. In the worst case, where every pixel is subdivided all the way down, this needs more computation than plain supersampling due to the intervening levels, but for many locations the adaptive method eventually wins in terms of efficiency. The efficiency is related to the apparent fractal dimension - though in the limit the dimension of the boundary of the Mandelbrot set is known to be 2, most views rendered are not that dense.
Here are some benchmarks, first for the sparse view:
depth samples per pixel time
0 100% 0.440s
1 133.92% 0.664s
2 228.73% 1.304s
3 493.53% 3.180s
4 1233.22% 8.737s
And for the dense view, which only beats plain supersampling at depth 3 or above:
depth samples per pixel time
0 100.00% 17.153s
1 431.31% 1m18.861s
2 1668.51% 5m18.520s
3 6282.81% 20m53.466s
4 23465.98% 81m35.974s
Code
One can run:
- make ( makes all png images)
- ./a.out ( to make one pgm file)
- periodicity scan - scanning parameter plane and finding period of each point c = period of each componnet. Here Robert P. Munafo's Jordan curve method and Newton method is used
- c code: box period
- original c code
The nucleus
Synonims
- center of hyperbolic component of Mandelbrot set
- Nucleus of a Mu-Atom
Applying Newton's method in one complex variable:
A reasonable starting guess
C99 Code
#include <complex.h>
double _Complex m_nucleus(double _Complex c0, int period, int mMax)
{
double _Complex c = c0;
for (int m = 0; m < mMax, ++m)
{
double _Complex z = 0;
double _Complex dc = 0;
for (int i = 0; i < period; ++i)
{
dc = 2 * z * dc + 1;
z = z * z + c;
}
c = c - z / dc;
}
return c;
}
Examples
c_0 = -1.8
c_1 = -1.75(78298397040690
c_2 = -1.7548913878280483
c_3 = -1.754877666(5449095
c_4 = -1.7548776662466929
c_0 = i
c_1 = -0.11926605504587157 + 0.9357798165137615 i
c_2 = -0.23004008246918839 + 1.0769584254327267 i
c_3 = -0.18143326956347502 + 1.0265172581955131 i
c_4 = -0.15552525333120717 + 1.0295018249244792 i
c_5 = -0.15654521565515017 + 1.0322758207945100 i
c_6 = -0.15652017318905109 + 1.0322471096777757 i
c_7 = -0.15652016683375508 + 1.0322471089228316 i
c_8 = -0.15652016683375505 + 1.0322471089228318 i
c_9 = -0.15652016683375503 + 1.0322471089228318 i
c_{10} = -0.15652016683375508 + 1.0322471089228318 i
Code
- nucleus.c - simple one function
- m_d_nucleus.c - complex function
- original file m-nucleus.c which uses m_d_nucleus.c
The attractor
Applying Newton's method in one complex variable:
$ w_{m+1} = w_m - \frac{F^p(w_m, c) - w_m}{\frac{\partial}{\partial z}F^p(w_m, c) - 1}$
A reasonable starting guess for Newton's method is
C99 Code
#include <complex.h>
double _Complex m_attractor(double _Complex w0, double _Complex c, int p, int n)
{
double _Complex w = w0;
for (int m = 0; m < n; ++m)
{
double _Complex z = w;
double _Complex dz = 1;
for (int i = 0; i < p; ++i)
{
dz = 2 * z * dz;
z = z * z + c;
}
w = w - (z - w) / (dz - 1);
}
return w;
}
Examples
w_0 = & 0.5 i
w_1 = -0.12500000000000000 + 0.37500000000000000i
w_2 = -0.13602941176470587 + 0.39338235294117646 i
w_3 = -0.13600977572623132 + 0.39307572864807383 i
w_4 = -0.13600982475703358 + 0.39307568887870914 i
w_5 = -0.13600982475703449 + 0.39307568887871164 i
w_0 = 0.1 - 0.12 i
w_1 = 0.095782435714904968 - 0.084585559740811250 i
w_2 = 0.097499098252211647 - 0.083677122424611575 i
w_3 = 0.097497068763801931 - 0.083682418871189990 i
w_4 = 0.097497068806210202 - 0.083682418894370822 i
The interior point
Applying Newton's method in two complex variables:
$\left(\begin{matrix} \frac{\partial}{\partial z}F^p(w_m,b_m) - 1 & \frac{\partial}{\partial c}F^p(w_m,b_m) \ \frac{\partial}{\partial z}\frac{\partial}{\partial z}F^p(w_m,b_m) & \frac{\partial}{\partial c}\frac{\partial}{\partial z}F^p(w_m,b_m) \end{matrix}\right)\left(\begin{matrix} w_{m+1} - w_m \ b_{m+1} - b_m\end{matrix}\right) = -\left(\begin{matrix} F^p(w_m,b_m) - w_m \ \frac{\partial}{\partial z}F^p(w_m,b_m) - t \end{matrix}\right)$
#include <complex.h>
void m_interior_point
( double _Complex *z_out, double _Complex *c_out
, double _Complex z0, double _Complex c0
, double _Complex t, int p, int n
)
{
double _Complex cc = c0;
double _Complex zz = z0;
for (int m = 0; m < n; ++m)
{
double _Complex c = cc;
double _Complex z = zz;
double _Complex dz = 1;
double _Complex dc = 0;
double _Complex dzdz = 0;
double _Complex dcdz = 0;
for (int i = 0; i < p; ++i)
{
dcdz = 2 * (z * dcdz + dc * dz);
dzdz = 2 * (z * dzdz + dz * dz);
dc = 2 * z * dc + 1;
dz = 2 * z * dz;
z = z * z + c;
}
double _Complex det = (dz - 1) * dcdz - dc * dzdz;
cc = cc - ((dz - 1) * (dz - t) - dzdz * (z - zz)) / det;
zz = zz - (dcdz * (z - zz) - dc * (dz - t)) / det;
}
*z_out = zz;
*c_out = cc;
}
Examples
w_0 = 0 b_0 = 0
w_1 = -0.24999999999999989 + 0.43301270189221935 i b_1 = -0.24999999999999989 + 0.43301270189221935 i
w_2 = -0.24999999999999989 + 0.43301270189221935 i b_2 = -0.12499999999999978 + 0.64951905283832900 i
A preperiodic Misiurewicz point c of preperiod (of critical point) q and period p = M_{q,p} satisfies:
description
code: misiurewicz-point.c
Child of a cardioid of size
Child of a circle of size
@MSetENotes2013 presents a method to estimate the size of a hyperbolic component:
Under iterations the critical orbit of period
$p$ consecutively cycles through$p$ narrow intervals$S_1 \to S_2 \to \ldots \to S_1$ each of width$s_j$ (we choose$S_1$ to include the critical point$z = 0$ ). We expand$F^p(z, c)$ for small$z$ (in the narrow central interval$S_1$ ) and$c$ near its value$c_0$ at superstability of period-$p$ attracting orbit. We see that the$s_j$ are small and the map in the intervals$S_2, S_3, \ldots S_n$ may be regarded as approximately linear; the full quadratic map must be retained for the central interval. One thus obtains$z_{n+p} \approx L_p (z_n^2 + b (c - c_0))$ where$L_p = l_2 l_3 \ldots l_p$ is the product of the map slopes,$l_n = 2 z_n$ in$p-1$ noncentral intervals and$b = 1 + l_2^{-1} + (l_2 l_3)^{-1} + ... + L_n^{-1}$ . We take$L_n$ at$c = c_0$ and treat it as a constant in narrow window. Introducing$Z = L_n z$ and$C = b L_n^2 (c - c_0)$ we get quadratic map$Z_{n+p} = Z_n^2 + C$ . Therefore the window width scales like$(b L_n^2)^{-1}$ .
C99 Code
#include <complex.h>
double _Complex m_size_estimate(double _Complex c, int p)
{
double _Complex b = 1;
double _Complex l = 1;
double _Complex z = 0;
for (int i = 1; i < p; ++i)
{
z = z * z + c;
l = 2 * z * l;
b = b + 1 / l;
}
double _Complex s = 1 / (b * l * l);
return s;
}
Examples
Cardioid
Circle
Cardioid
Cardioid
Dolotin in paper "On the shapes of elementary domains or why Mandelbrot Set is made from almost ideal circles?" eq. 5.8 derive a shape estimate
#include <complex.h>
#include <stdbool.h>
double _Complex m_shape_estimate(double _Complex c, int p)
{
double _Complex z = c;
double _Complex dc = 1;
double _Complex dz = 1;
double _Complex dcdc = 0;
double _Complex dcdz = 0;
for (int i = 1; i < p; ++i)
{
dcdc = 2 * (z * dcdc + dc * dc);
dcdz = 2 * (z * dcdz + dc * dz);
dc = 2 * z * dc + 1;
dz = 2 * z * dz;
z = z * z + c;
}
double _Complex s = - (dcdc / (2 * dc) + dcdz / dz) / (dc * dz);
return s;
}
bool m_shape_is_cardioid(double _Complex s)
{
bool d = cabs(s) < cabs(s - 1);
return d;
}
Examples
Code
atom domain size estimation page derives the atom domain size estimate for a nucleus
where
C99 Code
#include <complex.h>
double domain_size(double _Complex c, int p)
{
double _Complex z = c;
double _Complex dc = 1;
double abszq = cabs(z);
for (int_t q = 2; q <= p; ++q)
{
dc = 2 * z * dc + 1;
z = z * z + c;
double absz = cabs(z);
if (absz < abszq && q < p)
abszq = absz;
}
return abszq / cabs(dc);
}
Examples
Circle
Cardioid
The next point
where
When crossing dwell bands, double
C99 Code
#include <complex.h>
#include <math.h>
#include <gmp.h>
struct m_exray_in {
mpq_t angle;
mpq_t one;
int sharpness;
double er;
double _Complex c;
int j;
int k;
};
struct m_exray_in *m_exray_in_new(const mpq_t angle, int sharpness)
{
m_d_exray_in *ray = malloc(sizeof(*ray));
mpq_init(ray->angle);
mpq_set(ray->angle, angle);
mpq_init(ray->one);
mpq_set_ui(ray->one, 1, 1);
ray->sharpness = sharpness;
ray->er = 65536.0;
double a = twopi * mpq_get_d(ray->angle);
ray->c = ray->er * cexp(I * a);
ray->k = 0;
ray->j = 0;
return ray;
}
void m_exray_in_delete(struct m_exray_in *ray) {
mpq_clear(ray->angle);
mpq_clear(ray->one);
free(ray);
}
double _Complex m_exray_in_step(struct m_exray_in *ray, int n)
{
if (ray->j >= ray->sharpness)
{
mpq_mul_2exp(ray->angle, ray->angle, 1);
if (mpq_cmp_ui(ray->angle, 1, 1) >= 0)
mpq_sub(ray->angle, ray->angle, ray->one);
ray->k = ray->k + 1;
ray->j = 0;
}
double r = pow(ray->er, pow(0.5, (ray->j + 0.5) / ray->sharpness));
double a = twopi * mpq_get_d(ray->angle);
double _Complex target = r * cexp(I * a);
double _Complex c = ray->c;
for (int m = 0; m < n; ++m)
{
double _Complex z = 0;
double _Complex dc = 0;
for (int p = 0; p <= ray->k; ++p)
{
dc = 2 * z * dc + 1;
z = z * z + c;
}
c = c - (z - target) / dc;
}
ray->j = ray->j + 1;
ray->c = c;
return c;
}
Given a periodic external angle pair
Here
Haskell Code
type ExternalAngle = ([Bool], [Bool])
tuning :: (ExternalAngle, ExternalAngle) -> ExternalAngle -> ExternalAngle
tuning (([], per0), ([], per1)) (pre, per)
= (concatMap t pre, concatMap t per)
where
t False = per0
t True = per1
Examples
The external angle pair of the period
The lower angle of the period
The external angle of the tip of the antenna is
The child bulb of the period
where
$b_0 b_1 \ldots = \operatorname{map} \left(\in \left(1 - \frac{p}{q}, 1\right)\right) \circ \operatorname{iterate} \left(+\frac{p}{q}\right) $ \frac{p}{q}$
Haskell Code
import Data.Fixed (mod')
import Data.List (genericTake)
import Data.Ratio (denominator)
type InternalAngle = Rational
type ExternalAngle = ([Bool], [Bool])
primaryBulb :: InternalAngle -> (ExternalAngle, ExternalAngle)
primaryBulb pq = (([], bs ++ [False, True]), ([], bs ++ [True, False]))
where
q = denominator pq
bs
= genericTake (q - 2)
. map (\x -> 1 - pq < x && x < 1)
. iterate (\x -> (x + pq) `mod'` 1)
$ pq
Examples
Consider the bulb at internal angle
Therefore the external angles for the
fork of
- mandelbrot-book: how to write a book about the Mandelbrot set by Claude Heiland-Allen
- The Mandelbrot Set by Claude Heiland-Allen
- html book draft
- pdf 2014-04-25
- Examples c programs for drawing Mandelbrot set as described in the wiki by A Cheritat
- mandelbrot-book: how to write a book about the Mandelbrot set by Claude Heiland-Allen ( full repository )
- clone of mandelbrot-book, use c files not diffs and pdf file
- Linas Art Gallery
- in c99
- one file
- use
- OpenMP library
- complex numbers
- create pgm or ppm graphic files using bash pipeline
Programs create pgm/ppm files. To convert to png use Image Magic from console:
convert *.pgm -geometry 800x800 *.png
convert *.ppm -geometry 800x800 *.png
- Fractal dimension of Julia sets by Claude Heiland-Allen 2016-05-16
- PI in the Mandelbrot Set
- Mandelbrot symbolics in the browser
- Simpler series approximation
- Poincaré Zoom
- Zoom Interpolation
- The Farey Tree: Finding parents in the Farey tree by Claude Heiland-Allen
create a new repository on the command line
git init
git add README.md
git commit -m "first commit"
git branch -M main
git remote add origin [email protected]:adammaj1/Mandelbrot-book-book.git
git push -u origin main
Change:
- in general settings
- add Social Preview Image ( Images should be at least 640×320px (1280×640px for best display))
- in repository details ( near About) add
- description
- website
- Topics (separate with spaces)
Local repository
~/Mandelbrot-book-book
mkdir png
git add *.png
git mv *.png ./png
git commit -m "move"
git push -u origin main
then link the images:
![](./png/n.png "description")
to overwrite
git mv -f
git mv ./src/*.c ./src/modified/bash/
git mv ./src/*.sh ./src/modified/bash/
gitt mv ./src/Makefile ./src/modified/bash/
- GitHub Flavored Markdown Spec
- md cheat sheet
- CommonMark Spec
- Markdown parser
- setting-up-and-managing-your-github-profile
Math equation inside README.md
$\sqrt{3x-1}+(1+x)^2$