Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How does method reference casting work?

public class Main {
    interface Capitalizer {
        public String capitalize(String name);
    }

    public String toUpperCase() {
        return "ALLCAPS";
    }

    public static void main(String[] args) {
        Capitalizer c = String::toUpperCase; //This works
        c = Main::toUpperCase; //Compile error
    }
}

Both are instance methods with same signature. Why does one work and the other doesn't?

Signature of String::toUpperCase: String toUpperCase();

like image 207
Kshitiz Sharma Avatar asked Dec 25 '22 19:12

Kshitiz Sharma


2 Answers

There are 3 constructs to reference a method:

  1. object::instanceMethod
  2. Class::staticMethod
  3. Class::instanceMethod

The line:

Capitalizer c = String::toUpperCase; //This works

use 3'rd construct - Class::instanceMethod. In this case first parameter becomes the target of the method. This construct is equivalent (translates) to following Lambda:

Capitalizer = (String x) -> x.toUpperCase();

This Lambda expression works because Lambda gets String as parameter and returns String result - as required by Capitalizer interface.

The line:

c = Main::toUpperCase; //Compile error

Translates to:

(Main m) ->  m.toUpperCase();

Which does not work with the Capitalizer interface. You could verify this by changing Capitalizer to:

interface Capitalizer {
    public String capitalize(Main name);
}

After this change Main::toUpperCase will compile.

like image 73
gniewkos Avatar answered Dec 28 '22 05:12

gniewkos


You have a method which

public String capitalize(String name);

Takes a String and returns a String. Such a method can have a number of patterns.

A constructor

c = String::new; // calls new String(String)
// or
c = s -> new String(s);

A function on String which takes no arguments

c = String::toLowerCase; // instance method String::toLowerCase()
// or
c = s -> s.toLowerCase();

of a method which takes a String as the only argument

// method which takes a String, but not a Main
public static String toUpperCase(String str) { 

c = Main::toUpperCase;
// or
c = s -> toUpperCase(s);

In every case, the method referenced has to take the String.

If not you can do this instead.

c = s -> capitalize(); // assuming Main.capitalize() is static

This tells the compiler to ignore the input.

like image 40
Peter Lawrey Avatar answered Dec 28 '22 05:12

Peter Lawrey