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

broadcast makes sparse matrix full #11474

Closed
mauro3 opened this issue May 28, 2015 · 26 comments
Closed

broadcast makes sparse matrix full #11474

mauro3 opened this issue May 28, 2015 · 26 comments
Labels
broadcast Applying a function over a collection sparse Sparse arrays
Milestone

Comments

@mauro3
Copy link
Contributor

mauro3 commented May 28, 2015

julia> sp = sprand(3,3,0.1)
3x3 sparse matrix with 1 Float64 entries:
        [2, 2]  =  0.885289

julia> broadcast(*, [1,1,1], sp)
3x3 Array{Float64,2}:
 0.0  0.0       0.0
 0.0  0.885289  0.0
 0.0  0.0       0.0
@pao
Copy link
Member

pao commented May 28, 2015

@mauro3 Which version of Julia?

@mauro3
Copy link
Contributor Author

mauro3 commented May 28, 2015

Sorry, both 0.3 and 0.4. (edit: oddly broadcasting .* works in 0.4 but not in 0.3: #11475)

@JeffBezanson JeffBezanson added the sparse Sparse arrays label May 28, 2015
@tkelman
Copy link
Contributor

tkelman commented May 29, 2015

This is because of the Array on this line:

broadcast(f::Function, As...) = broadcast!(f, Array(promote_eltype(As...), broadcast_shape(As...)), As...)

You should be able to do either broadcast!(*, copy(sp), [1,1,1], sp), or the more-specialized but non-exported SparseMatrix.broadcast_zpreserving(*, [1,1,1], sp), ref #10106

@mauro3
Copy link
Contributor Author

mauro3 commented May 29, 2015

So, should it be a similar there instead?

And the broadcast! trick works on 0.3 as well, thanks for the work-around. (Which is good enough for me, so no need to fix this on 0.3 from my point of view).

@tkelman
Copy link
Contributor

tkelman commented May 29, 2015

Well, similar to which of the inputs? We'd need something to the effect of promote_containertype which we don't really have right now, and would seem to be quite difficult to do fully generically. We should try to move in that direction of being smarter and more generic about container types instead of falling back to dense Array all the time, but it's going to take some time and effort to get there.

@ViralBShah
Copy link
Member

I am pretty sure the dense output is due to a generic fallback.

@ViralBShah
Copy link
Member

Perhaps if one of the inputs is sparse, we should dispatch to the sparse version.

julia> @which broadcast(*, [1,1,1], sp)
broadcast(f::Function, As...) at broadcast.jl:236

julia> @which broadcast(*, sparse([1,1,1]), sp)
broadcast{Tv1,Ti1,Tv2,Ti2}(f::Function, A_1::Base.SparseMatrix.SparseMatrixCSC{Tv1,Ti1}, A_2::Base.SparseMatrix.SparseMatrixCSC{Tv2,Ti2}) at sparse/sparsematrix.jl:854

@tkelman
Copy link
Contributor

tkelman commented Jun 12, 2015

Not unless we know a-priori that the function is zero-preserving?

On second thought, maybe that would make sense, but the behavior of the function on zeros would have to be checked for that case.

@toivoh
Copy link
Contributor

toivoh commented Jun 13, 2015

With only one sparse input, I think it doesn't make sense to produce a
sparse output unless the function products a zero if either input is zero,
like * does (but not so many other functions).

@ViralBShah
Copy link
Member

This seems like it should be an easy thing to test. The unfortunate thing is that we will end up with a type unstable solution. I doubt that will be a problem, but still worth keeping in mind.

@ViralBShah ViralBShah added this to the 0.5 milestone Jul 3, 2015
@tkelman
Copy link
Contributor

tkelman commented Apr 21, 2016

I don't think we can add a whole lot of cleverness to broadcast on sparse matrices in time for 0.5. There are sparse-specialized internal methods but you have to look for them and your operation needs to satisfy the invariants.

@tkelman
Copy link
Contributor

tkelman commented Dec 15, 2016

@Sacha0 is there zero-preserving-detection machinery in sparse broadcast that should be kicking in on this case now? This still gives me the same behavior at 1362268

@Sacha0
Copy link
Member

Sacha0 commented Dec 16, 2016

is there zero-preserving-detection machinery in sparse broadcast that should be kicking in on this case now?

The existing machinery should only kick in if all arguments are SparseMatrixCSCs. In the example, one argument is an Array. Whether broadcast over combinations of SparseMatrixCSCs and Arrays should yield an Array or a SparseMatrixCSC has not been decided yet if memory serves? Best!

@Sacha0
Copy link
Member

Sacha0 commented Jan 1, 2017

Consensus seems in favor of returning an Array from broadcast when the arguments include at least one Array (alongside sparse/structured array types). As such and with sparse broadcast now functional, anything left to address in this issue? Best!

@tkelman
Copy link
Contributor

tkelman commented Jan 1, 2017

where's that? for concatenation we promote combinations of dense and sparse to sparse. the potential benefit here for operations like multiplication to promote results to sparse is large. dense in sparse format is not a terrible fallback when the operation acts more like addition, and you could use broadcast! if you really wanted a dense format output

@nalimilan
Copy link
Member

We really need a common and generic mechanism to choose the return type for cat and promote. Cf. #2326.

@Sacha0
Copy link
Member

Sacha0 commented Jan 7, 2017

where's that?

Vague impression from reading comments in the week prior to that post, but that vague impression could well have been the product of my then sleep-deprivation-addled mind :).

