在应用中,我们使用java反射的场景也不是很多,但偶尔也会用用。框架用反射用的比较多,例如mybatis、spring等。
在JDK7中,新增了java.lang.invoke.MethodHandle,称之为“现代化反射”。
其实反射和java.lang.invoke.MethodHandle都是间接调用方法的途径,但java.lang.invoke.MethodHandle比反射更简洁,用反射功能会写一大堆冗余代码。
官方api给出的解释:
A method handle is a typed, directly executable reference to an underlying method, constructor, field, or similar low-level operation, with optional transformations of arguments or return values.
其实看到这个,我想起了大学时代学的VC(6)中MFC框架中充满者各种Handle,其实都差不多,类似指针的东西,不过比指针更安全的指向一个东西。
下面例子,我们利用MethodHandle调用toString()方法:
public class MethodHandle {
@Override
public String toString() {
return super.toString() + "==MethodHandle";
}
public static void main(String[] args) throws Throwable {
MethodHandle handle = new MethodHandle();
MethodType methodType = MethodType.methodType(String.class);
Lookup lookup = MethodHandles.lookup();
java.lang.invoke.MethodHandle methodHandle = lookup.findVirtual(MethodHandle.class, "toString", methodType);
String toString = (String) methodHandle.invokeExact(handle);
System.out.println(toString);// com.doctor.java7.MethodHandle@355da254==MethodHandle
// or like this:
java.lang.invoke.MethodHandle methodHandle2 = methodHandle.bindTo(handle);
String toString2 = (String) methodHandle2.invokeWithArguments();
System.out.println(toString2);// com.doctor.java7.MethodHandle@355da254==MethodHandle
// 得到当前Class的不同表示方法,最后一个最好。一般我们在静态上下文用SLF4J得到logger用。
System.out.println(MethodHandle.class);
System.out.println(handle.getClass());
System.out.println(MethodHandles.lookup().lookupClass()); // like getClass()
}
}
看,其实
MethodType
就代表一个函数签名,符合这个函数签名的都是一类函数。我们可以用
Lookup
来查找特定类下的这个函数。
最后几行代码,是我们经常用日志如SLF4J的时候,各种不同方法获取当前类。