Loading...
Logo
Processing Forum
Hi all,

The code below is an example of some code I'm using for a game. The problem is the 1px(?) line appearing intermittently after the last Thing in the array. I suspect that this is because  draw() is rendering what's happening irrelevant of whether or not the ThingMover (on a separate Thread) has finished updating or not, which was really the point of using the thread in the first place. 

Does anyone have any ideas on how I might get rid of the flickering gap? I've had a go at using a boolean to make sure only either draw or the ThingMover's run() method is being accessed at any one time with little success, so any help would be much appreciated.

Thanks very much

Dave

Processing user on Google Plus? http://gplus.to/calx

int numThings = 15;
int thingSize = 64;
String here = "<< here";

ThingMover thingMover;

void setup() 
{  
  size(800, 480); 

  thingMover = new ThingMover();
  thingMover.start();
  noStroke();
  noSmooth();
  frameRate(6000);
}

void draw() 
{
  background(0, 255, 0);
  thingMover.displayThings();
}

class Thing
{

  float x, y, w, h, speed;
  int id;

  Thing(float x, float y, float w, float h, float speed, int id)
  {
    this.x = x;    
    this.y = y;    
    this.w = w;
    this.h = h;
    this.speed = speed;
    this.id = id;
  }

  void move()
  {
    x+=speed;
    if (x <= -thingSize)
    {
      x = (numThings-1)*thingSize;
    }
  }

  void display()

  {  
    fill(255);
    rect(x, y, w, h);
    if (id == 0) 
    {
      fill(0);
      text(here, x, y+100);
    }
  }
}

class ThingMover extends Thread
{

  Thing[] things = new Thing[numThings];
  boolean running;

  ThingMover()
  {
    for (int i = 0; i < things.length; i++) 
    {
      things[i] = new Thing(i*thingSize, (height/2)+(i*-2), 64, 400, -1, i);
    }

    running = true;
  }

  void run() 
  { 
    while (running) 
    {
      try 
      {
        sleep(5);
      } 
      catch (Exception e) 
      {
        //one day I really will fill this out.
      }
      updateThings();
    }
  }

  void updateThings()
  {
    for (int i = 0; i < things.length; i++) 
    {
      things[i].move();
    }
  }

  void displayThings()
  {
    for (int i = 0; i < things.length; i++) 
    {
      things[i].display();
    }
  }
}

Replies(2)

You are right in that you need a boolean (in threading this is called a semaphore) to control access  to the data. The line is caused by attempting to draw and update the things at the same time.

I have added a semaphore to your code and tested it. No flickering line now.

Copy code
  1. int numThings = 15;
  2. int thingSize = 64;
  3. String here = "<< here";
  4. ThingMover thingMover;
  5. boolean locked = false;

  6. void setup() { 
  7.   size(800, 480);
  8.   thingMover = new ThingMover();
  9.   thingMover.start();
  10.   noStroke();
  11.   noSmooth();
  12.   frameRate(6000);
  13. }

  14. void draw() {
  15.   background(0, 255, 0);
  16.   while (locked);
  17.   locked = true;
  18.   thingMover.displayThings();
  19.   locked = false;
  20. }

  21. class Thing {
  22.   float x, y, w, h, speed;
  23.   int id;
  24.   Thing(float x, float y, float w, float h, float speed, int id) {
  25.     this.x = x;   
  26.     this.y = y;   
  27.     this.w = w;
  28.     this.h = h;
  29.     this.speed = speed;
  30.     this.id = id;
  31.   }

  32.   void move() {
  33.     x+=speed;
  34.     if (x <= -thingSize) {
  35.       x = (numThings-1)*thingSize;
  36.     }
  37.   }

  38.   void display() { 
  39.     fill(255);
  40.     rect(x, y, w, h);
  41.     if (id == 0) {
  42.       fill(0);
  43.       text(here, x, y+100);
  44.     }
  45.   }
  46. }

  47. class ThingMover extends Thread {
  48.   Thing[] things = new Thing[numThings];
  49.   boolean running;
  50.   ThingMover() {
  51.     for (int i = 0; i < things.length; i++) {
  52.       things[i] = new Thing(i*thingSize, (height/2)+(i*-2), 64, 400, -1, i);
  53.     }
  54.     running = true;
  55.   }

  56.   void run() {
  57.     while (running) {
  58.       try {
  59.         sleep(5);
  60.       }
  61.       catch (Exception e) {
  62.         //one day I really will fill this out.
  63.       }
  64.       if (!locked) {
  65.         locked = true;
  66.         updateThings();
  67.       }
  68.       locked = false;
  69.     }
  70.   }

  71.   void updateThings() {
  72.     for (int i = 0; i < things.length; i++) {
  73.       things[i].move();
  74.     }
  75.   }

  76.   void displayThings() {
  77.     for (int i = 0; i < things.length; i++) {
  78.       things[i].display();
  79.     }
  80.   }
  81. }

Thankyou! The point of having updateThings() and displayThings() on separate threads was to get the smoothest possible movement, but here they are dependent on each other again, and the movement is back to stuttering as the move function waits for draw to complete.  I fixed this by switching the boolean the other way around, where displayThings() only runs if  updateThings()  has finished.

Thanks

Dave

Processing user on Google Plus? http://gplus.to/calx

int numThings = 15;
int thingSize = 64;
String here = "<< here";
ThingMover thingMover;
boolean locked = false;

void setup() { 
  size(800, 480);
  thingMover = new ThingMover();
  thingMover.start();
  noStroke();
  noSmooth();
  frameRate(6000);
}

void draw() {
  background(0, 255, 0);
  if (!locked) {
    locked = true;
    thingMover.displayThings();
  }
  locked = false;
}

class Thing {
  float x, y, w, h, speed;
  int id;
  Thing(float x, float y, float w, float h, float speed, int id) {
    this.x = x;   
    this.y = y;   
    this.w = w;
    this.h = h;
    this.speed = speed;
    this.id = id;
  }

  void move() {
    x+=speed;
    if (x <= -thingSize) {
      x = (numThings-1)*thingSize;
    }
  }

  void display() { 
    fill(255);
    rect(x, y, w, h);
    if (id == 0) {
      fill(0);
      text(here, x, y+100);
    }
  }
}

class ThingMover extends Thread {
  Thing[] things = new Thing[numThings];
  boolean running;
  ThingMover() {
    for (int i = 0; i < things.length; i++) {
      things[i] = new Thing(i*thingSize, (height/2)+(i*-2), 64, 400, -1, i);
    }
    running = true;
  }

  void run() {
    while (running) {
      try {
        sleep(5);
      }
      catch (Exception e) {
        //one day I really will fill this out.
      }
      while (locked);
      locked = true;
      updateThings();
      locked = false;
    }
  }

  void updateThings() {
    for (int i = 0; i < things.length; i++) {
      things[i].move();
    }
  }

  void displayThings() {
    for (int i = 0; i < things.length; i++) {
      things[i].display();
    }
  }
}