Groovysh использует пользовательскую команду внутри цикла

У меня есть проблема с groovysh, когда я заметил, что вы не можете использовать команды goovysh внутри контекста цикла или внутри функций. Кажется, что команды оцениваются во время синтаксического анализа, а не во время выполнения.

Есть ли какой-то волшебный синтаксис, чтобы обойти это?

Вот пример этого:

import org.codehaus.groovy.tools.shell.CommandSupport
import org.codehaus.groovy.tools.shell.Groovysh

class Rand extends CommandSupport {
    private Random random = new Random()

    protected Rand(final Groovysh shell) {
        super(shell, 'rand', 'r')
    }

    public Integer execute(List args) {
        random.nextInt()
    }

}

:register Rand

(1..3).each {
    println "number ${it}"
    rand
    foo = _
    println "Random number is ${foo}"
}

При выполнении вы видите, что случайное число не меняется, и вы можете видеть, что оно оценивалось, когда я вставил код в консоль, но до того, как он прошел через цикл:

Groovy Shell (2.4.11, JVM: 1.8.0_51)
Type ':help' or ':h' for help.
-----------------------------------------------------------------------------------------------------------------------
groovy:000> import org.codehaus.groovy.tools.shell.CommandSupport
===> org.codehaus.groovy.tools.shell.CommandSupport
groovy:000> import org.codehaus.groovy.tools.shell.Groovysh
===> org.codehaus.groovy.tools.shell.CommandSupport, org.codehaus.groovy.tools.shell.Groovysh
groovy:000>
groovy:000> class Rand extends CommandSupport {
groovy:001>     private Random random = new Random()
groovy:002>
groovy:002>     protected Rand(final Groovysh shell) {
groovy:003>         super(shell, 'rand', 'r')
groovy:004>     }
groovy:005>
groovy:005>     public Integer execute(List args) {
groovy:006>         random.nextInt()
groovy:007>     }
groovy:008>
groovy:008> }
===> true
groovy:000>
groovy:000> :register Rand
===> true
groovy:000>
groovy:000> (1..3).each {
groovy:001>     println "number ${it}"
groovy:002>     rand
===> -1321819102
groovy:002>     foo = _
groovy:003>     println "Random number is ${foo}"
groovy:004> }
number 1
Random number is -1321819102
number 2
Random number is -1321819102
number 3
Random number is -1321819102
===> [1, 2, 3]
groovy:000>

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


person Joel Pearson    schedule 24.05.2017    source источник


Ответы (1)


Хорошо, я только что придумал немного хакерское решение. Получение экземпляра Groovysh означает, что я могу оценить, когда захочу:

import org.codehaus.groovy.tools.shell.CommandSupport
import org.codehaus.groovy.tools.shell.Groovysh

class Rand extends CommandSupport {
    private Random random = new Random()

    protected Rand(final Groovysh shell) {
        super(shell, 'rand', 'r')
    }

    public Integer execute(List args) {
        random.nextInt()
    }

}

:register Rand

class Shell extends CommandSupport {

    private Groovysh shellint

    protected Shell(final Groovysh shell) {
        super(shell, 'shell', 's')
        shellint = shell
    }

    public Groovysh execute(List args) {
        shellint
    }

}

:register Shell

shell
myshell = _

(1..3).each {
    println "number ${it}"
    foo = myshell.execute("rand")
    println "Random number is ${foo}"
}

С выходом:

Groovy Shell (2.4.11, JVM: 1.8.0_51)
Type ':help' or ':h' for help.
-----------------------------------------------------------------------------------------------------------------------
groovy:000> import org.codehaus.groovy.tools.shell.CommandSupport
===> org.codehaus.groovy.tools.shell.CommandSupport
groovy:000> import org.codehaus.groovy.tools.shell.Groovysh
===> org.codehaus.groovy.tools.shell.CommandSupport, org.codehaus.groovy.tools.shell.Groovysh
groovy:000>
groovy:000> class Rand extends CommandSupport {
groovy:001>     private Random random = new Random()
groovy:002>
groovy:002>     protected Rand(final Groovysh shell) {
groovy:003>         super(shell, 'rand', 'r')
groovy:004>     }
groovy:005>
groovy:005>     public Integer execute(List args) {
groovy:006>         random.nextInt()
groovy:007>     }
groovy:008>
groovy:008> }
===> true
groovy:000>
groovy:000> :register Rand
===> true
groovy:000>
groovy:000> class Shell extends CommandSupport {
groovy:001>
groovy:001>     private Groovysh shellint
groovy:002>
groovy:002>     protected Shell(final Groovysh shell) {
groovy:003>         super(shell, 'shell', 's')
groovy:004>         shellint = shell
groovy:005>     }
groovy:006>
groovy:006>     public Groovysh execute(List args) {
groovy:007>         shellint
groovy:008>     }
groovy:009>
groovy:009> }
===> true
groovy:000>
groovy:000> :register Shell
===> true
groovy:000>
groovy:000> shell
===> org.codehaus.groovy.tools.shell.Groovysh@16ec132
groovy:000> myshell = _
===> org.codehaus.groovy.tools.shell.Groovysh@16ec132
groovy:000>
groovy:000> (1..3).each {
groovy:001>     println "number ${it}"
groovy:002>     foo = myshell.execute("rand")
groovy:003>     println "Random number is ${foo}"
groovy:004> }
number 1
===> -666149132
Random number is -666149132
number 2
===> -1675600826
Random number is -1675600826
number 3
===> 412144734
Random number is 412144734
===> [1, 2, 3]

Есть ли другой способ сделать это? В контексте, в котором мне это нужно, groovysh - это настроенный с удаленным :register

Я изменил файл jar groovysh, чтобы снова добавить команду :register, а затем я мог использовать приведенное выше решение. Я сделал это, просмотрев https://github.com/groovy/groovy-core/blob/master/subprojects/groovy-groovysh/src/main/groovy/org/codehaus/groovy/tools/shell./Groovysh.groovy и увидев, что org/codehaus/groovy/tools/shell/commands.xml содержит список команд, я добавил <command>org.codehaus.groovy.tools.shell.commands.RegisterCommand</command> в список.

person Joel Pearson    schedule 24.05.2017