Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Behavior of Functional Interface and Method Reference

What happens when the reference of a method which belongs to a variable is destroyed?

public class Hey{
    public double bar;

    public Hey(){
        bar = 2.0d;
    }

    public double square(double num){
        return Math.pow(num , bar);
    }
}

Function<Double, Double> square;
whatsGonnaHappen: {
    Hey hey = new Hey();
    square = hey::square;
}//is hey still kept around because its method is being referenced?

double ans = square.apply(23d);
like image 873
Michael Avatar asked Dec 11 '22 20:12

Michael


2 Answers

Scope is a compile time concept that governs where names in source code can be used. From the JLS

The scope of a declaration is the region of the program within which the entity declared by the declaration can be referred to using a simple name, provided it is visible (§6.4.1).

The fact that the name hey is restricted to the body of the whatsGonnaHappen labeled statement doesn't have any bearing on whether the instance referenced by hey at runtime is a candidate for garbage collection (which I assume is what you're worried about).

Concerning the variable capture, yes, the method reference hey::square is a reference to an instance method of a particular object (the one referenced by hey) and will therefore capture the value of the variable hey when the method reference expression is evaluated to produce an instance and use it when apply is invoked.

like image 81
Sotirios Delimanolis Avatar answered Dec 13 '22 10:12

Sotirios Delimanolis


Your method reference is essentially equivalent to doing this:

Hey hey = new Hey();
Function<Double, Double> square = new DoubleDoubleFunction(hey);

where the class DoubleDoubleFunction is defined like this

class DoubleDoubleFunction implements Function<Double, Double> {
    private final Hey hey;

    public DoubleDoubleFunction(Hey hey) {
        this.hey = hey;
    }

    @Override
    public Double apply(Double num) {
        return hey.square(num);
    }
}

In other words, square holds a reference to the Hey.

I don't know about eclipse, but with IntelliJ you can easily translate method references and lambdas into anonymous / static nested / inner classes to see what's going on. I find this a very instructive thing to do.

like image 27
Paul Boddington Avatar answered Dec 13 '22 10:12

Paul Boddington