Make conj assoc dissoc and transient versions handle args similarly

Description

Problem

The function arities and behaviors between conj, assoc, disj, and dissoc are asymmetric to their transient versions. This makes their use as higher-order functions over transient and non-transient structs difficult.

Background

conj vs conj!

assoc vs assoc!

Additionally, assoc! silently uses nil for the last value if an unmatched key is provided, while assoc throws:

dissoc vs dissoc!

disj vs disj!

These are symmetric.

Approach

The argilst rarities should be aligned per the tables above and analogous behavior provided for each arity between transient and non-transient versions. This latter aspect will raise an error on mismatched k->v pairs and assoc! whereas no such error was thrown previously. Docstrings and tests should be updated to match the new signatures and behaviors.

Patch:

Clj-1103-10.patch

Screened by: Alex Miller

Environment

None

Attachments

4

Activity

Show:

Brandon Bloom August 23, 2023 at 3:41 AM

I don’t think “asymmetric” means anything formal here. A more rigorous explanation would be that the guidance on (quoted below) doesn’t generalize to all arities.

Note how the transient-using version has exactly the same structure, just:

- Calling transient on the source vector

- Using conj! instead of conj

- Calling persistent! on return

That is, if you’re using a higher arity of conj, you need to make multiple calls to conj! instead of simply adding the exclamation point.

The description and patches actually does several things at once:

  • Add the above arity generalization for variadic conj! and arity 1 (dissoc! m)

  • Add an arity-1 base case to assoc.

    • To me, this is the main feature I’m interested in of this patch. I’ve encountered bugs in the wild due to the expectation that (apply assoc m xs) is valid for an empty xs.

    • Makes the same generalization for assoc

  • Adapt the patch for assoc!

It might be easier to evaluate the assoc arity-1 patch independently, which basically boils down to: this:

Rich Hickey August 21, 2023 at 2:23 PM

“The function arities and behaviors between conj, assoc, disj, and dissoc are asymmetric to their transient versions.” What does that sentence mean?

Andy Fingerhut October 10, 2017 at 4:17 AM

No worries, and I thank you for keeping the patch current, even while keeping my name on it.

Michael Blume October 7, 2017 at 1:17 AM

That was not intentional at all, thanks for catching it, I'm honestly not sure what happened on my end to make that happen. I think this patch should be a correct update. Sorry about that.

Andy Fingerhut October 7, 2017 at 12:00 AM

Is it intentional that clj-1103-9.patch leaves out the changes to assoc! that were in the earlier patches? I can create another updated one if so – just curious if there was a reason for the change.

Details

Assignee

Reporter

Approval

Triaged

Patch

Code and Test

Priority

Created November 5, 2012 at 12:00 AM
Updated May 21, 2024 at 3:43 PM