2

Java 8 中特定方法的方法参考可以通过Class::Method. 但是如何获取一个类的所有方法的方法引用呢?

所有所需的方法都具有不同的方法名称,但具有相同的类型签名。此外,这些方法的名称事先是未知的。

例子:

class Test {
    public static double op0(double a) { ... }
    public static double op1(double a) { ... }
    public static double op2(double a) { ... }
    public static double op3(double a) { ... }
    public static double op4(double a) { ... }
}

对已知方法的方法引用op0可以得到如下:

DoubleFunction<Double> f = Test::op0;

但是,如何获取类中所有方法的方法引用呢?

4

1 回答 1

3

没有反射就没有解决方案,因为现有方法的动态发现一种反射操作。但是,一旦发现方法并创建了方法引用实例(或它的动态等价物),代码的实际调用就会在没有反射的情况下运行:

class Test {
    public static double op0(double a) { ... }
    public static double op1(double a) { ... }
    public static double op2(double a) { ... }
    public static double op3(double a) { ... }
    public static double op4(double a) { ... }

    static final Map<String, DoubleUnaryOperator> OPS;
    static {
        HashMap<String, DoubleUnaryOperator> map=new HashMap<>();
        MethodType type=MethodType.methodType(double.class, double.class);
        MethodType inT=MethodType.methodType(DoubleUnaryOperator.class);
        MethodHandles.Lookup l=MethodHandles.lookup();
        for(Method m:Test.class.getDeclaredMethods()) try {
          if(!Modifier.isStatic(m.getModifiers())) continue;
          MethodHandle mh=l.unreflect(m);
          if(!mh.type().equals(type)) continue;
          map.put(m.getName(), (DoubleUnaryOperator)LambdaMetafactory.metafactory(
            l, "applyAsDouble", inT, type, mh, type).getTarget().invokeExact());
        } catch(Throwable ex) {
            throw new ExceptionInInitializerError(ex);
        }
        OPS=Collections.unmodifiableMap(map);
    }
}

初始化类后,您可以使用不使用反射调用特定操作OPS.get(name).applyAsDouble(doubleValue)或使用调用所有操作,例如

OPS.forEach((name,op)-> System.out.println(name+'('+42+") => "+op.applyAsDouble(42)));
于 2015-04-07T19:30:46.550 回答