<div dir="ltr"><span style="font-size:12.8px">> or, equivalently:</span><br style="font-size:12.8px">><br style="font-size:12.8px"><span style="font-size:12.8px">>        a `andthen` b = (>>) <$> a <*> b</span><br style="font-size:12.8px">><br style="font-size:12.8px"><span style="font-size:12.8px">> for which </span><a href="http://pointree.io/" rel="noreferrer" target="_blank" style="font-size:12.8px">http://pointree.io</a><span style="font-size:12.8px"> dutifully gives me:</span><br style="font-size:12.8px">><br style="font-size:12.8px"><span style="font-size:12.8px">>      andthen = (<*>) . ((>>) <$>)</span><div><br></div><div>That link is not working for me, and Google isn't finding it, and it sounds like a useful thing.<br style="font-size:12.8px"></div><div><br></div></div><div class="gmail_extra"><br><div class="gmail_quote">On Thu, Sep 28, 2017 at 1:55 AM, Viktor Dukhovni <span dir="ltr"><<a href="mailto:ietf-dane@dukhovni.org" target="_blank">ietf-dane@dukhovni.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><br>
When generating a report file from a database I found it much more<br>
efficient (significantly shorter runtime) to represent each row<br>
by an I/O action that prints the row, rather than to construct a<br>
Row object that to print and throw away.<br>
<br>
But the naive way to construct the I/O action can be tedious to<br>
maintain once the column count gets appreciably high:<br>
<br>
    newtype Foo = Foo { _foo :: IO () }<br>
    instance FromRow Foo where<br>
      fromRow = Foo <$> (rowPrinter <$> field <*> field <*> field <*> ... <*> field)<br>
        where<br>
          rowPrinter :: Type1 -> Type2 -> Type3 -> ... -> TypeN -> IO ()<br>
          rowPrinter p1 p2 p3 ... pN = do<br>
            printP1<br>
            printP2<br>
            printP3<br>
            ...<br>
            printPN<br>
<br>
So I decided to applicatively decompose the rowPrinter function<br>
(with the actual name of "andthen" to be determined later) as:<br>
<br>
   rowPrinter = (printP1 <$> field) `andthen`<br>
                (printP2 <$> field) `andthen`<br>
                (printP3 <$> field) `andthen`<br>
                ...<br>
                (printPN <$> field)<br>
<br>
which avoids the need to package the column printers explicitly into<br>
a single function, and may be somewhat more efficient a well.<br>
<br>
What was not immediately obvious to me was whether there's an "off the<br>
shelf" implementation of "andthen" I could just reuse. The necessary<br>
operator satisfies:<br>
<br>
        andthen (f (m a)) (f (m b)) = f (ma >> mb)<br>
<br>
or, equivalently:<br>
<br>
        a `andthen` b = (>>) <$> a <*> b<br>
<br>
for which <a href="http://pointree.io" rel="noreferrer" target="_blank">http://pointree.io</a> dutifully gives me:<br>
<br>
       andthen = (<*>) . ((>>) <$>)<br>
<br>
Its type signature is:<br>
<br>
    Prelude> :set prompt "l> "<br>
    l> :m + Control.Applicative<br>
    l> :m + Control.Monad<br>
    l> :t ((<*>) . ((>>) <$>))<br>
    ((<*>) . ((>>) <$>))<br>
      :: (Monad m, Applicative f) => f (m a) -> f (m b) -> f (m b)<br>
    l><br>
<br>
It seems to me that this would have been done before, and the operator<br>
would already be present in some package, but I'm having trouble finding<br>
it.<br>
<br>
(Due to the hidden constructors of FromRow defining a Semigroup does not<br>
 work out here, so I can't use (<>), which also inconveniently conflicts<br>
 with  Monoid (<>)).<br>
<br>
So my question is whether the operator in question is already available,<br>
under some name in some package, or else suggested names for it if new.<br>
<span class="HOEnZb"><font color="#888888"><br>
--<br>
        Viktor.<br>
<br>
______________________________<wbr>_________________<br>
Haskell-Cafe mailing list<br>
To (un)subscribe, modify options or view archives go to:<br>
<a href="http://mail.haskell.org/cgi-bin/mailman/listinfo/haskell-cafe" rel="noreferrer" target="_blank">http://mail.haskell.org/cgi-<wbr>bin/mailman/listinfo/haskell-<wbr>cafe</a><br>
Only members subscribed via the mailman list are allowed to post.</font></span></blockquote></div><br><br clear="all"><div><br></div>-- <br><div class="gmail_signature" data-smartmail="gmail_signature"><div dir="ltr"><div><div dir="ltr"><div><div dir="ltr"><div><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div dir="ltr"><div>Jeff Brown | Jeffrey Benjamin Brown</div><div dir="ltr"><a href="https://msu.edu/~brown202/" style="font-size:12.8px" target="_blank">Website</a>   |   <a href="https://www.facebook.com/mejeff.younotjeff" style="font-size:12.8px" target="_blank">Facebook</a>   |   <a href="https://www.linkedin.com/in/jeffreybenjaminbrown" style="font-size:12.8px" target="_blank">LinkedIn</a><span style="font-size:12.8px">(spammy, so I often miss messages here)   </span><span style="font-size:12.8px">|</span><span style="font-size:12.8px">   </span><a href="https://github.com/jeffreybenjaminbrown" style="font-size:12.8px" target="_blank">Github</a><span style="font-size:12.8px">   </span></div></div></div></div></div></div></div></div></div></div></div></div></div></div></div></div></div></div></div></div></div>
</div>