Loading...
Logo
Processing Forum

Update method tip

in Share your Work  •  1 year ago  
Hi,

I see a lot of people doing calculations in the draw method, which in some cases is necessary (when working on opengl textures etc.), but in most it can be avoided and usually only slows down the drawing thread.

I thought I'd share how I make a separate thread to handle the calculations.

I have a class called CalculationThread which looks like this:
Copy code
  1. import java.lang.reflect.InvocationTargetException;
  2. import java.lang.reflect.Method;
  3. import java.util.ArrayList;

  4. public class CalculationThread implements Runnable {

  5.   private Thread t;
  6.   private int sleepTime;
  7.   private ArrayList<UpdateListener> updateListeners = new ArrayList<UpdateListener>();

  8.   public CalculationThread(int sleepTime) {
  9.     this.sleepTime = sleepTime;
  10.   }

  11.   public void addUpdateListener(Object object) {
  12.     try {
  13.       updateListeners.add(new UpdateListener(object, object.getClass().getMethod("update", new Class[] {
  14.       }
  15.       )));
  16.     } 
  17.     catch (SecurityException e) {
  18.       e.printStackTrace();
  19.     } 
  20.     catch (NoSuchMethodException e) {
  21.       System.err.println("No public update method in " + object.getClass().getSimpleName());
  22.     } 
  23.     catch (IllegalArgumentException e) {
  24.       System.err.println("Update method in " + object.getClass().getSimpleName() + " shouldn't have any arguments");
  25.     }
  26.   }

  27.     public void run() {
  28.     Thread me = Thread.currentThread();

  29.     while (t == me) {
  30.       for (int i = 0; i < updateListeners.size(); i++) {
  31.         UpdateListener ul = updateListeners.get(i);
  32.         try {
  33.           ul.getMethod().invoke(ul.getObject(), new Object[] {
  34.           }
  35.           );
  36.         } 
  37.         catch (IllegalArgumentException e) {
  38.           e.printStackTrace();
  39.         } 
  40.         catch (IllegalAccessException e) {
  41.           e.printStackTrace();
  42.         } 
  43.         catch (InvocationTargetException e) {
  44.           e.printStackTrace();
  45.         }
  46.       }

  47.       try {
  48.         Thread.sleep(sleepTime);
  49.       } 
  50.       catch (InterruptedException e) {
  51.         e.printStackTrace();
  52.       }
  53.     }
  54.     t = null;
  55.   }

  56.   public void start() {
  57.     if (t == null) {
  58.       t = new Thread(this);
  59.       t.start();
  60.     }
  61.   }

  62.   public void stop() {
  63.     if (t != null) {
  64.       t.interrupt();
  65.     }
  66.     t = null;
  67.   }

  68.   public void dispose() {
  69.     stop();
  70.   }

  71.   public class UpdateListener {

  72.     private Object object;
  73.     private Method method;

  74.     public UpdateListener(Object object, Method method) {
  75.       this.object = object;
  76.       this.method = method;
  77.     }

  78.     public Object getObject() {
  79.       return object;
  80.     }

  81.     public void setObject(Object object) {
  82.       this.object = object;
  83.     }

  84.     public Method getMethod() {
  85.       return method;
  86.     }

  87.     public void setMethod(Method method) {
  88.       this.method = method;
  89.     }
  90.   }
  91. }
What it does it looping all update methods added to the ArrayList  updateListeners.

This is how I use it.
Copy code
  1. public static CalculationThread CalcThread;
  2. private PVector position;

  3. public void setup() {
  4.   size(500, 500);
  5.   background(255);
  6.   
  7.   CalcThread = new CalculationThread(16);
  8.   CalcThread.start();
  9.   CalcThread.addUpdateListener(this);

  10.   position = new PVector(random(0, 450), random(0, 450));
  11. }

  12. public void update() {
  13.   position.x = random(0, 450);
  14.   position.y = random(0, 450);
  15. }

  16. public void draw() {
  17.   background(255);

  18.   fill(0);
  19.   rect(position.x, position.y, 50, 50);
  20. }
Hope someone finds this useful!

