'UndefVarError: x not defined' 在其模块之外调用宏 (Julia v1.1)

FCo*_*FCo 4 julia

我正在尝试编写一个宏,该宏调用在同一模块中定义的函数(使用 Julia v1.1)。我总是得到ERROR: LoadError: UndefVarError: myModel not defined.

模块代码如下(Models.jl):

module Models

export Model
export createmodel
export add_variable
export @add_variable

mutable struct Model
    variable_symbols::Dict{String, Symbol}
    variable_data::Dict{Symbol, Array{Int64, 1}}
    Model(; variable_symbols = Dict([]), variable_data = Dict([])) = 
        new(variable_symbols, variable_data)
    Model(variable_symbols, variable_data) = new(variable_symbols, 
        variable_data)
end

function createmodel()
    model = Model()
    return model
end

function add_variable(model, label::String, value::Array{Int64, 1})
    symbol = Symbol(label)
    model.variable_symbols[label] = Symbol(symbol)
    model.variable_data[symbol] = value
end

macro add_variable(model, var)
    quote
        add_variable($model, $(string(var)), $var)
    end
end

end
Run Code Online (Sandbox Code Playgroud)

宏调用如下:

include("Models.jl")
using .Models

x = [1, 2, 3]

myModel = createmodel()
@add_variable(myModel, x)
show(myModel)
Run Code Online (Sandbox Code Playgroud)

我知道这是一个逃避/卫生问题,但在尝试了很多想法后,我没有得到预期的结果。到目前为止,我获得预期结果的唯一方法是在模块外部定义宏以获取:Model(Dict("x"=>:x), Dict(:x=>[1, 2, 3]))

Mat*_* B. 5

您只需要esc模拟传递给宏的变量:

macro add_variable(model, var)
    quote
        add_variable($(esc(model)), $(string(var)), $(esc(var)))
    end
end
Run Code Online (Sandbox Code Playgroud)

宏中的所有名称都在其定义的模块的上下文中解析,除非您使用esc它们 -esc它们在调用者的上下文中解析。您几乎总是想要esc宏的参数。add_variable但是,您希望在定义宏本身的位置解析诸如 之类的名称。