<html><head><meta http-equiv="Content-Type" content="text/html charset=utf-8"></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><br class=""><div><blockquote type="cite" class=""><div class="">On Jun 30, 2016, at 5:32 PM, Dave Abrahams via swift-evolution &lt;<a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a>&gt; wrote:</div><br class="Apple-interchange-newline"><div class=""><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">on Thu Jun 30 2016, Xiaodi Wu &lt;</span><a href="http://xiaodi.wu-at-gmail.com/" style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class="">xiaodi.wu-AT-gmail.com</a><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">&gt; wrote:</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><blockquote type="cite" style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class="">If Iterators become reference types that model single-pass sequences and<br class="">becomes for-in-able, as the write-up suggests, couldn't Sequence be<br class="">stipulated to be multipass and retain its refinement relationship with<br class="">Collection?<br class=""></blockquote><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">AFAIK there is no interesting multipass Sequence that cannot reasonably be</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">made to support indexing.</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">There *is* existing code that exposes multipass data structures without</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">exposing the ability to compare iteration state for equality. &nbsp;</span></div></blockquote><div><br class=""></div><div>It’s worth noting that indices require comparability, not just equality. &nbsp;I think comparability might cause more difficulty than equality (but haven’t thought too hard about it).</div><br class=""><blockquote type="cite" class=""><div class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">In every</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">case I can think of, index equality could easily have been exposed, but</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">wasn't.These designs can't be adapted to model Collection.</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""></div></blockquote><div><br class=""></div><div>Why can’t they be adapted to model Collection if equality could have been exposed? &nbsp;Is it because comparability would be difficult?</div><br class=""><blockquote type="cite" class=""><div class=""><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">Those designs are real, but I am unconvinced they are worth supporting</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">directly with a separate protocol in the standard library; I'm willing</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">to accept the idea that those data structures will simply be limited to</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">modeling Iterator.</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""></div></blockquote><div><br class=""></div><div>Can you elaborate on what designs / data structures you’re talking about here?</div><br class=""><blockquote type="cite" class=""><div class=""><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><blockquote type="cite" style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class="">On Thu, Jun 30, 2016 at 12:26 Dave Abrahams via swift-evolution &lt;<br class=""><a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a>&gt; wrote:<br class=""><br class=""><blockquote type="cite" class=""><br class="">on Wed Jun 29 2016, Haravikk &lt;<a href="http://swift-evolution-at-haravikk.me" class="">swift-evolution-AT-haravikk.me</a>&gt; wrote:<br class=""><br class=""><blockquote type="cite" class=""><blockquote type="cite" class="">On 29 Jun 2016, at 00:10, Matthew Johnson via swift-evolution &lt;<br class=""></blockquote></blockquote><a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a>&gt; wrote:<br class=""><blockquote type="cite" class=""><blockquote type="cite" class=""><br class="">Swift is a language that embraces value semantics. &nbsp;Many common<br class="">iterators *can* be implemented with value semantics. &nbsp;Just because we<br class="">can’t implement *all* iterators with value semantics doesn’t mean we<br class="">should require them to have reference semantics. &nbsp;It just means you<br class="">can’t *assume* value semantics when working with iterators in generic<br class="">code unless / until we have a way to specify a value semantics<br class="">constraint. &nbsp;That’s not necessarily a bad thing especially when it<br class="">leaves the door open to interesting future possibilities.<br class=""><br class="">-Matthew<br class=""></blockquote><br class="">I'm kind of undecided about this personally. I think one of the<br class="">problems with Swift is that the only indication that you have a<br class="">reference type is that you can declare it as a constant, yet still<br class="">call mutating methods upon it, this isn't a very positive way of<br class="">identifying it however. This may be more of a GUI/IDE issue though, in<br class="">that something being a class isn't always that obvious at a glance.<br class=""><br class="">I wonder, could we somehow force iterators stored in variables to be<br class="">passed via inout? This would make it pretty clear that you're using<br class="">the same iterator and not a copy in all cases, encouraging you to<br class="">obtain another if you really do need to perform multiple passes.<br class=""></blockquote><br class="">I'm going to push single-pass iteration on the stack briefly and talk<br class="">about the topic that's been under discussion here: infinite multipass<br class="">sequences.<br class=""><br class="">## Fitting “Infinite Multipass” Into the Model<br class=""><br class="">It remains to be decided whether it's worth doing, but if it's to<br class="">happen, the standard library team thinks the right design is roughly<br class="">this:<br class=""><br class="">&nbsp;/// A multipass sequence that may be infinite<br class="">&nbsp;protocol Collection {<br class=""><br class="">&nbsp;&nbsp;&nbsp;// Only eager algorithms that can terminate available here<br class="">&nbsp;&nbsp;&nbsp;func index(where predicate: (Element)-&gt;Bool) -&gt; Index<br class=""><br class="">&nbsp;&nbsp;&nbsp;// all lazy algorithms available here<br class="">&nbsp;&nbsp;&nbsp;var lazy: ...<br class=""><br class="">&nbsp;&nbsp;&nbsp;var startIndex: Index<br class="">&nbsp;&nbsp;&nbsp;var endIndex: Index // possibly not reachable from startIndex<br class=""><br class="">&nbsp;&nbsp;&nbsp;associatedtype SubSequence : Collection<br class="">&nbsp;&nbsp;&nbsp;// do we need an associated FiniteSubsequence, e.g. for prefixes?<br class="">&nbsp;}<br class=""><br class="">&nbsp;protocol FiniteCollection : Collection {<br class=""><br class="">&nbsp;&nbsp;&nbsp;// All eager algorithms available here<br class="">&nbsp;&nbsp;&nbsp;func map(...) -&gt;<br class="">&nbsp;&nbsp;&nbsp;var count: ...<br class="">&nbsp;}<br class=""><br class="">&nbsp;protocol BidirectionalCollection : Collection { ... }<br class=""><br class="">&nbsp;protocol RandomAccessCollection : BidirectionalCollection { ... }<br class=""><br class="">Q: Why should there be indices on an infinite multipass sequence?<br class="">A: Because the operations on indices apply equally well whether the<br class="">&nbsp;&nbsp;sequence is finite or not. &nbsp;Find the index of a value in the<br class="">&nbsp;&nbsp;sequence, slice the sequence, find again, etc.<br class=""><br class="">Q: Why is there an endIndex on an infinite seque?<br class="">A: So you can write algorithms such as index(where:) once.<br class=""><br class="">Q: Why not allow endIndex to have a different type from startIndex?<br class="">A: It appears to offer insufficient benefit for the associated<br class="">&nbsp;&nbsp;complexity in typical usage. &nbsp;A classic use case that argues for a<br class="">&nbsp;&nbsp;different endIndex type is the null-terminated C string. &nbsp;But you<br class="">&nbsp;&nbsp;can't index one of those safely without actually counting the length,<br class="">&nbsp;&nbsp;and once you've done that you can make the endIndex an Int.<br class=""><br class="">## Single Pass Iteration<br class=""><br class="">The refinement relationship between Sequence and Collection is<br class="">problematic, because it means either:<br class=""><br class="">a) algorithms such as map on single-pass sequences claim to be<br class="">&nbsp;&nbsp;nonmutating even though it's a lie (status quo)<br class=""><br class="">b) those algorithms can't be used on immutable (“let bound”) multipass<br class="">&nbsp;&nbsp;sequences. IMO that would be totally unacceptable.<br class=""><br class="">If we drop the refinement, we can have a saner world. &nbsp;We also don't<br class="">need to separate Sequence and Iterator anymore. &nbsp;We can simply drop<br class="">Sequence altogether, and the protocol for single-pass iteration becomes<br class="">Iterator.<br class=""><br class="">### Mutation and Reference Semantics<br class=""><br class="">Everything in Swift is copiable via `let copy = thing` (let's please not<br class="">argue over the definition of copy for classes; this is the one built<br class="">into the lowest level of the language—I refer to the other one, that<br class="">requires allocation, as “clone”).<br class=""><br class="">Anything you do with a sequence that's truly single-pass mutates the<br class="">sequence *and of its copies*. &nbsp;Therefore, such a type *fundamentally*<br class="">has reference semantics. One day we may be able to model single-pass<br class="">sequences with “move-only” value types, which cannot be copied. You can<br class="">find move-only types in languages like Rust and C++, but they are not<br class="">supported by Swift today. &nbsp;So it seems reasonable that all Iterators in<br class="">Swift today should be modeled as classes.<br class=""><br class="">The fact that Swift doesn't have a mutation model for classes, though,<br class="">means that mutating methods on a class constrained protocol can't be<br class="">labeled as such. &nbsp;So consuming operations on a class-constrained<br class="">Iterator protocol would not be labeled as mutating.<br class=""><br class="">The standard library team is currently trying to evaluate the tradeoffs<br class="">in this area. &nbsp;One possibility under consideration is simply dropping<br class="">support for single-pass sequences until Swift can support move-only<br class="">value types and/or gets a mutation model for class instances. &nbsp;It would<br class="">be very interesting to know about any real-world models of single-pass<br class="">sequences that people are using in Swift, since we don't supply any in<br class="">the standard library.<br class=""><br class="">--<br class="">Dave<br class="">_______________________________________________<br class="">swift-evolution mailing list<br class=""><a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a><br class="">https://lists.swift.org/mailman/listinfo/swift-evolution<br class=""><br class=""></blockquote></blockquote><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">--<span class="Apple-converted-space">&nbsp;</span></span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">Dave</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">_______________________________________________</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">swift-evolution mailing list</span><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><a href="mailto:swift-evolution@swift.org" style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class="">swift-evolution@swift.org</a><br style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><a href="https://lists.swift.org/mailman/listinfo/swift-evolution" style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; widows: auto; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class="">https://lists.swift.org/mailman/listinfo/swift-evolution</a></div></blockquote></div><br class=""></body></html>