Replies(2)

Interesting. That's how Processing does as they don't want to force to use classes, etc. It is simpler for the users.
Otherwise, it is better to use the method I explain in the https://forum.processing.org/topic/noob-q-i-d-like-to-learn-more-about-callbacks thread, defining an interface for this update() method.
Both methods have their merits...
Yes that's true, that is how I do it when working in the Eclipse environment. But not having a class extending PApplet and being able to implement the interface with the update method in the Processing IDE makes it impossible, no?       

This is how it looks for me in the Eclipse environment
Copy code
  1. import java.lang.reflect.InvocationTargetException;
  2. import java.lang.reflect.Method;
  3. import java.util.ArrayList;

  4. public class CalculationThread implements Runnable {

  5.   private Thread t;
  6.   private int sleepTime;
  7.   private ArrayList<DrawObject> drawObjects = new ArrayList<DrawObject>();

  8.   public CalculationThread(int sleepTime, ArrayList<DrawObject> drawObjects) {
  9.     this.sleepTime = sleepTime;
  10.     this.drawObjects = drawObjects;
  11.   }

  12.   public void run() {
  13.     Thread me = Thread.currentThread();

  14.     while (t == me) {
  15.       for (int i = 0; i < drawObjects.size(); i++) {
  16.         DrawObject d = drawObjects.get(i);
  17.         try {
  18.           d.getClass().getMethod("update", new Class[] {
  19.           }
  20.           ).invoke(d, new Object[] {
  21.           }
  22.           );
  23.         } 
  24.         catch (IllegalArgumentException e) {
  25.           e.printStackTrace();
  26.         } 
  27.         catch (IllegalAccessException e) {
  28.           e.printStackTrace();
  29.         } 
  30.         catch (InvocationTargetException e) {
  31.           e.printStackTrace();
  32.         }
  33.         catch (NoSuchMethodException e) {
  34.           e.printStackTrace();
  35.         }
  36.       }

  37.       try {
  38.         Thread.sleep(sleepTime);
  39.       } 
  40.       catch (InterruptedException e) {
  41.         e.printStackTrace();
  42.       }
  43.     }
  44.     t = null;
  45.   }

  46.   public void start() {
  47.     if (t == null) {
  48.       t = new Thread(this);
  49.       t.start();
  50.     }
  51.   }

  52.   public void stop() {
  53.     if (t != null) {
  54.       t.interrupt();
  55.     }
  56.     t = null;
  57.   }

  58.   public void dispose() {
  59.     stop();
  60.   }
  61. }
And
Copy code
  1. public static CalculationThread CalcThread;
  2. private ArrayList<DrawObject> drawObjects;

  3. private C c;
  4. private R r;

  5. public void setup() {
  6.   size(500, 500);
  7.   background(255);
  8.   
  9.   drawObjects = new ArrayList<DrawObject>();
  10.   drawObjects.add(c = new C(20));
  11.   drawObjects.add(r = new R(30));

  12.   CalcThread = new CalculationThread(16, drawObjects);
  13.   CalcThread.start();
  14. }

  15. public void draw() {
  16.   background(255);

  17.   for (int i = 0; i < drawObjects.size(); i++) {
  18.     drawObjects.get(i).draw();
  19.   }
  20. }

  21. public interface DrawObject {
  22.   void update();
  23.   void draw();
  24. }

  25. public class C implements DrawObject
  26. {
  27.   int diam;

  28.   public C(int r)
  29.   {
  30.     diam = r;
  31.   }

  32.   void update()
  33.   {
  34.     diam += 2;
  35.   }

  36.   void draw()
  37.   {
  38.     fill(#55AA88);
  39.     ellipse(100, 100, diam, diam);
  40.   }
  41. }

  42. public class R implements DrawObject
  43. {
  44.   int size;

  45.   public R(int s)
  46.   {
  47.     size = s;
  48.   }

  49.   void update()
  50.   {
  51.     size += 1;
  52.   }

  53.   void draw()
  54.   {
  55.     fill(#3388AA);
  56.     rect(300, 300, size, size);
  57.   }
  58. }