mirror of
https://github.com/EsotericSoftware/reflectasm.git
synced 2025-01-07 18:30:22 +08:00
Added getReturnTypes() method
Completing the reexistant getMethodNames() and getParameterTypes()
This commit is contained in:
parent
2c80473aaf
commit
d68e51991e
@ -1,3 +1,4 @@
|
|||||||
|
|
||||||
package com.esotericsoftware.reflectasm;
|
package com.esotericsoftware.reflectasm;
|
||||||
|
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
@ -16,6 +17,7 @@ import static org.objectweb.asm.Opcodes.*;
|
|||||||
public abstract class MethodAccess {
|
public abstract class MethodAccess {
|
||||||
private String[] methodNames;
|
private String[] methodNames;
|
||||||
private Class[][] parameterTypes;
|
private Class[][] parameterTypes;
|
||||||
|
private Class[] returnTypes;
|
||||||
|
|
||||||
abstract public Object invoke (Object object, int methodIndex, Object... args);
|
abstract public Object invoke (Object object, int methodIndex, Object... args);
|
||||||
|
|
||||||
@ -58,6 +60,10 @@ public abstract class MethodAccess {
|
|||||||
return parameterTypes;
|
return parameterTypes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Class[] getReturnTypes () {
|
||||||
|
return returnTypes;
|
||||||
|
}
|
||||||
|
|
||||||
static public MethodAccess get (Class type) {
|
static public MethodAccess get (Class type) {
|
||||||
ArrayList<Method> methods = new ArrayList<Method>();
|
ArrayList<Method> methods = new ArrayList<Method>();
|
||||||
boolean isInterface = type.isInterface();
|
boolean isInterface = type.isInterface();
|
||||||
@ -72,12 +78,15 @@ public abstract class MethodAccess {
|
|||||||
recursiveAddInterfaceMethodsToList(type, methods);
|
recursiveAddInterfaceMethodsToList(type, methods);
|
||||||
}
|
}
|
||||||
|
|
||||||
Class[][] parameterTypes = new Class[methods.size()][];
|
int n = methods.size();
|
||||||
String[] methodNames = new String[methods.size()];
|
String[] methodNames = new String[n];
|
||||||
for (int i = 0, n = methodNames.length; i < n; i++) {
|
Class[][] parameterTypes = new Class[n][];
|
||||||
|
Class[] returnTypes = new Class[n];
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
Method method = methods.get(i);
|
Method method = methods.get(i);
|
||||||
methodNames[i] = method.getName();
|
methodNames[i] = method.getName();
|
||||||
parameterTypes[i] = method.getParameterTypes();
|
parameterTypes[i] = method.getParameterTypes();
|
||||||
|
returnTypes[i] = method.getReturnType();
|
||||||
}
|
}
|
||||||
|
|
||||||
String className = type.getName();
|
String className = type.getName();
|
||||||
@ -117,14 +126,14 @@ public abstract class MethodAccess {
|
|||||||
mv.visitVarInsn(ASTORE, 4);
|
mv.visitVarInsn(ASTORE, 4);
|
||||||
|
|
||||||
mv.visitVarInsn(ILOAD, 2);
|
mv.visitVarInsn(ILOAD, 2);
|
||||||
Label[] labels = new Label[methods.size()];
|
Label[] labels = new Label[n];
|
||||||
for (int i = 0, n = labels.length; i < n; i++)
|
for (int i = 0; i < n; i++)
|
||||||
labels[i] = new Label();
|
labels[i] = new Label();
|
||||||
Label defaultLabel = new Label();
|
Label defaultLabel = new Label();
|
||||||
mv.visitTableSwitchInsn(0, labels.length - 1, defaultLabel, labels);
|
mv.visitTableSwitchInsn(0, labels.length - 1, defaultLabel, labels);
|
||||||
|
|
||||||
StringBuilder buffer = new StringBuilder(128);
|
StringBuilder buffer = new StringBuilder(128);
|
||||||
for (int i = 0, n = labels.length; i < n; i++) {
|
for (int i = 0; i < n; i++) {
|
||||||
mv.visitLabel(labels[i]);
|
mv.visitLabel(labels[i]);
|
||||||
if (i == 0)
|
if (i == 0)
|
||||||
mv.visitFrame(Opcodes.F_APPEND, 1, new Object[] {classNameInternal}, 0, null);
|
mv.visitFrame(Opcodes.F_APPEND, 1, new Object[] {classNameInternal}, 0, null);
|
||||||
@ -135,8 +144,9 @@ public abstract class MethodAccess {
|
|||||||
buffer.setLength(0);
|
buffer.setLength(0);
|
||||||
buffer.append('(');
|
buffer.append('(');
|
||||||
|
|
||||||
Method method = methods.get(i);
|
String methodName = methodNames[i];
|
||||||
Class[] paramTypes = method.getParameterTypes();
|
Class[] paramTypes = parameterTypes[i];
|
||||||
|
Class returnType = returnTypes[i];
|
||||||
for (int paramIndex = 0; paramIndex < paramTypes.length; paramIndex++) {
|
for (int paramIndex = 0; paramIndex < paramTypes.length; paramIndex++) {
|
||||||
mv.visitVarInsn(ALOAD, 3);
|
mv.visitVarInsn(ALOAD, 3);
|
||||||
mv.visitIntInsn(BIPUSH, paramIndex);
|
mv.visitIntInsn(BIPUSH, paramIndex);
|
||||||
@ -186,10 +196,10 @@ public abstract class MethodAccess {
|
|||||||
}
|
}
|
||||||
|
|
||||||
buffer.append(')');
|
buffer.append(')');
|
||||||
buffer.append(Type.getDescriptor(method.getReturnType()));
|
buffer.append(Type.getDescriptor(returnType));
|
||||||
mv.visitMethodInsn(isInterface ? INVOKEINTERFACE : INVOKEVIRTUAL, classNameInternal, method.getName(), buffer.toString());
|
mv.visitMethodInsn(isInterface ? INVOKEINTERFACE : INVOKEVIRTUAL, classNameInternal, methodName, buffer.toString());
|
||||||
|
|
||||||
switch (Type.getType(method.getReturnType()).getSort()) {
|
switch (Type.getType(returnType).getSort()) {
|
||||||
case Type.VOID:
|
case Type.VOID:
|
||||||
mv.visitInsn(ACONST_NULL);
|
mv.visitInsn(ACONST_NULL);
|
||||||
break;
|
break;
|
||||||
@ -248,6 +258,7 @@ public abstract class MethodAccess {
|
|||||||
MethodAccess access = (MethodAccess)accessClass.newInstance();
|
MethodAccess access = (MethodAccess)accessClass.newInstance();
|
||||||
access.methodNames = methodNames;
|
access.methodNames = methodNames;
|
||||||
access.parameterTypes = parameterTypes;
|
access.parameterTypes = parameterTypes;
|
||||||
|
access.returnTypes = returnTypes;
|
||||||
return access;
|
return access;
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
throw new RuntimeException("Error constructing method access class: " + accessClassName, ex);
|
throw new RuntimeException("Error constructing method access class: " + accessClassName, ex);
|
||||||
|
Loading…
Reference in New Issue
Block a user