Loading...
Logo
Processing Forum
Hello,

I try to use Threads in order to firtst move several balls.
You will find here my code and the result.
I don't understand why i obtain those ghost !
Maybe someone can help me...

Copy code
  1. Balle b,b1,b2;
    int fond=155;
    color rouge=color(255,0,0);
    color vert=color(0,255,0);

    void setup() {
      size(200,300);
      background(fond);
      ellipseMode(RADIUS);
     
      b=new Balle(0,0,10,10,0);
      b.start();
      b1=new Balle(0,0,5,2,rouge);
      b1.start();
      b2=new Balle(0,0,8,5,vert);
      b2.start();
    }

    void draw(){

    }



    class Balle extends Thread{ 
      int X;
      int dx;
      int Y;
      int dy;
      int r;
      int v;
      color couleur;

     
    Balle (int x,int y,int ra,int vi,color c){
      X=x;
      Y=y;
      dx=1;dy=1;
      r=ra;
      v=vi;
      couleur=c;


    void start(){
      super.start();
    }

     
    void run(){
      while (X<2000){
        stroke(fond);fill(fond);
        ellipse(X,Y,r,r);
        Deplace(X,Y);
        stroke(couleur);fill(couleur);
        ellipse(X,Y,r,r);
        stroke(fond);fill(fond);
        try {
            sleep((long)(100/v));
          } catch (Exception e) {
          }
      }

     
     
    void Deplace(int x,int y){
     
      X=x+dx;
      Y=y+dy;
      if (X+r>width)dx=-1;
      if (X<r)dx=1;
      if (Y+r>height)dy=-1;
      if (Y<r)dy=1;
    }
     
    }


Replies(6)

These are probably artifacts because of antialiasing on the borders. It is apparently slightly better by using noStroke() and removing the stroke() call. Also better by drawing the erasing circle with r+1. Can't get rid entirely of the artifacts with my quick tests. That's probably why the most common idiom in Processing is just to erase the whole background in draw() and just redraw everything. Which won't work with threads, of course.
Hello phi.lho,
nice to meet you again !

yes i already tried to erase circle with r+1, i tried too to use background but with "blinding"... and i told me "no it's not normal to do it it must be a good reason, a mistake in my code".
So according to you the code is good it'd be a processin broblem!


Actually, I was mistaken... Like you, I had a naive view on the way threads work (my experience with them is rather limited, I must admit).

I had an epiphany once I saw that the black circle had a partial green ghost (once I got rid of stroke): we forgot that fill() and other similar commands are global to the PApplet, so if you change it, you change it for all running threads!
I changed the code to use synchronized (on g, the global PGraphics of the PApplet, shared by the threads), so when doing an atomic change, the other threads are blocked, doesn't try to work on the same graphics.
Copy code
  1. Balle b,b1,b2;
  2. int fond=155;
  3. color rouge=color(255,0,0);
  4. color vert=color(0,255,0);
  5. void setup() {
  6.   size(200,300);
  7.   background(fond);
  8.   ellipseMode(RADIUS);
  9.   noStroke();
  10.   noSmooth();
  11.  
  12.   b=new Balle(0,0,10,10,0);
  13.   b.start();
  14.   b1=new Balle(0,0,5,2,rouge);
  15.   b1.start();
  16.   b2=new Balle(0,0,8,5,vert);
  17.   b2.start();
  18. }

  19. void draw(){
  20. }

  21. class Balle extends Thread{// Ball
  22.   int X;
  23.   int dx;
  24.   int Y;
  25.   int dy;
  26.   int r;
  27.   int v;
  28.   color couleur;
  29.  
  30. Balle (int x,int y,int ra,int vi,color c){
  31.   X=x;
  32.   Y=y;
  33.   dx=1;dy=1;
  34.   r=ra;
  35.   v=vi;
  36.   couleur=c;
  37. }

  38. void start(){
  39.   super.start();
  40. }
  41.  
  42. void run(){
  43.   while (X<2000){
  44.     Efface(X, Y);
  45.     Deplace();
  46.     Dessine(X, Y);
  47.     try {
  48.         sleep((long)(100/v));
  49.       } catch (Exception e) {
  50.       }
  51.   }
  52. }

  53. void Efface(int x, int y) { // Erase
  54.   synchronized (g) {
  55.     fill(fond);
  56.     ellipse(x,y,r,r);
  57.   }
  58. }

  59. void Dessine(int x, int y) { // Draw
  60.   synchronized (g) {
  61.     fill(couleur);
  62.     ellipse(x,y,r,r);
  63.   }
  64. }
  65.  
  66. void Deplace(){ // Move
  67.   X += dx;
  68.   Y += dy;
  69.   if (X+r>width)dx=-1;
  70.   if (X<r)dx=1;
  71.   if (Y+r>height)dy=-1;
  72.   if (Y<r)dy=1;
  73. }
  74.  
  75. }
I split the operation in two (well, three), so the other threads get a change to do something (more or less complex, so more or less time consuming) specific to themselves (not changing the global state). But here, the synch could be done on a higher level:
Copy code
  1. void run(){
  2.   while (X<2000){
  3.     synchronized (g) {
  4.       Efface(X, Y);
  5.       Deplace();
  6.       Dessine(X, Y);
  7.     }
  8.     try {
  9.         sleep((long)(100/v));
  10.       } catch (Exception e) {
  11.       }
  12.   }
  13. }
  14. void Efface(int x, int y) {
  15.     fill(fond);
  16.     ellipse(x,y,r,r);
  17. }
  18. void Dessine(int x, int y) {
  19.     fill(couleur);
  20.     ellipse(x,y,r,r);
  21. }
 
This scales:
Copy code
  1. Balle[] balles = new Balle[100];
  2. int fond=155;

  3. void setup() {
  4.   size(200,300);
  5.   background(fond);
  6.   ellipseMode(RADIUS);
  7.   noStroke();
  8.   noSmooth();

  9.   for (Balle ball : balles) {
  10.     ball = new Balle(0, 0, int(random(5, 20)), int(random(2, 20)),
  11.         color(random(0, 255), random(0, 255), random(0, 255)));
  12.     ball.start();
  13.   }
  14. }

Thanks phi.lho, I learnt a lot from this example!
Thanks phi lho,

I never heard about synchronized so i seach explanations and understand that it is very important for Threads !!

see here

your help was great!!!