Struct 如何将Julia结构完全解压为局部变量?

Struct 如何将Julia结构完全解压为局部变量?,struct,julia,metaprogramming,Struct,Julia,Metaprogramming,我已经创建了一个包含很多参数的复杂计算模型。因为我需要运行许多场景,所以我决定将所有这些输入参数包装到一个巨大的结构中: using Parameters @with_kw struct MyModel a::Int = 5 b::Float64 = 5.5 c::Matrix{Float64} = rand(3,4) # 40 other parameters go here end 例如,我有一个对象m: m = MyModel(a=15) 现在写数学代码

我已经创建了一个包含很多参数的复杂计算模型。因为我需要运行许多场景,所以我决定将所有这些输入参数包装到一个巨大的
结构中

using Parameters
@with_kw struct MyModel
    a::Int = 5
    b::Float64 = 5.5
    c::Matrix{Float64} = rand(3,4)
    # 40 other parameters go here
end
例如,我有一个对象
m

m = MyModel(a=15)
现在写数学代码时,我不想在每个符号前面写
m.
。因此,我需要将结构字段转换为局部变量。一种方法是使用
@unpack
宏:

@unpack a, b, c = m

对于我想要在各种函数中解包的大型结构,这很不方便(请注意,我的结构有大约40个字段)。如何解压结构而不花费时间并将代码与所有这些参数混在一起?

对于自定义情况,您可以为您的类型创建专用宏,对于其他情况,请参见上面的答案:

macro unpack_MyModel(q)
    code =  Expr(:block, [ :($field = $q.$field) for field in fieldnames(MyModel) ]...)
    esc(code)
end
这只需插入以下代码:

julia> @macroexpand @unpack_MyModel(m)
quote
    a = m.a
    b = m.b
    c = m.c
end
此宏可在任何函数内使用,例如:

function f(m::MyModel)
    @unpack_MyModel(m)
    return a+b
end

Parameters.jl中带有_kw的宏
@定义了一个用于此目的的宏:

julia> using Parameters

julia> @with_kw struct MyModel  # exactly as in the question
           a::Int = 5
           b::Float64 = 5.5
           c::Matrix{Float64} = rand(3,4)
           # 40 other parameters go here
       end
MyModel

julia> @macroexpand @unpack_MyModel x
quote
    a = x.a
    b = x.b
    c = x.c
end
因此,编写
@unpack\u MyModel m
相当于编写
@unpack a,b,c=m
,当您知道
m是MyModel

时,另一个选项是。以下是我从该软件包的自述文件中复制并粘贴的示例:

julia> struct Bar
           a
           b
       end

julia> @with Bar(1, 2) begin
           a^2, b^2
       end
(1, 4)

谢谢与自己编写宏相比,这确实节省了时间!我想这是希望包超过
基础的原因之一。@kwdef struct MyModel
,我最喜欢的理由是,当你在控制台中显示结构时,它的可视化效果更好:-)无论如何
?@with_kw
并不表示这个漂亮的宏得到了实现。通常我的计算代码相当大,并且经常包含其他宏调用(例如,因为使用了跳转)。我想知道对于复杂的代码块,
@和
的健壮性有多强。好问题。我还没有真正使用过StaticModules.jl