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

Add iterator for combinations #4735

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open

Add iterator for combinations #4735

wants to merge 1 commit into from

Conversation

thofma
Copy link
Collaborator

@thofma thofma commented Mar 15, 2025

No description provided.

@thofma thofma added enhancement New feature or request topic: combinatorics release notes: use title For PRs: the title of this PR is suitable for direct use in the release notes labels Mar 15, 2025
@thofma thofma requested a review from lgoettgens March 15, 2025 21:31
Copy link
Member

@lgoettgens lgoettgens left a comment

Choose a reason for hiding this comment

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

Could you please add some more extensive tests? You can either get inspiration at https://github.com/Nemocas/AbstractAlgebra.jl/pull/2034/files#diff-9383dd485d4500b18b2bfa2797938c914f4a8df63c205e8ea8ddcfe23a1ad538 or test/Combinatorics/EnumerativeCombinatorics/compositions.jl, but I would expect at least the following tests: edge-cases with zero (in particular combinations(0, 0)), something with allunique, testing the guaranteed iteration order (e.g. hardcoding the result of combinations(5,3) which is the smallest example where lex and revlex order differ, and some large example with issorted)

@lgoettgens
Copy link
Member

This implementation seems to be slower than AbstractAlgebra.combinations, even when wrapping another collect around it.

julia> @b collect(AbstractAlgebra.combinations(5,3))
615.111 ns (28 allocs: 1.578 KiB)

julia> @b collect(combinations(5,3))
807.188 ns (34 allocs: 1.312 KiB)

julia> @b collect(AbstractAlgebra.combinations(15,10))
250.671 μs (6027 allocs: 547.938 KiB)

julia> @b collect(combinations(15,10))
280.748 μs (9014 allocs: 539.844 KiB)

julia> @b collect(AbstractAlgebra.combinations(18,9))
3.293 ms (97269 allocs: 7.905 MiB)

julia> @b collect(combinations(18,9))
5.897 ms (145865 allocs: 7.790 MiB)

@thofma
Copy link
Collaborator Author

thofma commented Mar 15, 2025

Added more tests and fixed the timings (usual inlining nonsense)

@thofma

This comment was marked as off-topic.

@lgoettgens

This comment was marked as off-topic.

@thofma

This comment was marked as off-topic.

@joschmitt
Copy link
Member

Just to say before anything becomes Official API: For the other functions of this kind (partitions, compositions, ...), we have a dedicated type Partition, Composition, ...

@thofma
Copy link
Collaborator Author

thofma commented Mar 20, 2025

I am not quite sure what value that would add here.

@joschmitt
Copy link
Member

I am not quite sure what value that would add here.

We inherited the design from JuLie. From what I know, the idea was/is that this is helpful if one wants to work with combinations as objects themselves. So one can then write functions which take a combination as input or something.
I don't know whether there is interest in functionality in this direction, certainly not from my side.

@thofma
Copy link
Collaborator Author

thofma commented Mar 20, 2025

OK, makes sense. I can add that.

@thofma
Copy link
Collaborator Author

thofma commented Mar 20, 2025

Although, then it would not be a drop-in replacement for the current users of AbstractAlgebra.combinations

@lgoettgens
Copy link
Member

Just to say before anything becomes Official API: For the other functions of this kind (partitions, compositions, ...), we have a dedicated type Partition, Composition, ...

Although, then it would not be a drop-in replacement for the current users for AbstractAlgebra.combinations

What about having a dedicated return type for calls to combinations(::Int, ::Int), as in these cases people want some combinatorial object. But calls to combinations(::AbstractVector{T}, ::Int) could still return an iterator over Vector{T}.
For the already existing functions of this kind, we only have the former variant, so it would be consistent in this regard. But the latter could still be used as a drop-in-replacement for AbstractAlgebra.combinations.

Copy link
Member

@fingolfin fingolfin left a comment

Choose a reason for hiding this comment

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

Looks nice

end
break
end
state[1] > C.n - C.k + 1 && return
Copy link
Member

Choose a reason for hiding this comment

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

Maybe be explicit about returning nothing here, like in line 60?

Suggested change
state[1] > C.n - C.k + 1 && return
state[1] > C.n - C.k + 1 && return nothing

@lgoettgens
Copy link
Member

Tagging triage to decide about the return type

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request release notes: use title For PRs: the title of this PR is suitable for direct use in the release notes topic: combinatorics triage
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants