Major update

Fixed major bug in Async when constructing a new object causing an infinite loop
Added an Async constructor for constructing a new object without parameters
Async is now generic and return type of await() is also generic!
This commit is contained in:
Gabriel Tofvesson 2016-11-03 22:45:01 +01:00
parent 28c9b3b8e7
commit 9c014689a1

View File

@ -3,11 +3,11 @@ package com.tofvesson.async;
import java.lang.reflect.Constructor; import java.lang.reflect.Constructor;
import java.lang.reflect.Method; import java.lang.reflect.Method;
@SuppressWarnings({"WeakerAccess", "unused"}) @SuppressWarnings({"WeakerAccess", "unused", "unchecked"})
public class Async { public class Async<T> {
Thread task; Thread task;
volatile Object ret; // Assigned using native method volatile T ret; // Assigned using native method
volatile boolean complete = false, failed = false; // Used by anonymous class, therefore not private volatile boolean complete = false, failed = false; // Used by anonymous class, therefore not private
volatile Throwable t; volatile Throwable t;
@ -21,7 +21,7 @@ public class Async {
public Async(final Object o, final Method method, final Object... params){ public Async(final Object o, final Method method, final Object... params){
method.setAccessible(true); method.setAccessible(true);
task = new Thread(()-> { task = new Thread(()-> {
synchronized (this) { try { ret = method.invoke(o, params); complete = true; } catch (Throwable t1) { if(!failed) { failed = true; t=t1; } } } synchronized (this) { try { ret = (T) method.invoke(o, params); complete = true; } catch (Throwable t1) { if(!failed) { failed = true; t=t1; } } }
}); });
task.setDaemon(true); task.setDaemon(true);
task.start(); task.start();
@ -52,27 +52,28 @@ public class Async {
* @param c Constructor to use when instantiating object. * @param c Constructor to use when instantiating object.
* @param params Parameters to use when instantiaing object. * @param params Parameters to use when instantiaing object.
*/ */
public Async(final Constructor c, final Object... params){ public Async(final Constructor<T> c, final Object... params){
c.setAccessible(true); c.setAccessible(true);
//Lambdas are compiled into more methods than anonymous class and don't decrease overhead task = new Thread(() -> {
//noinspection Convert2Lambda synchronized (this) { try { ret = c.newInstance(params); complete = true; } catch (Throwable t1) { if(!failed) { failed = true; t=t1; } } }
task = new Thread(new Runnable() {
@Override
public void run() {
synchronized (this) { try { ret = c.newInstance(params); } catch (Throwable t1) { if(!failed) { failed = true; t=t1; } } }
}
}); });
task.setDaemon(true); task.setDaemon(true);
task.start(); task.start();
} }
/**
* Dispatch an asynchronous construction of the object corresponding to the passed constructor with no parameters.
* @param c Constructor to call.
*/
public Async(Constructor<T> c){ this(c, (Object[]) null); }
Async() {task = null;} // Only package-scoped because it should only be used when overriding standard construction Async() {task = null;} // Only package-scoped because it should only be used when overriding standard construction
/** /**
* Await completion of async task. Blocks thread if task isn't complete. * Await completion of async task. Blocks thread if task isn't complete.
* @return Return value from async method call. Return is null if {@link #cancel()} is called before this method and async task wan't finished. * @return Return value from async method call. Return is null if {@link #cancel()} is called before this method and async task wan't finished.
*/ */
public Object await(){ public T await(){
//noinspection StatementWithEmptyBody //noinspection StatementWithEmptyBody
while(!failed && !complete); // Check using variables rather than checking if worker thread is alive since method calls are more computationally expensive while(!failed && !complete); // Check using variables rather than checking if worker thread is alive since method calls are more computationally expensive
if(ret==null && t!=null) throw new RuntimeException(t); // Detect a unique error state, get error and throw in caller thread if(ret==null && t!=null) throw new RuntimeException(t); // Detect a unique error state, get error and throw in caller thread