Skip to content
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

Make LU factorization work for more types #26344

Merged
merged 7 commits into from
Jun 6, 2018

Conversation

blegat
Copy link
Contributor

@blegat blegat commented Mar 6, 2018

When determining the element type need, lufact does the following

S = typeof(zero(T)/one(T))

However LU factorization also multiplies elements of type T together and sum them together.
If the type T is not invariant under multiplication and/or addition, lufact does work, see JuliaAlgebra/MultivariatePolynomials.jl#69.
Note the strong similary between this PR and #18218 which did a similar change but to matrix multiplication.

@ararslan ararslan added linear algebra Linear algebra types and dispatch Types, subtyping and method dispatch labels Mar 6, 2018
Copy link
Member

@andreasnoack andreasnoack left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍 Minor detail but maybe we should call it something like rationalop since it would probably be the right operation for all triangular-triangular factorizations.

@fredrikekre
Copy link
Member

Better as something like

lutype(T) = (zero(T)*zero(T) + zero(T)*zero(T)) / (one(T)*one(T) + one(T)*one(T))

or is it OK to rely on promote_op?

@blegat
Copy link
Contributor Author

blegat commented Mar 8, 2018

we should call it something like rationalop since it would probably be the right operation for all triangular-triangular factorizations

Agreed, I have renamed it

is it OK to rely on promote_op ?

I have assumed that it was ok since matrix product relies on it

function (*)(A::AbstractMatrix, B::AbstractMatrix)
TS = promote_op(matprod, eltype(A), eltype(B))

but you think otherwise, I can change, let me know :)

@martinholters
Copy link
Member

martinholters commented Mar 8, 2018

Use of promote_op should be avoided if possible. Ceterum censeo promote_op esse delendam.

@andreasnoack
Copy link
Member

User of promote_op should be avoided if possible.

The price of using instances through zero(T) and one(T) here is that it might fail for types where these don't exist. That was the issue for matrix multiplication so it could also happen here although it probably is less likely.

@blegat After thinking a little more about this, I think we might be able to compute the type in an even better way here. This was discussed in JuliaLang/LinearAlgebra.jl#128 and actually, the issue isn't completely fixed. The problem is that L is unitless while U has units but they'd have to be stored in the same array. I think something like

UT = typeof(zero(T) - oneunit(T)/oneunit(T)*oneunit(T))
LT = typeof(oneunit(T)/oneunit(T))
S = promote_type(T, UT, LT)

should be correct. In most cases it should just collapse to a single type and for unitful elements, it should provide the right union.

@blegat
Copy link
Contributor Author

blegat commented Jun 3, 2018

@andreasnoack I have updated the PR based on your idea. The code now also works for quantities with units (tested with Unitful)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
linear algebra Linear algebra
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants