On Feb 15, 2009, at 8:22 PM, Mark Engelberg wrote:

>
> My thoughts so far:
>
>
> 4.  The new model is definitely more complicated to understand than
> the previous model.  There was already a certain degree of mental
> overlap between collections and the seq interface.  Now, there is also
> the subtle distinction between a seq and a sequence.

There will need to be good descriptions of these, but the similarity  
is more in names than anything else - seqs are what they always were -  
cursors, and sequences are just collections.

> rest and next
> are very similar, but one can return something empty, and one can
> return nil.  Making the right choice, and interfacing with other code
> is now a bit more complicated (although people can always call seq to
> convert it into the seq/nil paradigm with certainty, which is not much
> different than before).


Code that returns sequences should use rest. next is just a  
convenience function for terminal/consumer code. If you look through  
the ported core code, most recurs use next, most conses inside lazy- 
seqs use rest. But as you noted, if you want to ignore next, that's  
fine.

> I think the additional complexity is worth it
> to solve things like the filter problem, but I think it's definitely
> more confusing than before.
>
> 5.  At first glance, it seems like sticking with the original
> lazy-cons model, but removing nil punning and adding an empty sequence
> sentinel, along with your compiler enhancement, would accomplish
> everything the new "lazier" branch accomplishes with much less mental
> complexity and subtle overlap, and resulting in the most intuitive
> version of filter working as expected.

I realize you are focused on filter, but that point of the fully lazy  
branch is full laziness, which would not fall out of what you  
describe. lazy-cons requires the lazy sequence function do all the  
work that precedes the call to lazy-cons, making functions like drop  
and filter not fully lazy.

> I know you like the seq/nil
> model, and the nil punning, but since you're already moving in the
> direction of reducing reliance on this approach, I hope you've
> considered going "all the way", to see if it would solve the problem
> more elegantly.  If you have considered this already, I'd be curious
> to know whether it didn't solve the problem, or whether it just
> resulted in too much breakage with existing code, or whether it's just
> a style you don't like as a matter of taste...

I'm not sure what you mean by "all the way". If you mean removing seqs  
entirely, here are some issues:

(seq x) acts as a single interface to the sequence system, if it were  
to return a seq[uence] collection instead of seq/nil, then all  
sequence function bodies would have to return () explicitly, adding  
another branch to all implementations that could otherwise use when:

(defn map [f coll]
   (lazy-seq
    (if (empty? coll)
      ()
      (cons (f (first coll)) (map f (rest coll))))))

instead of:

(defn map [f coll]
   (lazy-seq
    (when (seq coll)
      (cons (f (first coll)) (map f (rest coll))))))

But the actual definition in the lazy branch pulls the seq out of the  
lazy sequence like so:

(defn map [f coll]
   (lazy-seq
    (when-let [s (seq coll)]
      (cons (f (first s)) (map f (rest s))))))

Doing so yields a significant (> 60%) speed improvement, without it,  
all lazy calls (empty?/first/rest) have a double indirection, and no  
way to get rid of it.

Rich


--~--~---------~--~----~------------~-------~--~----~
You received this message because you are subscribed to the Google Groups 
"Clojure" group.
To post to this group, send email to clojure@googlegroups.com
To unsubscribe from this group, send email to 
clojure+unsubscr...@googlegroups.com
For more options, visit this group at 
http://groups.google.com/group/clojure?hl=en
-~----------~----~----~----~------~----~------~--~---

Reply via email to