would perfectly work but I thought dataFolder as a variable might be more aesthetic? but you are not suggesting, I see. Then I think I will stick with my dictionary.
You could make it look a bit more aesthetic by defining a getproperty method that gives you slightly nicer my_dict.dataFolder syntax instead of my_dict["dataFolder"].
Some limitations of that approach:
you need my_dict to be a type that you’ve defined yourself otherwise defining a
Base.getproperty(d::Dict, s::Symbol) = ...
is definitely type piracy since you own neither Dict, Symbol, nor getproperty. So you’d need to wrap Dict in your own type:
struct MyDict
dict::Dict
end
Base.getproperty(d::MyDict, s::Symbol) = getfield(d, :dict)[string(s)]
The keys of the Dict must be valid Julia identifiers, so no dashes, spaces, etc.
You can make values stored in my_dict into variables if you use eval:
mod = Module()
for (k, v) in my_dict
Core.eval(mod, :($(Symbol(k)) = $v))
end
@assert mod.dataFolder == "my_data"
But as @jling mentioned above, generally this is not what you should be doing, and in this example doesn’t actually provide any benefit over the getproperty approach. It has it’s place at times, but those are pretty limited, and really a last resort.
Another way may be with keyword arguments like:
julia> function use_variables(;
tile_size = error("tile_size not provided"),
dataFolder = error("dataFolder not provided"),
dims = error("dims not provided"),
kws... # ignore anything else.
)
@show tile_size dataFolder dims # use as variables inside function.
# ...
end
julia> use_variables(; (Symbol(k)=>v for (k,v) in my_dict)...)
tile_size = [100, 200]
dataFolder = "my_data"
dims = 20
20
Which might work if you only need a predefined set of variables to be handled.
Hi, just necroposting to say that something like this is easy with the new destructuring syntax. Just do:
(; dataFolder, tile_size, dims) = my_dict
It’s also not bad practice because you are still manually specifying which variables to create. The only issue might that there will be type instability, as the types of the values of my_dict are not in general inferred.
You can define a simple function to convert the Dict recursively to a NamedTuple:
julia> named_tuple(dict::AbstractDict) = (; [Symbol(k) => named_tuple(v) for (k, v) in pairs(dict) ]... )
named_tuple (generic function with 1 method)
julia> named_tuple(x) = x
named_tuple (generic function with 2 methods)
This is a nice idea. Personally I also much prefer using a.x instead of a["x"].
I have used this in the past but I was never sure if this is something that one needs to have a bit of caution with do if one does not know that the size of the dict will be small.
Since large tuples put a lot of stress on the compiler, this might become problematic as well right?
I wonder if anyone has some wisdom about this.