获取Java方法参数的名称

简介:

最近有个需求,需要获取java方法参数的名称,网上查了下,然后自己也实践了下,总结出3点:
1.能不能获取方法参数的名称取决于class文件里是否含有LocalVariableTable。
2.javac编译生成的class文件不含有LocalVariableTable,但是eclipse编译生成的class文件却含有LocalVariableTable。
3.有2种方式可以获得,用Asm和Javassist,Asm的性能略高。
贴代码:

maven依赖:

		<dependency>
			<groupId>org.javassist</groupId>
			<artifactId>javassist</artifactId>
			<version>3.18.0-GA</version>
		</dependency>
		<dependency>
			<groupId>org.ow2.asm</groupId>
			<artifactId>asm</artifactId>
			<version>4.1</version>
		</dependency>

User类:

/**
 * 
 * @author <a href="mailto:yankai913@gmail.com">yankai</a>
 * @date 2013-7-17
 */
public class User {
	
	String name;
	String address;
	int age;
	
	public User(String name, String address, int age) {
		this.name = name;
		this.address = address;
		this.age = age;
	}
	
	public static User newInstance(String name, String address, int age) {
		return new User(name, address, age);
	}
	
	public User copy(String name, String address, int age) {
		return new User(name, address, age);
	}
}

asm实现:

import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.Label;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;
/**
 * @author <a href="mailto:yankai913@gmail.com">yankai</a>
 * @date 2013-7-17
 */
public class AsmTool {
	
	private static boolean sameType(Type[] types, Class<?>[] clazzes) {
		if (types.length != clazzes.length) {
			return false;
		}
		for (int i = 0; i < types.length; i++) {
			if (!Type.getType(clazzes[i]).equals(types[i])) {
				return false;
			}
		}
		return true;
	}
	
	public static String[] getMethodParamNames(final Method m) throws Exception {
		final String[] paramNames = new String[m.getParameterTypes().length];
		String methodClass = m.getDeclaringClass().getName();
		ClassReader cr = new ClassReader(methodClass);
		cr.accept(new ClassVisitor(Opcodes.ASM4) {
			@Override
			public MethodVisitor visitMethod(final int access,
					final String name, final String desc,
					final String signature, final String[] exceptions) {
				Type[] args = Type.getArgumentTypes(desc);
				if (!name.equals(m.getName())
						|| !sameType(args, m.getParameterTypes())) {
					return null;
				}
				return new MethodVisitor(Opcodes.ASM4) {
					@Override
					public void visitLocalVariable(String name, String desc,
							String signature, Label start, Label end, int index) {
						int i = index - 1;
						if (Modifier.isStatic(m.getModifiers())) {
							i = index;
						}
						if (i >= 0 && i < paramNames.length) {
							paramNames[i] = name;
						}
					}

				};
			}
		}, 0);
		return paramNames;
	}
}

asm测试:

import java.lang.reflect.Method;
import java.util.Arrays;
/**
 * 
 * @author <a href="mailto:yankai913@gmail.com">yankai</a>
 * @date 2013-7-17
 */
public class TestAsm {
	public static void main(String[] args) throws Exception {
		long start = System.currentTimeMillis();
		final Method method = 
				User.class.getMethod("newInstance", new Class<?>[]{String.class, String.class, int.class});
		String[] s = AsmTool.getMethodParamNames(method);
		System.out.println(Arrays.toString(s));
		System.out.println(System.currentTimeMillis() - start);//40
	}
}

javassist实现:

import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;
import javassist.Modifier;
import javassist.bytecode.CodeAttribute;
import javassist.bytecode.LocalVariableAttribute;
import javassist.bytecode.MethodInfo;
/**
 * 
 * @author <a href="mailto:yankai913@gmail.com">yankai</a>
 * @date 2013-7-17
 */
public class JavassistTool {
	
	// <ClassLoader, ClassPool>
	static final Map<ClassLoader, ClassPool> pool2map = 
			new ConcurrentHashMap<ClassLoader, ClassPool>(); 
	
