当前位置: 首页 > 面试题库 >

如何在Java8中为void(不是Void)方法指定函数类型?

朱毅
2023-03-14
问题内容

我正在玩Java 8,以了解如何发挥一流公民的作用。我有以下片段:

package test;

import java.util.*;
import java.util.function.*;

public class Test {

    public static void myForEach(List<Integer> list, Function<Integer, Void> myFunction) {
      list.forEach(functionToBlock(myFunction));
    }

    public static void displayInt(Integer i) {
      System.out.println(i);
    }


    public static void main(String[] args) {
      List<Integer> theList = new ArrayList<>();
      theList.add(1);
      theList.add(2);
      theList.add(3);
      theList.add(4);
      theList.add(5);
      theList.add(6);
      myForEach(theList, Test::displayInt);
    }
}

我想做的是使用方法引用displayInt将方法传递给方法myForEach。要编译会产生以下错误:

src/test/Test.java:9: error: cannot find symbol
      list.forEach(functionToBlock(myFunction));
                   ^
  symbol:   method functionToBlock(Function<Integer,Void>)
  location: class Test
src/test/Test.java:25: error: method myForEach in class Test cannot be applied to given ty
pes;
      myForEach(theList, Test::displayInt);
      ^
  required: List<Integer>,Function<Integer,Void>
  found: List<Integer>,Test::displayInt
  reason: argument mismatch; bad return type in method reference
      void cannot be converted to Void

编译器抱怨void cannot be converted to Void。我不知道如何myForEach在代码的签名中指定函数接口的类型。我知道我可以简单地改变的返回类型displayIntVoid,然后返回null。但是,在某些情况下,无法更改我想传递到其他地方的方法。有没有简单的方法可以重复使用displayInt


问题答案:

您正在尝试使用错误的接口类型。在这种情况下,函数类型是不合适的,因为它会接收参数并具有返回值。相反,您应该使用Consumer(以前称为Block)

函数类型声明为

interface Function<T,R> {
  R apply(T t);
}

但是,消费者类型与您要查找的类型兼容:

interface Consumer<T> {
   void accept(T t);
}

这样,Consumer与接收T且不返回任何值(无效)的方法兼容。这就是您想要的。

例如,如果我想显示列表中的所有元素,我可以简单地使用lambda表达式为其创建使用者:

List<String> allJedi = asList("Luke","Obiwan","Quigon");
allJedi.forEach( jedi -> System.out.println(jedi) );

您可以在上面看到,在这种情况下,lambda表达式接收一个参数并且没有返回值。

现在,如果我想使用方法引用而不是lambda表达式来创建这种类型的消耗,那么我需要一个可以接收String并返回void的方法,对吗?

我可以使用不同类型的方法引用的,但在这种情况下,让一个对象的方法引用的占据优势,通过使用println该方法System.out的对象,像这样:

Consumer<String> block = System.out::println

或者我可以简单地做

allJedi.forEach(System.out::println);

println方法是适当的,因为它接收一个值并且具有返回类型void,就像acceptConsumer中的方法一样。

因此,在您的代码中,您需要将方法签名更改为:

public static void myForEach(List<Integer> list, Consumer<Integer> myBlock) {
   list.forEach(myBlock);
}

然后,您应该能够通过以下方式使用静态方法引用创建使用者:

myForEach(theList, Test::displayInt);

最终,您甚至可以myForEach完全摆脱方法,只需执行以下操作:

theList.forEach(Test::displayInt);

关于作为头等公民的职能

所有人都说过,事实是Java
8将不具有作为一等公民的功能,因为不会将结构函数类型添加到该语言中。Java将简单地提供一种替代方法,以通过lambda表达式和方法引用来创建功能接口的实现。最终,lambda表达式和方法引用将绑定到对象引用,因此,我们所拥有的只是作为一等公民的对象。重要的是功能在这里,因为我们可以将对象作为参数传递,将它们绑定到变量引用,然后将它们作为其他方法的值返回,那么它们的作用差不多。



 类似资料:
  • 问题内容: 在上面的代码示例中,我尝试通过使用参数表单击按钮来调用方法( fetchData )。 错误是 类型’void’不能分配给类型’((event:MouseEvent)= > void)| 未定义”。 问题答案: 在您的代码中,您正在 调用 该函数。函数返回。不能期望哪个功能。 创建一个调用fetchData的新函数,例如。 更多 这是TypeScript防止的非常常见的错误

  • 我用java编写这个简单的类只是为了测试它的一些特性。 0 那么为什么当我在构造函数方法声明中使用时,类的静态字段不再改变了呢?

  • 所以我明白了如何使用一个递归方法,它有一些其他的返回类型,而不是作废。通常我会在同一个方法中再次调用相同的方法(在递归的情况下),同时在调用中递减或增加一些值以达到基本情况。然后在某个时候达到基本情况,问题就解决了,所以它开始返回每个调用的值。沿着这些路线。 但是 如果该方法的返回类型为void,那么您不能调用该方法,因为它不会/不能返回任何内容,该怎么办?我试着倒着写一个句子,我已经用for循环

  • void与空类型指针 空类型的指针可以存储任何类型变量(普通变量,一级指针变量,二级指针变量,三级指针变量,多级指针变量等等...)的地址,因为编译器决定了地址存储尺寸 空类型的指针既不可以间接取值,也不可以间接赋值(也就是只能存储内存地址,而不能根据内存地址进行间接访问操作) 将空类型的指针转化为具体指针类型,然后就达到了既明确了指针的解析步长,也明确了解析方式 ///01.void.c #in

  • 问题内容: 我正在为apache HttpClient包实现一个,如下所示: 但我希望函数不返回任何值,即。这可能吗?由于不是有效的Java类型,因此以下内容无法编译: 我想可以替换为以返回对象,但这并不是我真正想要的。 问 :是否有可能安排在这样一种方式,我可以返回此回调局面从? 问题答案: 泛型仅处理对象类。泛型不支持void和基本类型,您不能将它们用作参数化类型。您必须改用Void。 你能说