Minor update
- Added support for immediately posting values without async starting - Added method to safely set field values. It handles cases of static- or object-fields - A note has been added to the JavaDoc. If you are considering using this method, please read the JavaDoc as it contains some vital information - Properly defined the IgnorantRunnable - Replaced all calls to sun.misc.Unsafe.putBoolean() with putInt() to support Android
This commit is contained in:
parent
b0fe7287f3
commit
3d12866dc2
@ -2,7 +2,7 @@ package com.tofvesson.async;
|
|||||||
|
|
||||||
import java.lang.reflect.*;
|
import java.lang.reflect.*;
|
||||||
|
|
||||||
@SuppressWarnings({"WeakerAccess", "unused", "unchecked"})
|
@SuppressWarnings({"WeakerAccess", "unused", "unchecked", "SameParameterValue"})
|
||||||
public class Async<T> implements Awaitable{
|
public class Async<T> implements Awaitable{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -158,6 +158,13 @@ public class Async<T> implements Awaitable{
|
|||||||
*/
|
*/
|
||||||
public Async(Constructor<T> c){ this(c, (Object[]) null); }
|
public Async(Constructor<T> c){ this(c, (Object[]) null); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Should probably only be used in debugging applications when a preset return value should be given and there is no need to start a new thread.
|
||||||
|
* This won't even start a new thread. All it does is set it's current state to "finished" and allow for the return value to be read.
|
||||||
|
* @param o Return value.
|
||||||
|
*/
|
||||||
|
public Async(T o){ complete = true; ret = o; }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* WARNING: Package-scoped because it should only be used when overriding standard construction. Should not bw used haphazardly!
|
* WARNING: Package-scoped because it should only be used when overriding standard construction. Should not bw used haphazardly!
|
||||||
*/
|
*/
|
||||||
|
@ -4,6 +4,6 @@ package com.tofvesson.async;
|
|||||||
* A Runnable-esque interface that allows for running code without try-catch blocks.
|
* A Runnable-esque interface that allows for running code without try-catch blocks.
|
||||||
*/
|
*/
|
||||||
public abstract class IgnorantRunnable implements Runnable{
|
public abstract class IgnorantRunnable implements Runnable{
|
||||||
public void run(){ try { irun(); } catch (Throwable throwable) { throw new RuntimeException(throwable); } }
|
public final void run(){ try { irun(); } catch (Throwable throwable) { throw new RuntimeException(throwable); } }
|
||||||
abstract void irun() throws Throwable;
|
public abstract void irun() throws Throwable;
|
||||||
}
|
}
|
||||||
|
@ -8,11 +8,13 @@ import java.util.*;
|
|||||||
/**
|
/**
|
||||||
* Safe tools to help simplify code when dealing with reflection.
|
* Safe tools to help simplify code when dealing with reflection.
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings({"unused", "SameParameterValue", "UnusedReturnValue"})
|
||||||
public final class SafeReflection {
|
public final class SafeReflection {
|
||||||
|
|
||||||
|
|
||||||
private static final Unsafe unsafe;
|
public static final Unsafe unsafe;
|
||||||
private static final Method newInstance, aConAccess;
|
private static final Method newInstance, aConAccess;
|
||||||
|
private static final Field modifiers;
|
||||||
private static final String version;
|
private static final String version;
|
||||||
private static final long override;
|
private static final long override;
|
||||||
private static final boolean hasAConAccess;
|
private static final boolean hasAConAccess;
|
||||||
@ -20,7 +22,7 @@ public final class SafeReflection {
|
|||||||
static{
|
static{
|
||||||
Unsafe u = null;
|
Unsafe u = null;
|
||||||
Method m = null, m1 = null;
|
Method m = null, m1 = null;
|
||||||
Field f;
|
Field f = null;
|
||||||
long l = 0;
|
long l = 0;
|
||||||
String ver = "";
|
String ver = "";
|
||||||
boolean b = true;
|
boolean b = true;
|
||||||
@ -34,23 +36,30 @@ public final class SafeReflection {
|
|||||||
f = Unsafe.class.getDeclaredField("theUnsafe");
|
f = Unsafe.class.getDeclaredField("theUnsafe");
|
||||||
f.setAccessible(true);
|
f.setAccessible(true);
|
||||||
u = (Unsafe) f.get(null);
|
u = (Unsafe) f.get(null);
|
||||||
l=u.objectFieldOffset(AccessibleObject.class.getDeclaredField("override"));
|
f = Field.class.getDeclaredField("modifiers");
|
||||||
|
f.setAccessible(true);
|
||||||
|
try {
|
||||||
|
l = u.objectFieldOffset(AccessibleObject.class.getDeclaredField("override")); // Most desktop versions of Java
|
||||||
|
}catch(Exception e){
|
||||||
|
l = u.objectFieldOffset(AccessibleObject.class.getDeclaredField("flag")); // God-damned Android
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
m1 = Constructor.class.getDeclaredMethod("acquireConstructorAccessor");
|
m1 = Constructor.class.getDeclaredMethod("acquireConstructorAccessor");
|
||||||
m1.setAccessible(true);
|
m1.setAccessible(true);
|
||||||
m = Class.forName(ver + ".DelegatingConstructorAccessorImpl").getDeclaredMethod("newInstance", Object[].class);
|
m = Class.forName(ver + ".DelegatingConstructorAccessorImpl").getDeclaredMethod("newInstance", Object[].class);
|
||||||
u.putBoolean(m, l, true);
|
u.putInt(m, l, 1);
|
||||||
}catch(Exception e){
|
}catch(Exception e){
|
||||||
b = false;
|
b = false;
|
||||||
}
|
}
|
||||||
u.putBoolean(f, l, true);
|
u.putInt(f, l, 1);
|
||||||
}catch(Exception ignored){ ignored.printStackTrace(); } // Exception is never thrown
|
}catch(Exception ignored){ ignored.printStackTrace(); } // Exception is never thrown
|
||||||
unsafe = u;
|
unsafe = u;
|
||||||
newInstance = m;
|
newInstance = m;
|
||||||
aConAccess = m1;
|
aConAccess = m1;
|
||||||
override = l;
|
override = l;
|
||||||
version = ver;
|
version = ver;
|
||||||
hasAConAccess = b;
|
hasAConAccess = b;
|
||||||
|
modifiers = f;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -63,7 +72,7 @@ public final class SafeReflection {
|
|||||||
public static <T> Constructor<T> getConstructor(Class<T> c, Class<?>... params){
|
public static <T> Constructor<T> getConstructor(Class<T> c, Class<?>... params){
|
||||||
try{
|
try{
|
||||||
Constructor<T> c1 = c.getConstructor(params);
|
Constructor<T> c1 = c.getConstructor(params);
|
||||||
unsafe.putBoolean(c1, override, true);
|
unsafe.putInt(c1, override, 1);
|
||||||
return c1;
|
return c1;
|
||||||
}catch(Exception e){}
|
}catch(Exception e){}
|
||||||
return null;
|
return null;
|
||||||
@ -79,7 +88,7 @@ public final class SafeReflection {
|
|||||||
try {
|
try {
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
Constructor<T> c1 = (Constructor<T>) c.getDeclaredConstructors()[0];
|
Constructor<T> c1 = (Constructor<T>) c.getDeclaredConstructors()[0];
|
||||||
unsafe.putBoolean(c1, override, true);
|
unsafe.putInt(c1, override, 1);
|
||||||
return c1;
|
return c1;
|
||||||
}catch (Exception e){}
|
}catch (Exception e){}
|
||||||
return null;
|
return null;
|
||||||
@ -96,7 +105,7 @@ public final class SafeReflection {
|
|||||||
public static Method getMethod(Class<?> c, String name, Class<?>... params){
|
public static Method getMethod(Class<?> c, String name, Class<?>... params){
|
||||||
try{
|
try{
|
||||||
Method m = c.getDeclaredMethod(name, params);
|
Method m = c.getDeclaredMethod(name, params);
|
||||||
unsafe.putBoolean(m, override, true);
|
unsafe.putInt(m, override, 1);
|
||||||
return m;
|
return m;
|
||||||
}catch(Exception e){}
|
}catch(Exception e){}
|
||||||
return null;
|
return null;
|
||||||
@ -121,7 +130,7 @@ public final class SafeReflection {
|
|||||||
* @return Return value of method or null if method is null.
|
* @return Return value of method or null if method is null.
|
||||||
*/
|
*/
|
||||||
public static Object invokeMethod(Object inst, Method m, Object... params){
|
public static Object invokeMethod(Object inst, Method m, Object... params){
|
||||||
if(m!=null) unsafe.putBoolean(m, override, true);
|
if(m!=null) unsafe.putInt(m, override, 1);
|
||||||
try{ return m!=null?m.invoke(inst, params):null; }catch(Exception e){}
|
try{ return m!=null?m.invoke(inst, params):null; }catch(Exception e){}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -157,7 +166,7 @@ public final class SafeReflection {
|
|||||||
public static Method getFirstMethod(Class<?> c, String name){
|
public static Method getFirstMethod(Class<?> c, String name){
|
||||||
try{
|
try{
|
||||||
Method[] m = c.getDeclaredMethods();
|
Method[] m = c.getDeclaredMethods();
|
||||||
for (Method aM : m) if(aM.getName().equals(name)){ unsafe.putBoolean(aM, override, true); return aM;}
|
for (Method aM : m) if(aM.getName().equals(name)){ unsafe.putInt(aM, override, 1); return aM;}
|
||||||
}catch(Exception e){}
|
}catch(Exception e){}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -171,7 +180,7 @@ public final class SafeReflection {
|
|||||||
public static Field getField(Class<?> c, String name){
|
public static Field getField(Class<?> c, String name){
|
||||||
try{
|
try{
|
||||||
Field f = c.getDeclaredField(name);
|
Field f = c.getDeclaredField(name);
|
||||||
unsafe.putBoolean(f, override, true);
|
unsafe.putInt(f, override, 1);
|
||||||
return f;
|
return f;
|
||||||
}catch(Exception e){}
|
}catch(Exception e){}
|
||||||
return null;
|
return null;
|
||||||
@ -201,6 +210,28 @@ public final class SafeReflection {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set field to specified value.
|
||||||
|
* <br><h1 style="text-align: center; color: red;">Please note:</h1>A JIT compiler may inline private final fields in methods which will prevent the actual value
|
||||||
|
* from changing at runtime.<br>This means that the value stored in the field <i>will</i> be changed, but any methods referring directly
|
||||||
|
* (not by java.lang.reflect.Field or sun.misc.Unsafe) to the field in the source will not be affected.
|
||||||
|
* This should only happen, though, if the field isn't set during runtime i.e. in a static block or constructor. This means that only fields that are truly constant
|
||||||
|
* like<br>"<i>public static final boolean b = false;</i>"<br>might be problematic.
|
||||||
|
* @param inv Object whose field to set the value of. Can be null.
|
||||||
|
* @param f Field to modify.
|
||||||
|
* @param value Value to set the field to.
|
||||||
|
* @return True if setting value succeeded.
|
||||||
|
*/
|
||||||
|
public static boolean setFieldValue(Object inv, Field f, Object value){
|
||||||
|
try{
|
||||||
|
unsafe.putInt(f, override, 1); // Ensure field override flag is set
|
||||||
|
if(Modifier.isFinal(f.getModifiers()) && Modifier.isStatic(f.getModifiers())) modifiers.setInt(f, f.getModifiers() &~ Modifier.FINAL);
|
||||||
|
f.set(inv, value);
|
||||||
|
return true;
|
||||||
|
}catch(Exception e){ e.printStackTrace(); }
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets a reference to the enclosing class from a defined inner (nested) object.
|
* Gets a reference to the enclosing class from a defined inner (nested) object.
|
||||||
* @param nested Object instance of a nested class.
|
* @param nested Object instance of a nested class.
|
||||||
@ -209,7 +240,7 @@ public final class SafeReflection {
|
|||||||
public static Object getEnclosingClassObjectRef(Object nested){
|
public static Object getEnclosingClassObjectRef(Object nested){
|
||||||
try{
|
try{
|
||||||
Field f = nested.getClass().getDeclaredField("this$0");
|
Field f = nested.getClass().getDeclaredField("this$0");
|
||||||
unsafe.putBoolean(f, override, true);
|
unsafe.putInt(f, override, 1);
|
||||||
return f.get(nested);
|
return f.get(nested);
|
||||||
}catch(Exception e){}
|
}catch(Exception e){}
|
||||||
return null;
|
return null;
|
||||||
@ -243,7 +274,7 @@ public final class SafeReflection {
|
|||||||
if(def==null) def = new EnumDefinition();
|
if(def==null) def = new EnumDefinition();
|
||||||
// Get a reference to the static method values() and get values
|
// Get a reference to the static method values() and get values
|
||||||
Method v = clazz.getDeclaredMethod("values");
|
Method v = clazz.getDeclaredMethod("values");
|
||||||
unsafe.putBoolean(v, override, true);
|
unsafe.putInt(v, override, 1);
|
||||||
T[] values = (T[]) v.invoke(null);
|
T[] values = (T[]) v.invoke(null);
|
||||||
|
|
||||||
// Return object if it already exists
|
// Return object if it already exists
|
||||||
@ -258,7 +289,7 @@ public final class SafeReflection {
|
|||||||
|
|
||||||
// Get enum constructor (inherited from Enum.class)
|
// Get enum constructor (inherited from Enum.class)
|
||||||
Constructor<T> c = clazz.getDeclaredConstructor(paramList);
|
Constructor<T> c = clazz.getDeclaredConstructor(paramList);
|
||||||
if(hasAConAccess) unsafe.putBoolean(c, override, true);
|
if(hasAConAccess) unsafe.putInt(c, override, 1);
|
||||||
else c.setAccessible(true);
|
else c.setAccessible(true);
|
||||||
|
|
||||||
Object[] parameters = new Object[def.params.size()+2];
|
Object[] parameters = new Object[def.params.size()+2];
|
||||||
@ -307,6 +338,7 @@ public final class SafeReflection {
|
|||||||
/**
|
/**
|
||||||
* A definition for custom enum creation.
|
* A definition for custom enum creation.
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("UnusedReturnValue")
|
||||||
public static final class EnumDefinition {
|
public static final class EnumDefinition {
|
||||||
HashMap<Object, Class> params = new HashMap<Object, Class>(); // Assign a specific type to each parameter
|
HashMap<Object, Class> params = new HashMap<Object, Class>(); // Assign a specific type to each parameter
|
||||||
|
|
||||||
@ -342,7 +374,7 @@ public final class SafeReflection {
|
|||||||
* @param b Boolean parameter.
|
* @param b Boolean parameter.
|
||||||
* @return A reference to the EnumDefinition object this method was called on (for chaining).
|
* @return A reference to the EnumDefinition object this method was called on (for chaining).
|
||||||
*/
|
*/
|
||||||
public EnumDefinition putBoolean(boolean b){
|
public EnumDefinition putInt(boolean b){
|
||||||
try { return putPrimitive(b); } catch (NotAutoBoxedException e) { }
|
try { return putPrimitive(b); } catch (NotAutoBoxedException e) { }
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
@ -458,5 +490,4 @@ public final class SafeReflection {
|
|||||||
}
|
}
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user