-
Notifications
You must be signed in to change notification settings - Fork 40
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
Enumerated NonEmpty #51
base: main
Are you sure you want to change the base?
Conversation
Although I think this PR is a great idea and illustrates that there are some parts missing in this library to fully support all of Swift's collection and sequence operators, it might not be the best solution to introduce such ad-hoc types like Imagine continuing with this solution and adding something like func zip<E1, E2>(_ nonEmpty1: NonEmpty<E1>, _ nonEmpty2: NonEmpty<E2>) -> NonEmpty<???>.Zip2Sequence<E1, E2> The first problem would be that the generic requirement of func zip<E1, E2>(_ nonEmpty1: NonEmpty<E1>, _ nonEmpty2: NonEmpty<E2>) -> NonEmptyZip2Sequence<E1, E2> But now we are running into another problem. We cannot zip It would make the most sense to me, to imitate Apple's way of handling this problem like they did with eg. You could then even go further and extend those protocols with default implementations like you did with An example of what I was thinking about (for simplicity reasons I chose sequence over collection here): protocol NonEmptySequenceProtocol: Sequence {
associatedtype Base: Sequence
var base: Base { get }
}
extension NonEmptySequenceProtocol {
@inlinable
var underestimatedCount: Int { return base.underestimatedCount }
}
extension NonEmptySequenceProtocol where Element == Base.Element {
@inlinable
func makeIterator() -> _NonEmptyIterator<Base.Iterator> { return _NonEmptyIterator(_base: base.makeIterator()) }
@inlinable
@warn_unqualified_access
func min(by areInIncreasingOrder: (Element, Element) throws -> Bool) rethrows -> Element { return try self.min(by: areInIncreasingOrder)! }
@inlinable
@warn_unqualified_access
func max(by areInIncreasingOrder: (Element, Element) throws -> Bool) rethrows -> Element { return try self.max(by: areInIncreasingOrder)! }
}
extension NonEmptySequenceProtocol where Element == Base.Element, Element: Comparable {
@inlinable
@warn_unqualified_access
func min() -> Element { return self.min()! }
@inlinable
@warn_unqualified_access
func max() -> Element { return self.max()! }
} Here the rather simple map type implementation: struct NonEmptyMapSequence<Base, Element> where Base: NonEmptySequenceProtocol {
@usableFromInline
var _base: Base
@usableFromInline
let _transform: (Base.Element) -> Element
@inlinable
init(_base: Base, transform: @escaping (Base.Element) -> Element) {
self._base = _base
self._transform = transform
}
}
extension NonEmptyMapSequence {
struct Iterator {
@usableFromInline
var _base: Base.Iterator
@usableFromInline
let _transform: (Base.Element) -> Element
@inlinable
init(
_base: Base.Iterator,
_transform: @escaping (Base.Element) -> Element
) {
self._base = _base
self._transform = _transform
}
}
}
extension NonEmptyMapSequence.Iterator: IteratorProtocol, Sequence {
@inlinable
mutating func next() -> Element? { return _base.next().map(_transform) }
}
extension NonEmptyMapSequence: Sequence {
@inlinable
func makeIterator() -> Iterator { return .init(_base: _base.makeIterator(), _transform: _transform) }
}
extension NonEmptyMapSequence: NonEmptySequenceProtocol {
@inlinable
var base: Base { _base }
} And finally the extension on the protocol itself: extension NonEmptySequenceProtocol {
@inlinable
func map<U>(_ transform: @escaping (Element) -> U) -> NonEmptyMapSequence<Self, U> { return .init(_base: self, transform: transform) }
} The downsides I am currently seeing is that some implementations would then be lazy instead of eager like it is currently implemented, but I don't know if this would an actual problem and there will be a lot of reimplementations of already existing Swift code... And to round this up and come back to the example I made previously, the zipping. You would need to implement a type like func zip<Sequence1, Sequence2>(_ sequence1: Sequence1, _ sequence2: Sequence2) -> NonEmptyZip2Sequence<Sequence1, Sequence2> where Sequence1: NonEmptySequenceProtocol, Sequence2: NonEmptySequenceProtocol If something like "protocolizing" I hope I did not miss anything here, so please correct me if I got something wrong. Thanks. |
Thanks for your great comment @ph1ps! I didn't have time yet to carefully read your code snippets, but I got the general idea of your proposal, and I fully agree with it. As you can see in the But before we go down the rabbit hole, I think it's worth exploring how this work can be broken down and linked together.
For this specific PR I think it'd be beneficial to go from nested |
I think I'd favor the 2nd approach. I would imagine abstracting collections as not very straight forward (that's why I did not include it in my code samples) and furthermore One thing I also just realized is, that |
Problem Statement
NonEmpty
provides a compile-time guarantee that a collection contains a value. This invariant is preserved even after collection is transformed withmap
/flatMap
/shuffled
functions, which is expected. One would expect that enumerating aNonEmpty
collection would also preserve this invariant, but it's not the case:In order to get an enumerated
NonEmpty
result one has to either introduce an outer index, which feels gross and error prone:or temporarily leave the
NonEmpty
context and force their way back:Motivation
Semantically, there seems to be no apparent reason for the nonemptiness invariant to be lost during the enumeration. Enumeration is just a case of
map
transformation that embellishes original collection with indexes, andmap
already preserves nonemptiness. Hence, enumeration should too.Implementation
Introduce a dedicated
Enumerated
type nested inNonEmpty
, echoing Swift'sEnumeratedSequence
. Implement all functions that preserve nonemptiness invariant:map
,flatMap
,shuffled
,first
, etc. ConformNonEmpty.Enumerated
toSequence
to match the functionality ofEnumeratedSequence
for backwards compatibility:NB
Similar points could be made for other derivative sequence types, such as
LazySequence
,Zip2Sequence
andReversedCollection
. For example, Haskell preserves nonemptiness when zipping.