-
Notifications
You must be signed in to change notification settings - Fork 13
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Can StaticArrays dependency be made optional? #91
Comments
Hmm, I did not think of StaticArrays as a heavy dependency, but perhaps it is. For the time being, the main reason for DomainSets being larger than it should be is that a significant part of the code has to do with maps (function mappings). We plan to separate that out, though it will remain a dependency. I'm afraid quite a bit of code is specific to StaticArrays, in particular when dealing with product domains. E.g. you typically want the product of a rectangle with an interval to behave like a cube with simple vector elements, rather than some nested vector. We support both, but static arrays are given preferential treatment in many places. That said, I don't know Static.jl, but it might work. |
In most contexts it's really not. But in development of MeasureTheory, there was consistent feedback that we needed a lightweight "core" library with fewer dependencies; this led me to factor out MeasureBase.jl. To me, having StaticArrays as a dependency is not such a big deal. But I've been miscalibrated on this before, so I want to be cautious about it. Static.jl is really nice! You don't need separate types for static vs dynamic size information, for example you can have struct Foo{T}
dimension::T
end Then an instance might be Also worth checking out is ArrayInterface which uses |
Hi @daanhb , Thinking some more about this, the bigger challenge might be that products seem to always be represented using Do you think there could be a way to extend this so other array formats can be used? Or maybe this is too far outside the goals of the package? |
That is definitely within scope. That is in fact one of the reasons there are three subtypes of Some examples: julia> d1 = ProductDomain(map(Interval, rand(10), rand(10)))
D₁ × D₃ × D₂ × ... × D₄
D₁ = 0.04102775113924917..0.7543070839059667
D₂ = 0.0301860782231258..0.3334968805199827
D₃ = 0.1203453376504664..0.31294581525887777
D₄ = 0.8943245774798796..0.021646286220368793
julia> typeof(d1)
Rectangle{Vector{Float64}}
julia> eltype(d1)
Vector{Float64} (alias for Array{Float64, 1}) The julia> d2 = ProductDomain(0..1, UnitCircle())
(0.0..1.0) × UnitCircle()
julia> typeof(d2)
VcatDomain{3, Float64, (1, 2), Tuple{ClosedInterval{Float64}, EuclideanUnitSphere{2, Float64}}}
julia> eltype(d2)
StaticArrays.SVector{3, Float64} (alias for StaticArrays.SArray{Tuple{3}, Float64, 1, 3})
julia> d2.domains
(0.0..1.0, UnitCircle()) Here, the constructor is given two domains. The best guess is that the user wants to concatenate the 1D domain and the 2D domain, hence the result is a julia> TupleProductDomain(0..1, UnitCircle())
(0..1) × UnitCircle()
julia> eltype(ans)
Tuple{Int64, StaticArrays.SVector{2, Float64}} Now, on to vectors of domains: julia> ProductDomain(map(x -> x*UnitCircle(), 1:5))
Sphere(1.0, [0.0, 0.0]) × Sphere(2.0, [0.0, 0.0]) × Sphere(3.0, [0.0, 0.0]) × Sphere(4.0, [0.0, 0.0]) × Sphere(5.0, [0.0, 0.0])
julia> eltype(ans)
Vector{StaticArrays.SVector{2, Float64}} (alias for Array{StaticArrays.SArray{Tuple{2}, Float64, 1, 2}, 1}) This domain is the concatenation of circles with increasing radius. The element type is a julia> eltype(UnitSphere(Val(2)))
StaticArrays.SVector{2, Float64} (alias for StaticArrays.SArray{Tuple{2}, Float64, 1, 2})
julia> eltype(UnitSphere(2))
Vector{Float64} (alias for Array{Float64, 1})
julia> ProductDomain(map(x -> x*UnitSphere(2), 1:5))
Sphere(1.0, [0.0, 0.0]) × Sphere(2.0, [0.0, 0.0]) × Sphere(3.0, [0.0, 0.0]) × Sphere(4.0, [0.0, 0.0]) × Sphere(5.0, [0.0, 0.0])
julia> eltype(ans)
Vector{Vector{Float64}} (alias for Array{Array{Float64, 1}, 1}) The representation of the factors of a product domain is mostly based on how There are some trade-offs. The reason tuples are the default when possible is that they are more efficient for mixed types. A |
Well it looks like we can depend on StaticArraysCore.jl instead, without any other changes. It improves load time by about 0.5s on my machine (a reduction of more than 50%). Anyone working with higher-dimensional domains, and who hasn't separately loaded StaticArrays, will get an uninformative error message though (like |
From JuliaMath/MeasureBase.jl#16:
The text was updated successfully, but these errors were encountered: