将 java.lang.reflect.Method 转换为函数式接口

Xob*_*tun 4 java reflection

很难找到有关该主题的任何线索。我能找到的只是关于将一​​个函数式接口转换为另一个函数式接口的问题,以及一些关于 Java 中类型转换的文章。不是我要找的。

这个问题是关于转换的lambda ? Method,我想要相反的,转换Method为任何功能接口,例如,转换为Consumer.

我发现的方法是围绕该Method#invoke方法创建一个 lambda 适配器:

    public void registerCallbacks(final Object annotated) {
        Class clazz = annotated.getClass();
        for (Method method : clazz.getDeclaredMethods()) {
            if (method.isAnnotationPresent(Callback.class)) {
                Callback registration = method.getAnnotation(Callback.class);

                List<String> warnings = new ArrayList<>(3);

                if (!Modifier.isPublic(method.getModifiers()))
                    warnings.add(String.format("Method %s must be public", method));
                if (method.getParameterCount() != 1)
                    warnings.add(String.format("Method %s must consume only one argument", method));
                if (method.getParameterCount() == 1 && !method.getParameterTypes()[0].equals(Integer.class))
                    warnings.add(String.format("Method %s must consume %s", method, Integer.class));

                if (!warnings.isEmpty()) {
                    warnings.forEach(log::warn);
                    continue;
                }

                CALLBACKS_MAPPER.registerCallback((param) -> {
                    try {
                        method.invoke(annotated, param);
                    } catch (IllegalAccessException | InvocationTargetException e) {
                        // Should not happen due to checks before.
                        log.warn(String.format("Could not invoke %s on %s with %s", method, annotated, param), e);
                    }
                });
                log.info("Registered {} as a callback", method);
            }
        }
    }

Run Code Online (Sandbox Code Playgroud)

但是我想避免写作

    CALLBACKS_MAPPER.registerCallback((param) -> {
        try {
            method.invoke(annotated, param);
        } catch (IllegalAccessException | InvocationTargetException e) {
            // Should not happen due to checks before.
            log.warn(String.format("Could not invoke %s on %s with %s", method, annotated, param), e);
        }
    });
Run Code Online (Sandbox Code Playgroud)

赞成更简单的东西,比如

    CALLBACKS_MAPPER.registerCallback(SomeApacheLib.methodToFunction(annotated, method));
Run Code Online (Sandbox Code Playgroud)

? 那么,有没有办法将旧的 Java 1.1 反射库映射到较新的 Java 8 功能接口,或者是我愚蠢而上述 lambda 解决方案就这样很好?

kaq*_*qao 6

如果您满足于在幕后使用反射,只是不喜欢周围的try/ ,您可以创建一个简单的实用程序函数,例如:catchinvoke

public static <T> Consumer<T> toConsumer(Object annotated, Method m) {
    return param -> {
        try {
            m.invoke(annotated, param);
        } catch (IllegalAccessException | InvocationTargetException e) {
            throw new RuntimeException(e);
        }
    };
}
Run Code Online (Sandbox Code Playgroud)

这为您提供了您想要的语法:

CALLBACKS_MAPPER.registerCallback(toConsumer(annotated, method));
Run Code Online (Sandbox Code Playgroud)

但是如果你想完全避免反射,你可以使用LambdaMetafactory创建一个Consumer

static Consumer<String> toConsumer(MethodHandles.Lookup lookup, Object annotated, Method method) throws Throwable {
    MethodType consumeString = MethodType.methodType(void.class, String.class);
    MethodHandle handle = lookup.unreflect(method);
    final CallSite site = LambdaMetafactory.metafactory(lookup, "accept",
            MethodType.methodType(Consumer.class, annotated.getClass()),
            consumeString.changeParameterType(0, Object.class),
            handle,
            consumeString);
    return (Consumer<String>) site.getTarget().invoke(annotated);
}
Run Code Online (Sandbox Code Playgroud)

更改String为您的回调预期接受的任何内容。进而:

CALLBACKS_MAPPER.registerCallback(toConsumer(MethodHandles.lookup(), annotated, method));
Run Code Online (Sandbox Code Playgroud)

当然,这里唯一合适的解决方案是重构代码以使用已知的回调接口,您通常可以在该接口上调用已定义的方法,而不是传递Methods。