Loading...
Logo
Processing Forum

cellular automata

in Programming Questions  •  1 year ago  
Hi, I'm having some trouble understanding the code in this cellular automata example:  http://processing.org/learning/topics/wolfram.html

I don't quite understand what's going on in the class CA—here's the portion of code I need help understanding:

Copy code
  1. class CA {

  2.   int[] cells;     // An array of 0s and 1s 
  3.   int generation;  // How many generations?
  4.   int scl;         // How many pixels wide/high is each cell?

  5.   int[] rules;     // An array to store the ruleset, for example {0,1,1,0,1,1,0,1}

  6.   CA(int[] r) {
  7.     rules = r;
  8.     scl = 1;
  9.     cells = new int[width/scl];
  10.     restart();
  11.   }
  12.   
  13.    CA() {
  14.     scl = 1;
  15.     cells = new int[width/scl];
  16.     randomize();
  17.     restart();
  18.   }
  19.   
  20.   // Set the rules of the CA
  21.   void setRules(int[] r) {
  22.     rules = r;
  23.   }
  24.   
  25.   // Make a random ruleset
  26.   void randomize() {
  27.     for (int i = 0; i < 8; i++) {
  28.       rules[i] = int(random(2));
  29.     }
  30.   }
  31.   
  32.   // Reset to generation 0
  33.   void restart() {
  34.     for (int i = 0; i < cells.length; i++) {
  35.       cells[i] = 0;
  36.     }
  37.     cells[cells.length/2] = 1;    // We arbitrarily start with just the middle cell having a state of "1"
  38.     generation = 0;
  39.   }

  40.   // The process of creating the new generation
  41.   void generate() {
  42.     // First we create an empty array for the new values
  43.     int[] nextgen = new int[cells.length];
  44.     // For every spot, determine new state by examing current state, and neighbor states
  45.     // Ignore edges that only have one neighor
  46.     for (int i = 1; i < cells.length-1; i++) {
  47.       int left = cells[i-1];   // Left neighbor state
  48.       int me = cells[i];       // Current state
  49.       int right = cells[i+1];  // Right neighbor state
  50.       nextgen[i] = rules(left,me,right); // Compute next generation state based on ruleset
  51.     }
  52.     // Copy the array into current value
  53.     cells = (int[]) nextgen.clone();
  54.     generation++;
  55.   }
  56.   
  57.   // This is the easy part, just draw the cells, fill 255 for '1', fill 0 for '0'
  58.   void render() {
  59.     for (int i = 0; i < cells.length; i++) {
  60.       if (cells[i] == 1) fill(255);
  61.       else               fill(0);
  62.       noStroke();
  63.       rect(i*scl,generation*scl, scl,scl);
  64.     }
  65.   }
  66.   
  67.   // Implementing the Wolfram rules
  68.   // Could be improved and made more concise, but here we can explicitly see what is going on for each case
  69.   int rules (int a, int b, int c) {
  70.     if (a == 1 && b == 1 && c == 1) return rules[0];
  71.     if (a == 1 && b == 1 && c == 0) return rules[1];
  72.     if (a == 1 && b == 0 && c == 1) return rules[2];
  73.     if (a == 1 && b == 0 && c == 0) return rules[3];
  74.     if (a == 0 && b == 1 && c == 1) return rules[4];
  75.     if (a == 0 && b == 1 && c == 0) return rules[5];
  76.     if (a == 0 && b == 0 && c == 1) return rules[6];
  77.     if (a == 0 && b == 0 && c == 0) return rules[7];
  78.     return 0;
  79.   }
  80.   
  81.   // The CA is done if it reaches the bottom of the screen
  82.   boolean finished() {
  83.     if (generation > height/scl) {
  84.        return true;
  85.     } else {
  86.        return false;
  87.     }
  88.   }
  89. }
Thanks!

Replies(4)

What part are you specifically confused about?
Well, for one thing, I don't understand this part:

Copy code
  1.   CA(int[] r) {
  2.     rules = r;
  3.     scl = 1;
  4.     cells = new int[width/scl];
  5.     restart();
  6.   }
  7.   
  8.    CA() {
  9.     scl = 1;
  10.     cells = new int[width/scl];
  11.     randomize();
  12.     restart();
  13. }
  
There are two keys to understanding this code.

1)  You need to undertand what the cellular autometa is doing.

2) You need to be famliar with arrays and array syntax.  If you're not, go read up about Arrays in processing, and come back when you understand the basics.

3) Okay three keys.  You need to understand the basics of Object Oriented programming in Processing or Java.

Also, the second constructor has a bug - it doesn't allocate an empty rule set.  You can fix it by inserting the following just before line 12 (before the line that says "randomize();")

    rules = new int[8];


Here is some code to exercise this class, in case you don't have it.

CA ca;

void setup()
{
  size(500, 400);
  ca = new CA();  
}

void draw()
{
  ca.generate();
  ca.render();
}

Perhaps a little easier to see what's going on if you run it.  But basically, the two routines you posted are constructors.

The first receives a set of rules, which are an array of 8 numbers (each of which is 1 or 0).  You can see the effect of each rule in lines 74-82.  Basically for each pattern of 3 bits, it selects one of the rules and uses that to generate the next row.  So it scans through the parent cells, and for each group of 3 cells (a,b,c) it generates a new cell z, which is based on the 8 possible combinations of a,b,c.

a b c
   z

Oh, and the author is right, that code "c ould be improved and made more concise".  Like it could be reduced to a single line, but it is perhaps easier to understand as is.

The first constructor is passed a set of 8 rules. It initializes a set of cells (one cell for each pixel), and initializes some other variables.  The second constructor is essentially the same, but it creates a random set of rules.  That's the version I used in my sample code.


The rules mentioned are not specific to Processing, they were invented by Stephen Wolfram and are Wolfram code. I'd recommend reading about this before making sense of the code: