Processing math: 0%
\newcommand{\n}{\hat{n}}\newcommand{\thetai}{\theta_\mathrm{i}}\newcommand{\thetao}{\theta_\mathrm{o}}\newcommand{\d}[1]{\mathrm{d}#1}\newcommand{\w}{\hat{\omega}}\newcommand{\wi}{\w_\mathrm{i}}\newcommand{\wo}{\w_\mathrm{o}}\newcommand{\wh}{\w_\mathrm{h}}\newcommand{\Li}{L_\mathrm{i}}\newcommand{\Lo}{L_\mathrm{o}}\newcommand{\Le}{L_\mathrm{e}}\newcommand{\Lr}{L_\mathrm{r}}\newcommand{\Lt}{L_\mathrm{t}}\newcommand{\O}{\mathrm{O}}\newcommand{\degrees}{{^{\large\circ}}}\newcommand{\T}{\mathsf{T}}\newcommand{\mathset}[1]{\mathbb{#1}}\newcommand{\Real}{\mathset{R}}\newcommand{\Integer}{\mathset{Z}}\newcommand{\Boolean}{\mathset{B}}\newcommand{\Complex}{\mathset{C}}\newcommand{\un}[1]{\,\mathrm{#1}}

   

Number Spirals

-

   

Introduction

This is a slightly updated version of this post from back in 2009

The well-known Ulam spiral and the variant developed by Robert Sacks, the Sacks spiral, show interesting geometric patterns in the positions of primes. This page explores a simple extension of these spirals to visualize the number of unique prime factors for each number and provides Python code for drawing them, along with some pre-rendered examples, in PostScript and PNG format.

   

Construction

The layout of the Sacks spiral aligns the squares (1,4,9,16,... etc) along a straight line heading east from the center. Its construction is very simple: the polar co-ordinates of each integer i is just:

\theta = 2 \pi \sqrt{i} \\ r = \sqrt{i},

and thus its Cartesian x,y co-ordinates are given by:

x = -\sqrt{i}\cos(2\pi\sqrt{i})\\ y = \sqrt{i}\sin(2\pi\sqrt{i})

The result is a spiral like this:

Primes highlighted on the polar spiral. Primes are colored blue, squares pink.


   

Prime Factor Spirals

In the original construction, points are coloured if they prime and uncoloured if not. Dense clustering of primes along particular paths appears. This is quite unexpected, and to an extent unexplained. By generalizing to visualize the number of unique prime factors, other geometric features appear. For example, the radius of each drawn circle can be made proportional to the number of unique prime factors the associated number has. The result is an extremely rich and varied pattern.

10,000 numbers on a Sacks Spiral

This is an image of the first 10000 numbers laid out in such a spiral, produced by the Python code given below.

Why unique prime factors? Well, it seems to have a lot of visual structure — more interesting than the same plots showing total prime factors, or other variations. It's easy to plot variations if you feel like exploring them; after all the code you need is all here.

   

Interesting Features — A Short Tour

There are a number of geometric features that appear on the plots.

To see most of these features, we need to look at a high-resolution image of the spiral with at least 100,000 points. The big PNG version (6600×6700px) has sufficient detail.

100,000 points on the Sacks spiral

All of the angle references apply to the compass drawn around the 100,000 point version.

The sparse curves. There are several curves which are very sparse (i.e. there is a high density of prime numbers and few-factored composites). The most prominent of these meets the exterior at about 203°. A second, smaller one meets the exterior at 189.5°. Another meets at 37° and has a fainter parallel at 29°.

The vertical lines. Between about 90 and 70°, and between 270 and 250, there are distinct, unevenly spaced vertical lines, getting more tightly spaced as the axes (90 and 270) are reached.

The diagonals. At exactly 60 and 300° two fuzzy lines extending from the center are clearly visible. A number of fainter parallel lines can be seen, anti-clockwise from the original lines. A symmetric pair at 120 and 240 are very faintly visible.

Dense horizontals. In the quadrant from 90 to 180°, numerous dense lines can be seen, becoming more tightly spaced towards 180°. The line at exactly 180° is the densest line on entire spiral.


   

Code

The code for generating rendering the spirals is very simple. For maximum quality, a vector format is desirable; I've used PyX package to render to PDF.

   

Pre-requisites

To run these examples, you need Python, PyX, and the elementary number theory package ent.py by William Stein. Both of these are pure Python and should run on any platform.

   

Basic version

The simplest code looks like this (this produces the image of the first 10000 points shown earlier):

from pyx import canvas, document, path
from ent import factor
from math import sin, cos, sqrt, pi
        
n = 10000    
ca = canvas.canvas()

for j in range(n):   
    i = j + 1
    r = sqrt(i)
    theta = r * 2 * pi  
    x = cos(theta)*r
    y = -sin(theta)*r        
    factors = factor(i)               
    if(len(factors)>1):            
        radius = 0.05*pow(2,len(factors)-1)
        ca.fill(path.circle(x,y, radius))
d = document.document(pages = [document.page(ca, 
                            paperformat=document.paperformat.A4, 
                            fittosize=1)])
d.writePDFfile('spiral_1e4.pdf')

spiral_1e4.py

The PDF file generated is spiral_1e4.pdf. This could easily be changed to color the points differently instead of modulating the radius, e.g. by replacing the ca.fill() call with

ca.fill(path.circle(x,y, 0.3), 
        color.palette.RedGreen.getcolor((len(factors)-1)/8.0)))