for concatenation we promote combinations of dense and sparse to sparse.

Good point!

the potential benefit here for operations like multiplication to promote results to sparse is large.

Yes, given still more sophisticated generic sparse broadcast, though not presently. Best!

@Sacha0
Copy link
Member

Sacha0 commented Jan 7, 2017

We really need a common and generic mechanism to choose the return type for cat and promote. Cf. #2326.

Agreed. Base.Broadcast.broadcast_containertype seems to be evolving into that mechanism? Best!

@nalimilan
Copy link
Member

Agreed. Base.Broadcast.broadcast_containertype seems to be evolving into that mechanism? Best!

Yes, maybe, in which case it could be renamed to something more general. You understand these issues better than I do.

@Sacha0
Copy link
Member

Sacha0 commented Jan 8, 2017

You understand these issues better than I do.

Doubtful :).

Near term (i.e. for 0.6), when broadcast'ing over combinations including both dense arrays and sparse/structured arrays, we could treat dense arrays as we do(/will, #19926) structured arrays, i.e. promote them to sparse. That change should be straightforward and would close this issue. Thoughts? Thanks!

@nalimilan
Copy link
Member

Shouldn't the type of the result depend on whether the operation is zero-preserving? It wouldn't be great to have sp .+ a return a sparse matrix with no zero entries; better return a dense array in that case.

@tkelman
Copy link
Contributor

tkelman commented Jan 9, 2017

That wouldn't be type stable though, since we can't dispatch on zero-preserving-ness from the type of a function alone.

@KristofferC
Copy link
Member

Memory of a dense sparse matrix is only ~2x a dense matrix. User can always convert it to a full matrix.

@tkelman
Copy link
Contributor

tkelman commented Jan 9, 2017

or use broadcast! or .= into a dense output

@nalimilan
Copy link
Member

That wouldn't be type stable though, since we can't dispatch on zero-preserving-ness from the type of a function alone.

OK, sorry, I haven't followed the discussions regarding this closely enough. But Sacha didn't believe me...

@Sacha0
Copy link
Member

Sacha0 commented Jan 25, 2017

The example from the original post now works (#20009):

julia> sp = sprand(3,3,0.1)
3×3 sparse matrix with 2 Float64 stored entries:
  [1, 1]  =  0.930125
  [3, 2]  =  0.781933

julia> broadcast(*, [1,1,1], sp)
3×3 sparse matrix with 2 Float64 stored entries:
  [1, 1]  =  0.930125
  [3, 2]  =  0.781933

Best!

@Sacha0 Sacha0 closed this as completed Jan 25, 2017
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
broadcast Applying a function over a collection sparse Sparse arrays
Projects
None yet
Development

No branches or pull requests