2007-07-21 17:50:26 +00:00
|
|
|
package java.lang.reflect;
|
|
|
|
|
2007-07-21 20:44:39 +00:00
|
|
|
public class Method<T> extends AccessibleObject implements Member {
|
2007-07-21 17:50:26 +00:00
|
|
|
private byte vmFlags;
|
2007-09-26 23:23:03 +00:00
|
|
|
private byte returnCode;
|
2007-07-21 17:50:26 +00:00
|
|
|
private byte parameterCount;
|
2007-09-26 23:23:03 +00:00
|
|
|
private byte parameterFootprint;
|
2007-07-21 17:50:26 +00:00
|
|
|
private short flags;
|
|
|
|
private short offset;
|
|
|
|
private byte[] name;
|
|
|
|
private byte[] spec;
|
|
|
|
private Class<T> class_;
|
|
|
|
private Object code;
|
2007-10-04 00:41:54 +00:00
|
|
|
private long compiled;
|
2007-07-21 17:50:26 +00:00
|
|
|
|
|
|
|
private Method() { }
|
|
|
|
|
|
|
|
public boolean isAccessible() {
|
|
|
|
return (vmFlags & Accessible) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setAccessible(boolean v) {
|
|
|
|
if (v) vmFlags |= Accessible; else vmFlags &= ~Accessible;
|
|
|
|
}
|
2007-07-21 20:44:39 +00:00
|
|
|
|
2007-07-30 23:19:05 +00:00
|
|
|
public static native Method getCaller();
|
|
|
|
|
2007-07-21 20:44:39 +00:00
|
|
|
public Class<T> getDeclaringClass() {
|
|
|
|
return class_;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getModifiers() {
|
|
|
|
return flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getName() {
|
|
|
|
return new String(name, 0, name.length - 1, false);
|
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
|
|
|
|
private static int next(char c, String s, int start) {
|
|
|
|
for (int i = start; i < s.length(); ++i) {
|
|
|
|
if (s.charAt(i) == c) return i;
|
|
|
|
}
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Class[] getParameterTypes() {
|
|
|
|
int count = parameterCount;
|
|
|
|
|
|
|
|
Class[] types = new Class[count];
|
|
|
|
int index = 0;
|
|
|
|
|
|
|
|
String spec = new String(this.spec, 1, this.spec.length - 1, false);
|
|
|
|
|
2007-07-28 16:55:24 +00:00
|
|
|
try {
|
|
|
|
for (int i = 0; i < spec.length(); ++i) {
|
|
|
|
char c = spec.charAt(i);
|
|
|
|
if (c == ')') {
|
|
|
|
break;
|
|
|
|
} else if (c == 'L') {
|
|
|
|
int start = i + 1;
|
|
|
|
i = next(';', spec, start);
|
2007-07-24 03:16:59 +00:00
|
|
|
String name = spec.substring(start, i);
|
2007-07-24 01:44:20 +00:00
|
|
|
types[index++] = Class.forName(name);
|
2007-07-28 16:55:24 +00:00
|
|
|
} else if (c == '[') {
|
|
|
|
int start = i;
|
|
|
|
while (spec.charAt(i) == '[') ++i;
|
|
|
|
|
|
|
|
if (spec.charAt(i) == 'L') {
|
|
|
|
i = next(';', spec, i + 1);
|
|
|
|
String name = spec.substring(start, i);
|
|
|
|
types[index++] = Class.forName(name);
|
|
|
|
} else {
|
|
|
|
String name = spec.substring(start, i + 1);
|
2007-08-20 02:57:32 +00:00
|
|
|
types[index++] = Class.forCanonicalName(name);
|
2007-07-28 16:55:24 +00:00
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
} else {
|
2007-07-28 16:55:24 +00:00
|
|
|
String name = spec.substring(i, i + 1);
|
2007-08-20 02:57:32 +00:00
|
|
|
types[index++] = Class.forCanonicalName(name);
|
2007-07-24 01:44:20 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-28 16:55:24 +00:00
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
throw new RuntimeException(e);
|
2007-07-24 01:44:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return types;
|
|
|
|
}
|
|
|
|
|
2007-08-18 17:53:30 +00:00
|
|
|
public Object invoke(Object instance, Object ... arguments)
|
|
|
|
throws InvocationTargetException, IllegalAccessException
|
|
|
|
{
|
2007-09-26 23:23:03 +00:00
|
|
|
if ((flags & Modifier.STATIC) != 0 || class_.isInstance(instance)) {
|
2007-08-18 17:53:30 +00:00
|
|
|
if (arguments.length == parameterCount) {
|
|
|
|
return invoke(this, instance, arguments);
|
|
|
|
} else {
|
|
|
|
throw new ArrayIndexOutOfBoundsException();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
throw new IllegalArgumentException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static native Object invoke(Method method, Object instance,
|
|
|
|
Object ... arguments)
|
2007-08-14 00:37:00 +00:00
|
|
|
throws InvocationTargetException, IllegalAccessException;
|
2007-09-14 02:19:44 +00:00
|
|
|
|
|
|
|
public Class getReturnType() {
|
|
|
|
for (int i = 0; i < spec.length - 1; ++i) {
|
|
|
|
if (spec[i] == ')') {
|
|
|
|
return Class.forCanonicalName
|
|
|
|
(new String(spec, i + 1, spec.length - i - 2, false));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
2007-07-21 17:50:26 +00:00
|
|
|
}
|