There are lots of visualisation techniques that could be used — for example false colouring the image using one color channel for the number of unique factors, one for the primes, and another for the total number of factors. If you find any interesting ones, please let me know. In this code, the radius of each point is 2^{f-1} (where f is number of unique prime factors).

Prime numbers are omitted entirely. The exponential scaling is largely arbitrary; I tried a number of different functions and found this to be the most revealing. Since numbers with large numbers of unique prime factors are rare in small integers (no number below 9699690 can have more than seven), the mapping works well.

   

Scaling Up

The output for 100,000 points is spiral_1e5.pdf.

100,000 number prime spiral

spiral_with_labels.py adds a compass (divided into tenths of degrees) around the utside of the area, and adds a textual label. NOTE: If you're using this code, you'll either need to have LaTeX installed to do the text rendering, or comment out the “texrunner” lines from the source in write_label() and draw_axis(), and live without the labels.

For 1,000,000 points the extremely dense result is shown below (you'll have to run it yourself if you want the 178M PDF file!)

One million numbers on the Sacks spiral

I've rendered up to 10,000,000 points successfully with this code. However the files are huge and slow, so if you want to see you'll need to render it yourself!

   

Vogel Spiral

Using Vogel's floret model for layout also gives nice results. This model gives each integer i polar co-ordinates:

r = \sqrt{i}\\ \theta = \frac{2\pi i}{\phi^2}

where \phi (the golden ratio) is given by:

\phi = \frac{1 + \sqrt{5}}{2}

The result of this arrangement is to align the Fibonacci numbers along the eastern axis (although the first few are slightly off axis).

The Vogel spiral, with the Fibonacci numbers highlighted in blue.

In contrast to the original spiral, which had a square on every turn, the spacing between Fibonacci numbers increases rapidly. The above image was generated by vogel_labeled.py.

10,000 numbers, marked according to unique prime factors, on the Vogel spiral.

spiral_vogel_1e5.pdf is the 100,000 point PDF. This unadorned spiral was generated with spiral_1e5_vogel.py .

The Vogel spiral has quite a different pattern when plotting the total number of factors rather than the number of unique ones (100,000 point plot). In fact, if the Vogel spiral plots of the total and unique factors are overlaid, they show very little visual relation to each other.

100,000 numbers with point size indicating total number of factors on the Vogel spiral.

The python source that generates this image is spiral_vogel_all.py , and the PDF is pdf/spiral_vogel_all.pdf.

   

Fibonacci sums

A quick aside: every integer can be represented a sum of one or more distinct Fibonacci numbers. Some numbers cna be represented only one way (e.g. the Fibonacci numbers themselves), while others can be represented in multiple ways (e.g. 8=8, 8=5+3 and 8=5+2+1). The number of ways a number can be represented is notated H(n), and is Sloane sequence A0000119 . Plotting this function on the Vogel spiral is easy: spiral_nfib.py

The result of plotting the Fibonacci decompositions on the Vogel spiral is a very clear block patterning.

   

Direct Rasterization

Above 1M points, producing lossless vector files is too inefficient to be very useful. Instead, the points can be directly rasterized onto a grid as they are computed, and a grayscale bitmap output file created. Using a simple antialiased pixel rendering technique (Wu pixels) avoids aliasing as the image is built up. I've done renders up to 100,000,000 point using direct rasterisation on a 4096 x 4096 grid. The image below shows a 100,000,000 point render.

100,000,000 points directly rasterised

John H Williamson

GitHub / @jhnhw

formatted by Markdeep 1.18