corda/classpath/java/lang/Class.java

405 lines
9.8 KiB
Java
Raw Normal View History

package java.lang;
2007-07-26 20:39:53 -06:00
import java.lang.reflect.Constructor;
2007-07-23 19:44:20 -06:00
import java.lang.reflect.Method;
import java.lang.reflect.Field;
2007-07-27 17:56:19 -06:00
import java.lang.reflect.Modifier;
import java.lang.reflect.InvocationTargetException;
import java.io.InputStream;
2007-09-13 21:59:39 -06:00
import java.io.IOException;
import java.net.URL;
2007-07-23 19:44:20 -06:00
public final class Class <T> {
private static final int PrimitiveFlag = 1 << 4;
private short flags;
private byte vmFlags;
private byte arrayDimensions;
private short fixedSize;
private short arrayElementSize;
private int[] objectMask;
private byte[] name;
private Class super_;
2007-07-23 19:44:20 -06:00
private Object[] interfaceTable;
private Method[] virtualTable;
private Field[] fieldTable;
private Method[] methodTable;
2007-11-02 15:08:14 -06:00
private Object staticTable;
2007-07-30 17:19:05 -06:00
private ClassLoader loader;
private Class() { }
2007-08-19 20:57:32 -06:00
public String toString() {
return getName();
}
private static byte[] replace(int a, int b, byte[] s, int offset,
int length)
{
byte[] array = new byte[length];
for (int i = 0; i < length; ++i) {
byte c = s[i];
array[i] = (byte) (c == a ? b : c);
}
return array;
}
public String getName() {
return new String
(replace('/', '.', name, 0, name.length - 1), 0, name.length - 1, false);
}
2007-07-23 19:44:20 -06:00
2007-11-02 15:08:14 -06:00
public Object staticTable() {
return staticTable;
}
public T newInstance()
throws IllegalAccessException, InstantiationException
{
try {
return (T) getConstructor().newInstance();
} catch (NoSuchMethodException e) {
throw new RuntimeException(e);
} catch (InvocationTargetException e) {
throw new RuntimeException(e);
}
}
2007-07-30 17:19:05 -06:00
public static Class forName(String name) throws ClassNotFoundException {
return forName
(name, true, Method.getCaller().getDeclaringClass().getClassLoader());
}
public static Class forName(String name, boolean initialize,
ClassLoader loader)
throws ClassNotFoundException
{
Class c = loader.loadClass(name);
if (initialize) {
c.initialize();
2007-07-30 17:19:05 -06:00
}
return c;
}
private static native Class primitiveClass(char name);
private native void initialize();
public static Class forCanonicalName(String name) {
try {
if (name.startsWith("[")) {
return forName(name);
} else if (name.startsWith("L")) {
return forName(name.substring(1, name.length() - 1));
} else {
if (name.length() == 1) {
return primitiveClass(name.charAt(0));
} else {
throw new ClassNotFoundException(name);
}
}
} catch (ClassNotFoundException e) {
throw new RuntimeException(e);
}
}
2007-07-23 19:44:20 -06:00
public Class getComponentType() {
if (isArray()) {
return (Class) staticTable;
} else {
return null;
}
}
2007-07-23 19:44:20 -06:00
public native boolean isAssignableFrom(Class c);
2007-07-26 20:39:53 -06:00
private Field findField(String name) {
2007-07-28 10:10:13 -06:00
if (fieldTable != null) {
for (int i = 0; i < fieldTable.length; ++i) {
if (fieldTable[i].getName().equals(name)) {
return fieldTable[i];
}
2007-07-23 19:44:20 -06:00
}
}
2007-07-26 20:39:53 -06:00
return null;
}
2007-07-23 19:44:20 -06:00
2007-07-26 20:39:53 -06:00
public Field getDeclaredField(String name) throws NoSuchFieldException {
Field f = findField(name);
if (f == null) {
throw new NoSuchFieldException(name);
} else {
return f;
}
}
public Field getField(String name) throws NoSuchFieldException {
for (Class c = this; c != null; c = c.super_) {
Field f = c.findField(name);
if (f != null) {
return f;
}
}
2007-07-23 19:44:20 -06:00
throw new NoSuchFieldException(name);
}
private static boolean match(Class[] a, Class[] b) {
if (a.length == b.length) {
for (int i = 0; i < a.length; ++i) {
if (! a[i].isAssignableFrom(b[i])) {
return false;
}
}
return true;
} else {
return false;
}
}
2007-07-26 20:39:53 -06:00
private Method findMethod(String name, Class[] parameterTypes) {
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
for (int i = 0; i < methodTable.length; ++i) {
if (methodTable[i].getName().equals(name)
&& match(parameterTypes, methodTable[i].getParameterTypes()))
{
return methodTable[i];
}
2007-07-23 19:44:20 -06:00
}
}
2007-07-26 20:39:53 -06:00
return null;
}
public Method getDeclaredMethod(String name, Class ... parameterTypes)
throws NoSuchMethodException
{
2007-07-27 17:56:19 -06:00
if (name.startsWith("<")) {
throw new NoSuchMethodException(name);
}
Method m = findMethod(name, parameterTypes);
if (m == null) {
2007-07-26 20:39:53 -06:00
throw new NoSuchMethodException(name);
} else {
2007-07-27 17:56:19 -06:00
return m;
2007-07-26 20:39:53 -06:00
}
}
2007-07-23 19:44:20 -06:00
2007-07-26 20:39:53 -06:00
public Method getMethod(String name, Class ... parameterTypes)
throws NoSuchMethodException
{
2007-07-27 17:56:19 -06:00
if (name.startsWith("<")) {
throw new NoSuchMethodException(name);
}
2007-07-26 20:39:53 -06:00
for (Class c = this; c != null; c = c.super_) {
2007-07-27 17:56:19 -06:00
Method m = c.findMethod(name, parameterTypes);
if (m != null) {
return m;
2007-07-26 20:39:53 -06:00
}
}
2007-07-23 19:44:20 -06:00
throw new NoSuchMethodException(name);
}
2007-07-26 20:39:53 -06:00
public Constructor getConstructor(Class ... parameterTypes)
throws NoSuchMethodException
{
2007-07-27 17:56:19 -06:00
Method m = findMethod("<init>", parameterTypes);
if (m == null) {
throw new NoSuchMethodException();
} else {
return new Constructor(m);
}
2007-07-26 20:39:53 -06:00
}
2007-07-27 17:56:19 -06:00
private int countConstructors(boolean publicOnly) {
2007-07-26 20:39:53 -06:00
int count = 0;
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
for (int i = 0; i < methodTable.length; ++i) {
if (((! publicOnly)
|| ((methodTable[i].getModifiers() & Modifier.PUBLIC)) != 0)
&& methodTable[i].getName().equals("<init>"))
{
++ count;
}
2007-07-26 20:39:53 -06:00
}
}
2007-07-27 17:56:19 -06:00
return count;
}
2007-07-26 20:39:53 -06:00
2007-07-27 17:56:19 -06:00
public Constructor[] getDeclaredConstructors() {
Constructor[] array = new Constructor[countConstructors(false)];
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
int index = 0;
for (int i = 0; i < methodTable.length; ++i) {
if (methodTable[i].getName().equals("<init>")) {
array[index++] = new Constructor(methodTable[i]);
}
2007-07-26 20:39:53 -06:00
}
}
return array;
}
2007-07-27 17:56:19 -06:00
public Constructor[] getConstructors() {
Constructor[] array = new Constructor[countConstructors(true)];
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
int index = 0;
for (int i = 0; i < methodTable.length; ++i) {
if (((methodTable[i].getModifiers() & Modifier.PUBLIC) != 0)
&& methodTable[i].getName().equals("<init>"))
{
array[index++] = new Constructor(methodTable[i]);
}
2007-07-27 17:56:19 -06:00
}
}
return array;
2007-07-26 20:39:53 -06:00
}
public Field[] getDeclaredFields() {
2007-07-28 10:10:13 -06:00
if (fieldTable != null) {
Field[] array = new Field[fieldTable.length];
System.arraycopy(fieldTable, 0, array, 0, fieldTable.length);
return array;
} else {
return new Field[0];
}
2007-07-26 20:39:53 -06:00
}
2007-07-27 17:56:19 -06:00
private int countPublicFields() {
int count = 0;
2007-07-28 10:10:13 -06:00
if (fieldTable != null) {
for (int i = 0; i < fieldTable.length; ++i) {
if (((fieldTable[i].getModifiers() & Modifier.PUBLIC)) != 0) {
++ count;
}
2007-07-27 17:56:19 -06:00
}
}
return count;
}
public Field[] getFields() {
Field[] array = new Field[countPublicFields()];
2007-07-28 10:10:13 -06:00
if (fieldTable != null) {
int ai = 0;
2007-07-28 10:10:13 -06:00
for (int i = 0; i < fieldTable.length; ++i) {
if (((fieldTable[i].getModifiers() & Modifier.PUBLIC)) != 0) {
array[ai++] = fieldTable[i];
2007-07-28 10:10:13 -06:00
}
2007-07-27 17:56:19 -06:00
}
}
return array;
}
private int countMethods(boolean publicOnly) {
2007-07-26 20:39:53 -06:00
int count = 0;
2007-08-19 20:57:32 -06:00
if (methodTable != null) {
for (int i = 0; i < methodTable.length; ++i) {
if (((! publicOnly)
|| ((methodTable[i].getModifiers() & Modifier.PUBLIC)) != 0)
&& (! methodTable[i].getName().startsWith("<")))
{
++ count;
}
2007-07-26 20:39:53 -06:00
}
}
2007-07-27 17:56:19 -06:00
return count;
}
2007-07-26 20:39:53 -06:00
2007-07-27 17:56:19 -06:00
public Method[] getDeclaredMethods() {
Method[] array = new Method[countMethods(false)];
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
int ai = 0;
2007-07-28 10:10:13 -06:00
for (int i = 0; i < methodTable.length; ++i) {
if (! methodTable[i].getName().startsWith("<")) {
array[ai++] = methodTable[i];
2007-07-28 10:10:13 -06:00
}
2007-07-27 17:56:19 -06:00
}
}
return array;
}
public Method[] getMethods() {
Method[] array = new Method[countMethods(true)];
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
int index = 0;
for (int i = 0; i < methodTable.length; ++i) {
if (((methodTable[i].getModifiers() & Modifier.PUBLIC) != 0)
&& (! methodTable[i].getName().startsWith("<")))
{
array[index++] = methodTable[i];
}
2007-07-26 20:39:53 -06:00
}
}
return array;
}
public Class[] getInterfaces() {
2007-07-28 10:10:13 -06:00
if (interfaceTable != null) {
Class[] array = new Class[interfaceTable.length / 2];
for (int i = 0; i < array.length; ++i) {
array[i] = (Class) interfaceTable[i * 2];
}
return array;
} else {
return new Class[0];
2007-07-26 20:39:53 -06:00
}
}
2007-11-17 11:39:29 -07:00
public T[] getEnumConstants() {
if (Enum.class.isAssignableFrom(this)) {
2007-11-17 11:39:29 -07:00
try {
return (T[]) getMethod("values").invoke(null);
} catch (Exception e) {
throw new Error();
}
} else {
return null;
}
}
2007-07-26 20:39:53 -06:00
public ClassLoader getClassLoader() {
2007-07-30 17:19:05 -06:00
return loader;
2007-07-26 20:39:53 -06:00
}
public int getModifiers() {
return flags;
}
public Class getSuperclass() {
return super_;
}
public boolean isArray() {
return arrayElementSize != 0;
}
public boolean isInstance(Object o) {
return o != null && isAssignableFrom(o.getClass());
2007-07-26 20:39:53 -06:00
}
public boolean isPrimitive() {
return (vmFlags & PrimitiveFlag) != 0;
2007-07-26 20:39:53 -06:00
}
2007-09-13 21:59:39 -06:00
public URL getResource(String path) {
if (! path.startsWith("/")) {
2007-09-13 21:59:39 -06:00
String name = new String(this.name, 0, this.name.length - 1, false);
int index = name.lastIndexOf('/');
if (index >= 0) {
path = name.substring(0, index) + "/" + path;
}
}
return getClassLoader().getResource(path);
}
public InputStream getResourceAsStream(String path) {
URL url = getResource(path);
try {
return (url == null ? null : url.openStream());
} catch (IOException e) {
return null;
}
}
}