Lambda函数如何成为Comparator的compare()方法 - java

我已经看到在Java 8中,可以这样定义一个比较器:

Comparator c = (Computer c1, Computer c2) -> c1.getAge().compareTo(c2.getAge());

等效于:

Comparator d = new Comparator<Computer> () {
    @Override
    public int compare(Computer c1, Computer c2){
        return c1.getAge().compareTo(c2.getAge());
    }
};

我想了解这是如何工作的。在第二个示例中,它非常简单:使用方法Comparator创建compare对象,该方法使用compareToage属性中的Computer方法执行比较。当我们这样做时,我们简单地调用此方法:

Computer comp1 = new Computer(10);
Computer comp2 = new Computer(11);
d.compare(comp1, comp2); // -1

但是,在使用lambda的第一个示例中发生了什么?在我看来,我们正在将Comparator设置为等于执行比较的方法。但这不能,因为Comparator对象是具有compare方法的对象。我了解到lambda可以与功能接口(只有一种方法的接口)一起使用。但是Comparator不是功能性接口(除了compare,它还有许多其他方法!)。那么Java解释器如何知道它是我们正在实现的compare方法?

参考方案

说明
Comparator功能接口(仅需要一种方法)。因此,您可以使用lambda表达式创建它的实例。

它的行为与创建实例的其他方法非常相似,例如扩展的常规类或匿名类。

lambda指功能接口需要的一种方法。因为只有一种方法,所以它不是模棱两可的。 Lambda命名输入参数,然后给出该方法的实现(它提供一个主体)。

总览

您可以使用以下选项来创建接口或抽象类的实例:

  • 创建一个扩展并使用新
  • 的类

  • 使用匿名类
  • 假设我们有一个仅提供方法的接口(此后称为功能接口),我们还具有以下两个选项来创建它的实例:

  • 使用lambda表达式
  • 使用方法参考
  • 例如,我们要使用以下接口创建一个乘法实例:

    @FunctionalInterface
    public interface Operation {
        int op(int a, int b);
    }
    
  • 创建一个扩展并使用new的类:
    public class Multiplicator implements Operation {
        @Override
        public int op(int a, int b) {
            return a * b;
        }
    }
    
    // Usage
    Operation operation = new Multiplicator();
    System.out.println(operation.op(5, 2)); // 10
    
  • 使用匿名类:
    Operation operation = new Operation() {
        @Override
        public int op(int a, int b) {
            return a * b;
        }
    };
    
    // Usage
    System.out.println(operation.op(5, 2)); // 10
    
  • 使用lambda表达式:
    Operation operation = (a, b) -> a * b;
    System.out.println(operation.op(5, 2)); // 10
    
  • 使用方法参考:
    // Somewhere else in our project, in the `MathUtil` class
    public static int multiply(int a, int b) {
        return a * b;
    }
    
    // Usage
    Operation operation = MathUtil::multiply;
    System.out.println(operation.op(5, 2)); // 10
    
  • Java-将int更改为ascii - java

    java有没有办法将int转换为ascii符号? 参考方案 您是否要将int转换为char?:int yourInt = 33; char ch = (char) yourInt; System.out.println(yourInt); System.out.println(ch); // Output: // 33 // ! 还是要将int转换为Stri…

    Java:正则表达式模式匹配器是否有大小限制? - java

    我的模式类似于OR:“word1 | word2 | word3”我大约有800个字。可能有问题吗? 参考方案 您仅受记忆和理智的限制。 :)

    Java:线程池如何将线程映射到可运行对象 - java

    试图绕过Java并发问题,并且很难理解线程池,线程以及它们正在执行的可运行“任务”之间的关系。如果我创建一个有10个线程的线程池,那么我是否必须将相同的任务传递给池中的每个线程,或者池化的线程实际上只是与任务无关的“工人无人机”可用于执行任何任务?无论哪种方式,Executor / ExecutorService如何将正确的任务分配给正确的线程? 参考方案 …

    JAVA:字节码和二进制有什么区别? - java

    java字节代码(已编译的语言,也称为目标代码)与机器代码(当前计算机的本机代码)之间有什么区别?我读过一些书,他们将字节码称为二进制指令,但我不知道为什么。 参考方案 字节码是独立于平台的,在Windows中运行的编译器编译的字节码仍将在linux / unix / mac中运行。机器代码是特定于平台的,如果在Windows x86中编译,则它将仅在Win…

    java:继承 - java

    有哪些替代继承的方法? java大神给出的解决方案 有效的Java:偏重于继承而不是继承。 (这实际上也来自“四人帮”)。他提出的理由是,如果扩展类未明确设计为继承,则继承会引起很多不正常的副作用。例如,对super.someMethod()的任何调用都可以引导您通过未知代码的意外路径。取而代之的是,持有对本来应该扩展的类的引用,然后委托给它。这是与Eric…