Don’t think that is possible, since Union types are immutable
julia> x = Union{Float64,Array}
Union{Float64, Array}
julia> x.a
Float64
julia> x.a = Int
ERROR: type Union is immutable
Stacktrace:
[1] setproperty!(::Type, ::Symbol, ::Type) at ./sysimg.jl:16
[2] top-level scope at none:0
You could keep an array of types and check typeof(x) in SymExpr, where SymExpr = [Symbol,Expr,Int], then you could add to the array.
Why would you not want to have Int in your union in the first place? Why does it need to be dynamic? Are you only going to add Int to it, or is there a more general pattern you need to satisfy?
I’m trying to understand more about your motivations behind this.
The only way to change dispatch is to define (or delete…) methods so no you can’t do anything else and change dispatch. You can always add your own type check that is as dynamic as you want though.
Have you read the link I posted above? It tells you how to do it:
struct SymExpr end
SymExpr(::Expr) = SymExpr()
SymExpr(::Symbol) = SymExpr()
SymExpr(::Any) = nothing
# Do the trait dispatch
# f(::SymExpr) = 1
# needs to be written as
f(x) = _f(SymExpr(x), x)
_f(::SymExpr, x) = 1
_f(::Nothing, x) = 2 # or leave it away for a method error
f(:a) # 1
f(1) # 2
SymExpr(::Int) = SymExpr()
f(1) # 1
If you can’t be bothered with writing out the trait-dispatch (and you just need a simple trait-dispatch setup) then you can use https://github.com/mauro3/SimpleTraits.jl
using SimpleTraits
@traitdef SymExpr{X}
@traitimpl SymExpr{Expr}
@traitimpl SymExpr{Symbol}
@traitfn f(::::SymExpr) = 1
@traitfn f(::::!(SymExpr)) = 2 # again, not defining this will just give a Method error
f(:a) # 1
f(:(1+1)) # 1
f(1) # 2
@traitimpl SymExpr{Int}
f(1) # 1
I thank you very much for your kind answer mauro3.
I have figured out a similar way to yours, I have my own traits package.
One of the reasons I asked that question to see what type level programming utilities Julia is providing.
Thanks