Boost logo

Boost :

Subject: Re: [boost] [Optional] Monadic bind
From: Tobias Loew (Tobias.Loew_at_[hidden])
Date: 2015-06-22 10:59:57


I deliberately used this std::bind syntax since it accepts any callable (including things like pointer-to-member [functions]).
I tried lambdas with my code and they just work. E.g.

    auto opt_i3 = map(opt_a, [](const A& a){ return a.i;});
    assert(*opt_i3 == 42);

Furthermore without the std::decay boost::optional<decltype(f(*o))> will be a boost::optional<T const&>.

Tobias

Von: Andrzej Krzemienski [via Boost] [mailto:ml-node+s2283326n4677436h22_at_[hidden]]
Gesendet: Montag, 22. Juni 2015 16:44
An: Löw, Tobias (STEAG Energy Services GmbH)
Betreff: Re: [Optional] Monadic bind

2015-06-22 16:17 GMT+02:00 Tobias Loew <[hidden email]</user/SendEmail.jtp?type=node&node=4677436&i=0>>:

> Sorry for that,
>
> I've made in implementation with free functions which compiles on VS 2012:
>
>
> #include <assert.h>
> #include <boost/optional.hpp>
>
>
> template<class T, class B>
> auto bind(const boost::optional<T>& optional, B&& binder)
> -> typename std::decay<decltype(std::bind(binder, *optional)())>::type
> {
> if(optional)
> {
> return std::bind(binder, *optional)();
> }
> else
> {
> return boost::none;
> }
> }
>
> template<class T, class B>
> auto map(const boost::optional<T>& optional, B&& binder)
> -> boost::optional<typename std::decay<decltype(std::bind(binder,
> *optional)())>::type>
> {
> if(optional)
> {
> return std::bind(binder, *optional)();
> }
> else
> {
> return boost::none;
> }
> }
>
>
> void foo()
> {
> struct A {
> int i;
> };
> boost::optional<A> opt_a;
>
> auto opt_i = map(opt_a, &A::i);
> assert(!opt_i);
>
> A a = {42};
> opt_a = a;
>
> auto opt_i2 = map(opt_a, &A::i);
> assert(*opt_i2 == 42);
>
> }
>

Would you find the following an acceptable solution?

```
#include <assert.h>
#include <boost/bind.hpp>
#include <boost/optional.hpp>

template <typename T, typename F>
auto map2(const boost::optional<T>& o, F f)
  -> boost::optional<decltype(f(*o))>
{
  if (o) return f(*o);
  else return boost::none;
}

void foo()
{
    struct A { int i; };
    boost::optional<A> opt_a;

    auto opt_i = map2(opt_a, boost::bind(&A::i, _1));
    assert(!opt_i);

    A a = {42};
    opt_a = a;

    auto opt_i2 = map2(opt_a, boost::bind(&A::i, _1));
    assert(opt_i2);
    assert(*opt_i2 == 42);
}

int main()
{
    foo();
}
```

It requires of you to type a bit more, but at the same time it is more
flexible: you can type any free function on T, including a lambda.

Regards,
&rzej

_______________________________________________
Unsubscribe & other changes: http://lists.boost.org/mailman/listinfo.cgi/boost

________________________________
If you reply to this email, your message will be added to the discussion below:
http://boost.2283326.n4.nabble.com/Optional-Monadic-bind-tp4677431p4677436.html
To unsubscribe from [Optional] Monadic bind, click here<http://boost.2283326.n4.nabble.com/template/NamlServlet.jtp?macro=unsubscribe_by_code&node=4677431&code=VG9iaWFzLkxvZXdAc3RlYWcuY29tfDQ2Nzc0MzF8MTI1MDAzMDE5Mg==>.
NAML<http://boost.2283326.n4.nabble.com/template/NamlServlet.jtp?macro=macro_viewer&id=instant_html%21nabble%3Aemail.naml&base=nabble.naml.namespaces.BasicNamespace-nabble.view.web.template.NabbleNamespace-nabble.view.web.template.NodeNamespace&breadcrumbs=notify_subscribers%21nabble%3Aemail.naml-instant_emails%21nabble%3Aemail.naml-send_instant_email%21nabble%3Aemail.naml>

--
View this message in context: http://boost.2283326.n4.nabble.com/Optional-Monadic-bind-tp4677431p4677437.html
Sent from the Boost - Dev mailing list archive at Nabble.com.

Boost list run by bdawes at acm.org, gregod at cs.rpi.edu, cpdaniel at pacbell.net, john at johnmaddock.co.uk