Minor update
- Added more Android compatibility code
This commit is contained in:
parent
8ef3df8245
commit
da48afd91c
@ -164,12 +164,12 @@ public class Async<T> implements Awaitable{
|
|||||||
* @param o Return value.
|
* @param o Return value.
|
||||||
*/
|
*/
|
||||||
public Async(T o){ complete = true; ret = o; }
|
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!
|
||||||
*/
|
*/
|
||||||
Async() { task = null; }
|
Async() { task = null; }
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 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.
|
||||||
|
@ -11,4 +11,8 @@ public final class Collections {
|
|||||||
for(int i = c.size()-1; i>0; --i) a.add(t[i]);
|
for(int i = c.size()-1; i>0; --i) a.add(t[i]);
|
||||||
return a;
|
return a;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static boolean arrayContains(Object[] o, PredicateCompat p){ for(Object o1 : o) if(p.apply(o1)) return true; return false; }
|
||||||
|
|
||||||
|
public interface PredicateCompat<T>{ boolean apply(T t); }
|
||||||
}
|
}
|
||||||
|
@ -1,14 +1,14 @@
|
|||||||
package com.tofvesson.reflection;
|
package com.tofvesson.reflection;
|
||||||
|
|
||||||
import sun.misc.Unsafe;
|
import sun.misc.Unsafe;
|
||||||
|
|
||||||
import java.lang.reflect.*;
|
import java.lang.reflect.*;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
import java.util.Collections;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Safe tools to help simplify code when dealing with reflection.
|
* Safe tools to help simplify code when dealing with reflection.
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings({"unused", "SameParameterValue", "UnusedReturnValue"})
|
@SuppressWarnings({"unused", "SameParameterValue", "UnusedReturnValue", "ConstantConditions"})
|
||||||
public final class SafeReflection {
|
public final class SafeReflection {
|
||||||
|
|
||||||
|
|
||||||
@ -39,11 +39,14 @@ public final class SafeReflection {
|
|||||||
try {
|
try {
|
||||||
f = Field.class.getDeclaredField("modifiers");
|
f = Field.class.getDeclaredField("modifiers");
|
||||||
f.setAccessible(true);
|
f.setAccessible(true);
|
||||||
l = u.objectFieldOffset(AccessibleObject.class.getDeclaredField("override")); // Most desktop versions of Java
|
|
||||||
}catch(Exception e){
|
}catch(Exception e){
|
||||||
try { f = Field.class.getDeclaredField("artField").getType().getDeclaredField("accessFlags"); }
|
try { f = Field.class.getDeclaredField("artField").getType().getDeclaredField("accessFlags"); }
|
||||||
catch(Exception ignored){ f = Field.class.getDeclaredField("accessFlags"); }
|
catch(Exception ignored){ f = Field.class.getDeclaredField("accessFlags"); }
|
||||||
f.setAccessible(true);
|
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
|
l = u.objectFieldOffset(AccessibleObject.class.getDeclaredField("flag")); // God-damned Android
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
@ -55,7 +58,7 @@ public final class SafeReflection {
|
|||||||
b = false;
|
b = false;
|
||||||
}
|
}
|
||||||
u.putInt(f, l, 1);
|
u.putInt(f, l, 1);
|
||||||
}catch(Exception ignored){ ignored.printStackTrace(); } // Exception is never thrown
|
}catch(Exception ignored){ ignored.printStackTrace(); } // Exception should never be thrown
|
||||||
unsafe = u;
|
unsafe = u;
|
||||||
newInstance = m;
|
newInstance = m;
|
||||||
aConAccess = m1;
|
aConAccess = m1;
|
||||||
@ -190,27 +193,46 @@ public final class SafeReflection {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the static object stored in the field with the specified name in the defined class.
|
* Gets the object stored in the field with the specified name in the class of the defined object.
|
||||||
* @param c Class to find object in.
|
* @param from Object to find object in.
|
||||||
|
* @param c Class to get field from (if <b>from</b> is null). This is used for static fields.
|
||||||
* @param name Name of field.
|
* @param name Name of field.
|
||||||
* @return Object or null if object is null or field doesn't exist.
|
* @return Object or null if object is null or field doesn't exist.
|
||||||
*/
|
*/
|
||||||
public static Object getStaticFieldObject(Class<?> c, String name){
|
public static Object getValue(Object from, Class<?> c, String name){
|
||||||
Field f = getField(c, name);
|
try{ return getField(from!=null?from.getClass():c, name).get(from); }catch(Throwable ignored){}
|
||||||
try { return f!=null?f.get(null):null; } catch (Exception e) { }
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the object stored in the field with the specified name in the class of the defined object.
|
* Gets the object stored in the static field with the specified name in the class of the defined object.
|
||||||
* @param o Object to find object in.
|
* @param c Class to get field from (if <b>from</b> is null). This is used for static fields.
|
||||||
* @param name Name of field.
|
* @param name Name of field.
|
||||||
* @return Object or null if object is null or field doesn't exist.
|
* @return Object or null if object is null or field doesn't exist.
|
||||||
*/
|
*/
|
||||||
public static Object getFieldObject(Object o, String name){
|
public static Object getValue(Class<?> c, String name){ return getValue(null, c, name); }
|
||||||
Field f = getField(o.getClass(), name);
|
|
||||||
try{ return f!=null?f.get(o):null; }catch(Exception e){}
|
/**
|
||||||
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 in Class to find field in. (If inv is null)
|
||||||
|
* @param name Name of field to modify.
|
||||||
|
* @param value Value to set the field to.
|
||||||
|
* @param vol Whether or not the operation should be volatile.
|
||||||
|
* @return True if setting value succeeded.
|
||||||
|
*/
|
||||||
|
public static boolean setValue(Object inv, Class<?> in, String name, Object value, boolean vol){
|
||||||
|
try{
|
||||||
|
Field f = getField(inv==null?in:inv.getClass(), name);
|
||||||
|
getPutMethod(f.getType(), vol).invoke(unsafe, createUnsafePutParams(inv, f, value));
|
||||||
|
return true;
|
||||||
|
}catch(Exception e){ e.printStackTrace(); }
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -221,19 +243,55 @@ public final class SafeReflection {
|
|||||||
* 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
|
* 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.
|
* 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 inv Object whose field to set the value of. Can be null.
|
||||||
* @param f Field to modify.
|
* @param in Class to find field in. (If inv is null)
|
||||||
|
* @param name Name of field to modify.
|
||||||
* @param value Value to set the field to.
|
* @param value Value to set the field to.
|
||||||
* @return True if setting value succeeded.
|
* @return True if setting value succeeded.
|
||||||
*/
|
*/
|
||||||
public static boolean setFieldValue(Object inv, Field f, Object value){
|
public static boolean setValue(Object inv, Class<?> in, String name, Object value){ return setValue(inv, in, name, value, false); }
|
||||||
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);
|
* Set field to specified value as a volatile operations.
|
||||||
f.set(inv, 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
|
||||||
return true;
|
* from changing at runtime.<br>This means that the value stored in the field <i>will</i> be changed, but any methods referring directly
|
||||||
}catch(Exception e){ e.printStackTrace(); }
|
* (not by java.lang.reflect.Field or sun.misc.Unsafe) to the field in the source will not be affected.
|
||||||
return false;
|
* 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 in Class to find field in. (If inv is null)
|
||||||
|
* @param name Name of field to modify.
|
||||||
|
* @param value Value to set the field to.
|
||||||
|
* @return True if setting value succeeded.
|
||||||
|
*/
|
||||||
|
public static boolean setValueVolatile(Object inv, Class<?> in, String name, Object value){ return setValue(inv, in, name, value, true); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set static 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 in Class to find field in.
|
||||||
|
* @param name Name of field to modify.
|
||||||
|
* @param value Value to set the field to.
|
||||||
|
* @return True if setting value succeeded.
|
||||||
|
*/
|
||||||
|
public static boolean setValue(Class<?> in, String name, Object value){ return setValue(null, in, name, value, false); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set static field to specified value as a volatile operation.
|
||||||
|
* <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 in Class to find field in.
|
||||||
|
* @param name Name of field to modify.
|
||||||
|
* @param value Value to set the field to.
|
||||||
|
* @return True if setting value succeeded.
|
||||||
|
*/
|
||||||
|
public static boolean setValueVolatile(Class<?> in, String name, Object value){ return setValue(null, in, name, value, true); }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 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.
|
||||||
@ -459,7 +517,7 @@ public final class SafeReflection {
|
|||||||
*/
|
*/
|
||||||
public static Class<?> getCallerClass(){
|
public static Class<?> getCallerClass(){
|
||||||
ArrayList<StackTraceElement> s = getStacktraceWithoutReflection();
|
ArrayList<StackTraceElement> s = getStacktraceWithoutReflection();
|
||||||
try { return Class.forName(s.get(s.size()==2?1:2).getClassName()); } catch (ClassNotFoundException e) { }
|
try { return Class.forName(s.get(s.size()==3?2:s.size()==2?1:0).getClassName()); } catch (ClassNotFoundException e) { }
|
||||||
assert false:"Unreachable code reached";
|
assert false:"Unreachable code reached";
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -493,4 +551,80 @@ public final class SafeReflection {
|
|||||||
}
|
}
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static Method getSetMethod(Class<?> c){
|
||||||
|
try {
|
||||||
|
Method m = Field.class.getDeclaredMethod(getSetMethodName(c), Object.class, isAutoBoxedClass(c)?unbox(c):c.isPrimitive()?c:Object.class);
|
||||||
|
m.setAccessible(true);
|
||||||
|
return m;
|
||||||
|
} catch (Exception e) { e.printStackTrace(); }
|
||||||
|
return null; // Not object version of primitive
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String getSetMethodName(Class<?> c){
|
||||||
|
try {
|
||||||
|
String s;
|
||||||
|
boolean b;
|
||||||
|
Field f = null;
|
||||||
|
try{
|
||||||
|
f = c.getDeclaredField("value");
|
||||||
|
}catch(Exception e){}
|
||||||
|
if(f!=null) f.setAccessible(true);
|
||||||
|
return"set"+
|
||||||
|
(f!=null && Number.class.isAssignableFrom(f.getType()) && f.getType().isPrimitive()?
|
||||||
|
(s=f.getType().getSimpleName()).replaceFirst(s.charAt(0)+"", Character.toUpperCase(s.charAt(0))+""):
|
||||||
|
c.isPrimitive()?(s=c.getSimpleName()).replaceFirst(s.charAt(0)+"", Character.toUpperCase(s.charAt(0))+"")
|
||||||
|
:"");
|
||||||
|
} catch (Exception e) { e.printStackTrace(); }
|
||||||
|
return null; // Not object version of primitive
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isAutoBoxedClass(final Class<?> c){
|
||||||
|
return Number.class.isAssignableFrom(c) && com.tofvesson.collections.Collections.arrayContains(c.getDeclaredFields(),
|
||||||
|
new com.tofvesson.collections.Collections.PredicateCompat() {
|
||||||
|
public boolean apply(Object o) {
|
||||||
|
return ((Field) o).getName().equals("value") && box(((Field) o).getType())==c;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Method getPutMethod(Class<?> c, boolean vol){
|
||||||
|
try{
|
||||||
|
Method m = Unsafe.class.getDeclaredMethod(
|
||||||
|
getSetMethodName(c).replaceFirst("set", "put") + (isAutoBoxedClass(c)||c.isPrimitive()?"":"Object")+(vol?"Volatile":""),
|
||||||
|
Object.class,
|
||||||
|
long.class,
|
||||||
|
c.isPrimitive()?c:Object.class
|
||||||
|
);
|
||||||
|
m.setAccessible(true);
|
||||||
|
return m;
|
||||||
|
}catch(Exception ignored){ ignored.printStackTrace(); }
|
||||||
|
return null; // Something went wrong...
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Object[] createUnsafePutParams(Object invokee, Field field, Object value){
|
||||||
|
return new Object[]{ invokee==null?field.getDeclaringClass():invokee, invokee==null?unsafe.staticFieldOffset(field):unsafe.objectFieldOffset(field), value };
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Class<?> box(Class<?> c){
|
||||||
|
if(!c.isPrimitive()) return c;
|
||||||
|
if(c==int.class) return Integer.class;
|
||||||
|
if(c==char.class) return Character.class;
|
||||||
|
if(c==byte.class) return Byte.class;
|
||||||
|
if(c==float.class) return Float.class;
|
||||||
|
if(c==long.class) return Long.class;
|
||||||
|
if(c==boolean.class) return Boolean.class;
|
||||||
|
return Double.class;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Class<?> unbox(Class<?> c){
|
||||||
|
if(c.isPrimitive()) return c;
|
||||||
|
if(c==Integer.class) return int.class;
|
||||||
|
if(c==Character.class) return char.class;
|
||||||
|
if(c==Byte.class) return byte.class;
|
||||||
|
if(c==Float.class) return float.class;
|
||||||
|
if(c==Long.class) return long.class;
|
||||||
|
if(c==Boolean.class) return boolean.class;
|
||||||
|
return double.class;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user