Julia+JuMP:函数参数数量可变

Dan*_*ton 4 julia ipopt julia-jump

我正在尝试使用 JuMP 来解决非线性问题,其中变量的数量由用户决定 - 也就是说,在编译时未知。

为了实现这一点,该@NLobjective行如下所示:

@eval @JuMP.NLobjective(m, Min, $(Expr(:call, :myf, [Expr(:ref, :x, i) for i=1:n]...)))
Run Code Online (Sandbox Code Playgroud)

例如,如果n=3,编译器将该行解释为与以下内容相同:

@JuMP.NLobjective(m, Min, myf(x[1], x[2], x[3]))
Run Code Online (Sandbox Code Playgroud)

问题是它@eval仅在全局范围内有效,并且当包含在函数中时,会引发错误。

我的问题是:如何在函数的本地编译范围内完成相同的功能(使用可变数量的参数@NLobjective进行调用)?myfx[1],...,x[n]

def testme(n)
    myf(a...) = sum(collect(a).^2)

    m = JuMP.Model(solver=Ipopt.IpoptSolver())

    JuMP.register(m, :myf, n, myf, autodiff=true)
    @JuMP.variable(m, x[1:n] >= 0.5)

    @eval @JuMP.NLobjective(m, Min, $(Expr(:call, :myf, [Expr(:ref, :x, i) for i=1:n]...)))
    JuMP.solve(m)
end

testme(3)
Run Code Online (Sandbox Code Playgroud)

谢谢!

Dan*_*etz 5

正如http://jump.readthedocs.io/en/latest/nlp.html#raw-expression-input中所解释的,可以在没有宏的情况下给出目标函数。相关表述:

    JuMP.setNLobjective(m, :Min, Expr(:call, :myf, [x[i] for i=1:n]...))
Run Code Online (Sandbox Code Playgroud)

@eval基于的更简单并且在函数中工作。代码是:

using JuMP, Ipopt

function testme(n)
    myf(a...) = sum(collect(a).^2)

    m = JuMP.Model(solver=Ipopt.IpoptSolver())

    JuMP.register(m, :myf, n, myf, autodiff=true)
    @JuMP.variable(m, x[1:n] >= 0.5)

    JuMP.setNLobjective(m, :Min, Expr(:call, :myf, [x[i] for i=1:n]...))
    JuMP.solve(m)
    return [getvalue(x[i]) for i=1:n]
end

testme(3)
Run Code Online (Sandbox Code Playgroud)

它返回:

julia> testme(3)

:

 EXIT: Optimal Solution Found.
3-element Array{Float64,1}:
 0.5
 0.5
 0.5
Run Code Online (Sandbox Code Playgroud)

  • 这是一个很好的技巧,连我自己都没有想到。对于阅读本文的任何人,我会警告不要将“autodiff=true”与高维输入函数一起使用。当前的实现使用前向模式AD,随着输入维度的增加,它不能很好地扩展。 (2认同)