- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
假设你有一个方法
void m(String s) {}
通过使用新的方法引用,您可以创建一个 Consumer
从中
Consumer<String> c = Class::m;
如果您现在通过反射查看创建的 Consumer 的类,您将看到它唯一声明的方法是
void accept(Object)
如果通过创建匿名内部类以旧方式创建消费者
Consumer<String> c = new Consumer<String>() {
public void accept(String s){}
}
会有桥接方法void accept(Object)
以及void accept(String)
.
现在假设我们必须传递这个消费者,从而失去它的通用类型。如果你实现了 Consumer<String>
而不是 Lambda 表达式,您可以通过使用反射来访问它的方法来找回它。由于通过方法引用创建的消费者只有通用方法,因此无法正常工作。有没有办法获取方法引用创建的消费者的参数类型?
最佳答案
调用原始消费者(没有通用类型)总是会导致警告。你根本不应该这样做,因为你不知道类型。原始类型没有类型。 Java 8 并没有改变这一点。
Consumer 接口(interface)具有通用类型,但您在运行时不知道。即使不是 lambda。有些类(class)可能实际上有这些信息,但这对你没有帮助。如果您确实需要知道类型,那么只需创建这样一个接口(interface)即可。
@FunctionalInterface
public static interface StringConsumer {
void consume(String s);
}
public static void main(String[] args) throws Throwable {
StringConsumer sc = System.out::println;
sc.consume("Hello");
Method m = StringConsumer.class.getMethods()[0]; // only one: "consume"
m.invoke(sc, m.getParameters()[0].getType().getSimpleName());
}
下面是一些示例代码和输出,以了解有哪些方法以及它们如何可以和不可以被调用:
package com.example.foo;
import static java.util.Arrays.asList;
import java.lang.reflect.Method;
import java.util.function.Consumer;
public class SomeClass {
@FunctionalInterface
static interface FI {
public void m(String s);
}
static final class MyRegularClass {
@SuppressWarnings("static-method")
public void m(String s) {
System.out.println("MyRegularClass: " + s);
};
}
static final class MyGenericClass<T> {
public void m(T s) {
System.out.println("MyGenericClass: " + s);
};
}
public static void main(String[] args) throws Exception {
Consumer<String> c1 = (s) -> {
System.out.println("Lambda: " + s);
};
Consumer<String> c2 = new Consumer<String>() {
public void accept(String s) {
System.out.println("Anonym: " + s);
}
};
Consumer<String> c3 = new MyRegularClass()::m;
Consumer<String> c4 = new MyGenericClass<String>()::m;
for (Consumer<String> c : asList(c1, c2, c3, c4)) {
c.accept("regular invocation of accept(String)");
for (Method m : c.getClass().getDeclaredMethods()) {
String n = m.getName() + "(" + m.getParameters()[0].getType().getSimpleName() + ")";
try {
m.invoke(c, n);
} catch (Exception e) {
System.out.println("Did not accept String: " + n + " => " + e);
}
try {
m.setAccessible(true);
m.invoke(c, new StringBuilder("StringBuilder of ").append(n));
} catch (Exception e) {
System.out.println("Did not accept StringBuilder: " + n + " => " + e);
}
}
System.out.println("-----");
}
}
}
/* ==========================================
Output:
Lambda: regular invocation of accept(String)
Lambda: accept(Object)
Did not accept StringBuilder: accept(Object) => java.lang.reflect.InvocationTargetException
-----
Anonym: regular invocation of accept(String)
Anonym: accept(String)
Did not accept StringBuilder: accept(String) => java.lang.IllegalArgumentException: argument type mismatch
Anonym: accept(Object)
Did not accept StringBuilder: accept(Object) => java.lang.reflect.InvocationTargetException
-----
MyRegularClass: regular invocation of accept(String)
MyRegularClass: accept(Object)
Did not accept StringBuilder: accept(Object) => java.lang.reflect.InvocationTargetException
Did not accept String: get$Lambda(MyRegularClass) => java.lang.IllegalAccessException: Class com.example.foo.SomeClass can not access a member of class com.example.foo.SomeClass$$Lambda$2/1175962212 with modifiers "private static"
Did not accept StringBuilder: get$Lambda(MyRegularClass) => java.lang.IllegalArgumentException: argument type mismatch
-----
MyGenericClass: regular invocation of accept(String)
MyGenericClass: accept(Object)
Did not accept StringBuilder: accept(Object) => java.lang.reflect.InvocationTargetException
Did not accept String: get$Lambda(MyGenericClass) => java.lang.IllegalAccessException: Class com.example.foo.SomeClass can not access a member of class com.example.foo.SomeClass$$Lambda$3/617901222 with modifiers "private static"
Did not accept StringBuilder: get$Lambda(MyGenericClass) => java.lang.IllegalArgumentException: argument type mismatch
-----
*/
关于java - 方法引用的通用功能接口(interface)的方法签名,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22254949/
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: template pass by value or const reference or…? 以下对于将函数
我用相同的参数列表重载了一个运算符两次。但返回类型不同: T& operator()(par_list){blablabla} const T& operator()(par_list){bla
假设我有实现接口(interface) I 的 Activity A。我的 ViewModel 类 (VM) 持有对实现接口(interface) I 的对象的引用: class A extends
PHP 如何解释 &$this ?为什么允许? 我遇到了以下问题,这看起来像是 PHP 7.1 和 7.2 中的错误。它与 &$this 引用和跨命名空间调用以及 call_user_func_arr
谁能解释一下下面“&”的作用: class TEST { } $abc =& new TEST(); 我知道这是引用。但是有人可以说明我为什么以及什么时候需要这样的东西吗?或者给我指向一个对此有很好解
引用变量是一个别名,也就是说,它是某个已存在变量的另一个名字。一旦把引用初始化为某个变量,就可以使用该引用名称或变量名称来指向变量。 C++ 引用 vs 指针 引用很容易与指针混淆,它们之间有三
目录 引言 背景 结论 引言 我选择写C++中的引用是因为我感觉大多数人误解了引用。而我之所以有这个感受是因为我主持过很多C++的面试,并且我很少
Perl 中的引用是指一个标量类型可以指向变量、数组、哈希表(也叫关联数组)甚至函数,可以应用在程序的任何地方 创建引用 定义变量的时候,在变量名前面加个 \,就得到了这个变量的一个引用 $sc
我编写了一个将从主脚本加载的 Perl 模块。该模块使用在主脚本中定义的子程序(我不是维护者)。 对于主脚本中的一个子例程,需要扩展,但我不想修补主脚本。相反,我想覆盖我的模块中的函数并保存对原始子例
我花了几个小时试图掌握 F# Quotations,但我遇到了一些障碍。我的要求是从可区分的联合类型中取出简单的函数(只是整数、+、-、/、*)并生成一个表达式树,最终将用于生成 C 代码。我知道使用
很多时候,问题(尤其是那些标记为 regex 的问题)询问验证密码的方法。似乎用户通常会寻求密码验证方法,包括确保密码包含特定字符、匹配特定模式和/或遵守最少字符数。这篇文章旨在帮助用户找到合适的密码
我想通过 MIN 函数内的地址(例如,C800)引用包含文本的最后一个单元格。你能帮忙吗? Sub Set_Formula() ' ----------------------------- Dim
使用常规的 for 循环,我可以做类似的事情: for (let i = 0; i < objects.length; i++) { delete objects[i]; } 常规的 for-
在 Cucumber 中,您定义了定义 BDD 语法的步骤;例如,您的测试可能有: When I navigate to step 3 然后你可以定义一个步骤: When /^I navigate t
这是什么UnaryExpression的目的,以及应该怎样使用? 最佳答案 它需要一个 Expression对象并用另一个 Expression 包裹它.例如,如果您有一个用于 lambda 的表达式
给出以下内容 $("#identifier div:first, #idetifier2").fadeOut(300,function() { // I need to reference jus
我不知道我要找的东西的正确术语,但我要找的是一个完整的引用,可以放在双引号之间的语句,比如 *, node()、@* 以及所有列出的 here加上任何其他存在的。 我链接到的答案提供了一些细节,但还
This question's answers are a community effort。编辑现有答案以改善此职位。它当前不接受新的答案或互动。 这是什么? 这是常见问答的集合。这也是一个社区Wi
Closed. This question does not meet Stack Overflow guidelines。它当前不接受答案。 想改善这个问题吗?更新问题,以便将其作为on-topic
考虑下一个代码: fn get_ref(slice: &'a Vec, f: fn(&'a Vec) -> R) -> R where R: 'a, { f(slice) } fn m
我是一名优秀的程序员,十分优秀!