	public static String[] getMethodParamNames(Method method)
			throws Exception {
		Class<?> clazz = method.getDeclaringClass();
		String methodName = method.getName();
		ClassPool pool = ClassPool.getDefault();
		CtClass cc = null;
		try {
			cc = pool.get(clazz.getName());
		} catch (Exception e) {
			if (e instanceof NotFoundException) {
				pool = pool2map.get(clazz.getClassLoader());
				if (pool == null) {
					pool = new ClassPool(true);
					pool.appendClassPath(new LoaderClassPath(clazz.getClassLoader()));
					pool2map.put(clazz.getClassLoader(), pool);
				}
			}
		}
		cc = pool.get(clazz.getName());
		CtMethod cm = cc.getDeclaredMethod(methodName);
		MethodInfo methodInfo = cm.getMethodInfo();
		CodeAttribute codeAttribute = methodInfo.getCodeAttribute();
		LocalVariableAttribute attr = 
				(LocalVariableAttribute) codeAttribute.getAttribute(LocalVariableAttribute.tag);
				
		if (attr == null) {
			throw new IllegalStateException("LocalVariableAttribute is null");
		}
		String[] paramNames = new String[cm.getParameterTypes().length];
		int offset = Modifier.isStatic(cm.getModifiers()) ? 0 : 1;
		for (int i = 0; i < paramNames.length; i++) {
			paramNames[i] = attr.variableName(i + offset);
		}
		return paramNames;
	}
}

javassist测试:

import java.util.Arrays;
/**
 * 
 * @author <a href="mailto:yankai913@gmail.com">yankai</a>
 * @date 2013-7-17
 */
public class TestJavassist {

	public static void main(String[] args) throws Exception {
		long start = System.currentTimeMillis();
		String[] paramNames = JavassistTool.getMethodParamNames(User.class, "newInstance");
		System.out.println(Arrays.toString(paramNames));
		System.out.println(System.currentTimeMillis() - start);//90
	}
}

相关文章
|
12天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
53 4
|
23天前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
43 17
|
16天前
|
Java 测试技术 Maven
Java一分钟之-PowerMock:静态方法与私有方法测试
通过本文的详细介绍,您可以使用PowerMock轻松地测试Java代码中的静态方法和私有方法。PowerMock通过扩展Mockito,提供了强大的功能,帮助开发者在复杂的测试场景中保持高效和准确的单元测试。希望本文对您的Java单元测试有所帮助。
32 2
|
24天前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
19 3
|
24天前
|
Java 开发者
在Java多线程编程中,选择合适的线程创建方法至关重要
【10月更文挑战第20天】在Java多线程编程中,选择合适的线程创建方法至关重要。本文通过案例分析,探讨了继承Thread类和实现Runnable接口两种方法的优缺点及适用场景,帮助开发者做出明智的选择。
16 2
|
24天前
|
安全 Java
Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧
【10月更文挑战第20天】Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧,包括避免在循环外调用wait()、优先使用notifyAll()、确保线程安全及处理InterruptedException等,帮助读者更好地掌握这些方法的应用。
16 1
|
24天前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
28 1
|
24天前
|
Java
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅。它们用于线程间通信,使线程能够协作完成任务。通过这些方法,生产者和消费者线程可以高效地管理共享资源,确保程序的有序运行。正确使用这些方法需要遵循同步规则,避免虚假唤醒等问题。示例代码展示了如何在生产者-消费者模型中使用`wait()`和`notify()`。
24 1
|
24天前
|
安全 Java 开发者
Java多线程中的`wait()`、`notify()`和`notifyAll()`方法,探讨了它们在实现线程间通信和同步中的关键作用
本文深入解析了Java多线程中的`wait()`、`notify()`和`notifyAll()`方法,探讨了它们在实现线程间通信和同步中的关键作用。通过示例代码展示了如何正确使用这些方法,并分享了最佳实践,帮助开发者避免常见陷阱,提高多线程程序的稳定性和效率。
34 1
|
18天前
|
Java Spring
JAVA获取重定向地址URL的两种方法
【10月更文挑战第17天】本文介绍了两种在Java中获取HTTP响应头中的Location字段的方法:一种是使用HttpURLConnection,另一种是使用Spring的RestTemplate。通过设置连接超时和禁用自动重定向,确保请求按预期执行。此外,还提供了一个自定义的`NoRedirectSimpleClientHttpRequestFactory`类,用于禁用RestTemplate的自动重定向功能。