编写返回函数的 Julia 宏

alb*_*bep 6 metaprogramming julia

第一次发帖,感谢阅读!

问题:我有一个Vector{String}- 调用它A- 其中每个元素都是等式的一部分,例如Ais的第一个元素"x[1] - (0.8*x[1])"。我想编写一个宏,它作为参数 i) a String- 调用它fn_name- 使用函数的名称,ii) 向量A,并返回一个名为的函数fn_name,它看起来像

function fn_name(f, x)
  f[1] = x[1] - (0.8*x[1])
  f[2] = (exp(x[4]) - 0.8*exp(x[3]))^(-1.1) - (0.99*(exp(x[4]) - 0.8*exp(x[4]))^(-1.1)*(1.0 - 0.025 + 0.30*exp(x[1])*exp(x[2])^(0.30 - 1.0)))
  f[3] = exp(x[2]) - ((1.0 - 0.025)*exp(x[2]) + exp(x[1])*exp(x[2])^0.30 - exp(x[4]))
  f[4] = x[3] - (x[4])
end 

Run Code Online (Sandbox Code Playgroud)

其中每个 rhs 是一个元素

A = ["x[1] - (0.8*x[1])", "(exp(x[4]) - 0.8*exp(x[3]))^(-1.1) - (0.99*(exp(x[4]) - 0.8*exp(x[4]))^(-1.1)*(1.0 - 0.025 + 0.30*exp(x[1])*exp(x[2])^(0.30 - 1.0)))", "exp(x[2]) - ((1.0 - 0.025)*exp(x[2]) + exp(x[1])*exp(x[2])^0.30 - exp(x[4]))", "x[3] - (x[4])"]
Run Code Online (Sandbox Code Playgroud)

我的尝试:我解决问题的最佳尝试如下

macro make_fn(fn_name, A)
    esc(quote
        function $(Symbol(fn_name))(f, x)
            for i = 1:length($(A))
                f[$i] = Meta.parse($(A)[$i])
            end
        end
    end)
end
Run Code Online (Sandbox Code Playgroud)

但是这不起作用:当我运行时@make_fn("my_name", A)出现错误LoadError: UndefVarError: i not defined

我发现很难完全理解 Julia 元编程,虽然我很乐意避免使用它,但我认为这个问题是不可避免的。

你能帮我理解我的错误在哪里吗?

谢谢

phi*_*ler 4

在这种情况下,宏不仅是可以避免的,甚至是不适用的,除非A在编译时是已知的。

eval我可以使用一些闭包提供解决方案:

julia> function make_fn2(A)
           Af = [@eval(x -> $(Meta.parse(expr))) for expr in A]
           function (f, x)
               for i in eachindex(A, f)
                   f[i] = Af[i](x)
               end
               return f
           end
       end
make_fn2 (generic function with 1 method)

julia> fn_name = make_fn2(A)
#46 (generic function with 1 method)

julia> fn_name(zeros(4), [1,2,3,4])
4-element Array{Float64,1}:
  0.19999999999999996
 -0.06594092302655707
 49.82984401122239
 -1.0

Run Code Online (Sandbox Code Playgroud)

与限制

  1. eval将在定义此模块的全局范围内评估表达式(因此它可能是与调用函数的范围不同的范围),并且
  2. 仅当您首先返回全局范围时,新创建的函数才会起作用(即,如果您尝试在创建它的函数中运行它,它将不起作用)。

但我真的建议考虑一种比字符串更好的输入格式。

  • [我可能有第 1 点的解决方案。](https://gist.github.com/phipsgabler/8c80e3af2c06581329f67d88ffd8944a),但评论太长且未经过正确测试。 (2认同)