libRefTools/src/com/tofvesson/async/WorkerThread.java
Gabriel Tofvesson 803fb55643 Major update
Fixed killing threads. Didn't work before if user started async with infinite loop
2016-11-06 17:55:29 +01:00

99 lines
3.7 KiB
Java

package com.tofvesson.async;
import javafx.util.Pair;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.*;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.ThreadLocalRandom;
/**
* A thread tasked with accepting multiple instructions. This is useful for people who don't want to constantly create new threads for heavy work.
* Also good if the fields in an object instantiated in this thread aren't volatile.
*/
public class WorkerThread extends Thread {
List<Long> ids = new ArrayList<>();
volatile Queue<Pair<Long, Pair<Method, Pair<Object, Object>>>> queue;
volatile Map<Long, Object> output = new HashMap<>();
volatile boolean alive=true, completed=false;
/**
* Create a WorkerThread.
* @param queueSize Maximum amount of instructions to be queued.
*/
public WorkerThread(int queueSize){
super();
try{
Field f = Thread.class.getDeclaredField("target");
f.setAccessible(true);
f.set(this, (Runnable) ()->
{
synchronized (Thread.currentThread()) {
while (alive) {
if (queue.size() != 0) {
Pair<Long, Pair<Method, Pair<Object, Object>>> q = queue.poll();
Pair<Method, Pair<Object, Object>> instr = q.getValue();
try {
output.put(q.getKey(), instr.getKey().invoke(instr.getValue().getKey(), (Object[]) instr.getValue().getValue()));
completed = true;
} catch (IllegalAccessException | InvocationTargetException e) {
e.printStackTrace();
}
}
}
}
});
}catch(Exception e){}
queue = new ArrayBlockingQueue<>(queueSize);
}
/**
* Add a new instruction for the worker thread.
* @param invokeOn Object to invoke method on.
* @param m Method to invoke.
* @param params Parameters for method.
* @return A UID corresponding to the queued instruction.
*/
public long enqueue(Object invokeOn, Method m, Object... params){
m.setAccessible(true);
long id;
do{ id = ThreadLocalRandom.current().nextLong(); }while(ids.contains(id));
queue.add(new Pair<>(id, new Pair<>(m, new Pair<>(invokeOn, params))));
ids.add(id);
return id;
}
/**
* Waits for instruction to be processed and return value to be acquired.
* @param id UID of the supplied instruction.
* @return Return value from instruction called in worker thread.
*/
public Object pop(long id){
if(!ids.contains(id)) return null;
if(Thread.currentThread() == this) throw new RuntimeException("Attempting to await result in worker thread! This causes the thread to lock.");
while(!output.containsKey(id)) ; // Block caller thread until result is received
Object o = output.get(id);
output.remove(id);
ids.remove(id);
return o;
}
/**
* Waits for current method invocation to finish before stopping thread.
*/
public void stopGraceful(){
alive = false;
}
/**
* Interrupts thread to kill it. NOTE: This method is unsafe and could cause issues if it is performing an I/O operation.
* Only call this if you are 100% sure that it won't break something or if are ready to deal with the consequences.
*/
public void stopForced(){
alive = false;
stop();
}
}