Замыкания Groovy и перегруженные методы с функциональными параметрами

Я получаю Ambiguous error, когда пытаюсь использовать код, содержащий перегруженные методы с функциональными аргументами.

Я написал небольшой фрагмент, который показывает неоднозначное поведение:

import java.util.function.BiConsumer
import java.util.function.Consumer

class Test {

    static void main(String... args) {
        execute({ x -> println("Consumer") })
        execute({ x, y -> println("BiConsumer") })
    }

    static void execute(Consumer<Integer> c) {
        c.accept(100)
    }

    static void execute(BiConsumer<Integer, Integer> c) {
        c.accept(1, 2)
    }
}

Вывод (2.4.9 заводной):

    Exception in thread "main" groovy.lang.GroovyRuntimeException: Ambiguous method overloading for method Test#execute.
Cannot resolve which method to invoke for [class Test$_main_closure1] due to overlapping prototypes between:
    [interface java.util.function.BiConsumer]
    [interface java.util.function.Consumer]
    at groovy.lang.MetaClassImpl.chooseMostSpecificParams(MetaClassImpl.java:3268)
    at groovy.lang.MetaClassImpl.chooseMethodInternal(MetaClassImpl.java:3221)
    at groovy.lang.MetaClassImpl.chooseMethod(MetaClassImpl.java:3164)
    at groovy.lang.MetaClassImpl.pickStaticMethod(MetaClassImpl.java:1516)
    at groovy.lang.MetaClassImpl.retrieveStaticMethod(MetaClassImpl.java:1412)
    at org.codehaus.groovy.vmplugin.v7.Selector$MethodSelector.chooseMeta(Selector.java:553)
    at org.codehaus.groovy.vmplugin.v7.Selector$MethodSelector.setCallSiteTarget(Selector.java:954)
    at org.codehaus.groovy.vmplugin.v7.IndyInterface.selectMethod(IndyInterface.java:228)
    at Test.main(Test.groovy:7)

Но это работает с java-лямбдами, и я не понимаю, как использовать groovy-замыкания в этой ситуации.


person Alex Suslov    schedule 09.03.2017    source источник


Ответы (2)


Вам нужно поставить as XXX после ваших замыканий, чтобы дать среде выполнения подсказку, куда вы хотите пойти.

    execute({ x -> println("Consumer") } as Consumer)
    execute({ x, y -> println("BiConsumer") } as BiConsumer)

Должен сделать это

person tim_yates    schedule 09.03.2017
comment
Спасибо, это помогает в этой ситуации. Но напр. если бы какая-то java-библиотека изменила свой API с добавлением новых перегруженных методов, это могло бы сломать заводной код? - person Alex Suslov; 09.03.2017
comment
Вы можете использовать @CompileStatic, чтобы поймать их во время компиляции. - person tim_yates; 09.03.2017

Возможно, с помощью замыкания:

execute(Closure <Integer> c) {
  c (10)
}

execute (Closure <Integer, Integer> c) {
  c (10, 30)
}
person Skaparate    schedule 09.03.2017
comment
Вы не можете сделать Closure<Integer, Integer>, закрытие принимает только один общий параметр, возвращаемый тип закрытия - person tim_yates; 09.03.2017