Java-метод не может быть применен с лямбда-выражением - PullRequest
0 голосов
/ 10 сентября 2018

Я смотрел и читал https://caveofprogramming.com/java/whats-new-in-java-8-lambda-expressions.html, и я следую той же схеме, что и для объекта бегуна, который отлично работает.

Runner runner = new Runner();
runner.run(() -> System.out.println("Print from Lambda expression"));

Затем я пытаюсь создать простой интерфейс и класс, чтобы применить полученные знания. Я просто хочу заменить анонимный класс лямбда-выражением. Насколько я понимаю, лямбда-выражение является более коротким кодом для анонимного класса и улучшает читабельность.

Итак, я попытался запустить другой экземпляр с именем eucalyptus1 и попытаться @Override метод grow(), но в моем сообщении об ошибке IDE было сказано:

grow() в com.smith.Eucalyptus нельзя применить к (lambda expression)

Может ли кто-нибудь указать мне, что я здесь неправильно понимаю

Код ниже:

// a simple interface
interface Plant {
    public void grow();
}

// apply interface to class
class Eucalyptus implements Plant {
    @Override
    public void grow() {
        System.out.println("This is from Eucalyptus");
    }
}

public class Main {
    public static void main(String[] args) {

        // Create an instance of Eucalyptus
        Eucalyptus eucalyptus = new Eucalyptus();
        eucalyptus.grow();

        // Anonymous class Myrtle from Plant interface
        Plant myrtle = new Plant() {
            @Override
            public void grow() {
                System.out.println("This was running from anonymous class from Plant Interface");
            }
        };

        myrtle.grow();

        // Try to create a lambda expression from Plant interface
        // and override grow() method
        // by print ("This was running from Lambda expression")

        // this won't work. why?
        Eucalyptus eucalyptus1 = new Eucalyptus();
        eucalyptus1.grow(() -> System.out.println("This from Lambda expression"));
    }
}

Ответы [ 3 ]

0 голосов
/ 10 сентября 2018

Задача

Метод grow не принимает никаких параметров, поэтому вы получили ошибку компиляции.

Объяснение

Лямбда () -> System.out.println("This from Lambda expression") сама может представлять Plant (не Eucalyptus*) экземпляр:

Plant plant = () -> System.out.println("This from Lambda expression");

Попробуйте создать лямбда-выражение из интерфейса Plant и переопределить grow() метод путем печати "This was running from Lambda expression".

Здесь есть небольшое недоразумение. Лямбда не должна переопределять метод , она может предоставлять метод на основе типа @FunctionalInterface.


* Если бы вы определяли объект Eucalyptus с помощью лямбды, было бы неоднозначно и неясно, какой метод будет представлять лямбда. Поэтому запрещено (даже для абстрактных классов с одним абстрактным методом.)

0 голосов
/ 10 сентября 2018

Ваше использование лямбда-выражения здесь неверно.

Вы используете лямбда-выражения для реализации интерфейса. В этом случае вы предоставляете реализацию Plant, используя лямбда-выражение, а не вызываете метод этого интерфейса с лямбда-выражением:

Вот обычное использование:

Plant eucalyptus1 = () -> System.out.println("This from Lambda expression");
eucalyptus1.grow(); // Prints "This from Lambda expression"

Другими словами, поскольку у вас есть функциональный интерфейс, вы должны избегать создания классов, реализующих его (анонимно или нет).

Так что вам вообще не нужно создавать класс Eucalyptus.

0 голосов
/ 10 сентября 2018

Разница в том, что вы пытаетесь переопределить реализацию Eucalyptus, которая является классом, реализующим интерфейс.

Eucalyptus eucalyptus1 = new Eucalyptus();
eucalyptus1.grow(() -> System.out.println("This from Lambda expression")); 
^__  // you cannot override a method using an instance of a class which is just an implementation of the interface

Все, что вы в конечном итоге делаете, это передаете лямбда-параметр и, конечно, метод без аргументов в своем определении не будет компилироваться, если будет предоставлен с одним из них во время вызова метода.


Вместо этого вы можете сравнить способ реализации лямбды как:

//Anonymous class Myrtle from Plant interface
Plant myrtle = new Plant() {
          @Override
          public void grow() {
               System.out.println("This was running from anonymous class from Plant Interface");
          }
};
myrtle.grow();

может быть представлено как лямбда-представление:

Plant lambdaRep =  () -> System.out.println("This is running via lambda from Plant Interface");
lambdaRep.grow();
...