|
Boost : |
Subject: [boost] [next gen futures] Lightweight monad ready for inspection
From: Niall Douglas (s_sourceforge_at_[hidden])
Date: 2015-06-19 13:03:59
Some may remember the thread starting from
http://boost.2283326.n4.nabble.com/next-gen-future-promise-What-to-cal
l-the-monadic-return-type-td4676039.html and that I would firstly
prepare an optimally lightweight monad<T> for review here before
going on to base a next-gen lightweight future-promise on that
monad<T>.
I didn't expect it would take nearly four weeks to get there, but
writing STL quality C++ with full conformance test suite is amazingly
labourious. Anyway you can now view the design and tell me what you
think. A quick overview of the design:
This monad can hold a fixed variant list of empty, a type R, a
lightweight error_type or a heavier exception_type at a space cost of
max(20, sizeof(R)+4). Features:
* Very lightweight on build times and run times up to the point of
zero execution cost and just a four byte space overhead. Requires min
clang 3.2, GCC 4.7 or VS2015. A quick sample of runtime overhead, min
to max opcodes generated by GCC 5.1:
1 opcodes <= Value transport <= 113 opcodes
8 opcodes <= Error transport <= 119 opcodes
22 opcodes <= Exception transport <= 214 opcodes
4 opcodes <= then() <= 154 opcodes
5 opcodes <= bind() <= 44 opcodes
* Just enough monad, nothing more, nothing fancy. Replicates the
future API with a fair chunk of the Expected<T> API, so if you know
how to use a future you already know how to use this.
* Enables convenient all-noexcept mathematically verifiable close
semantic design, so why bother with Rust anymore? :)
* Can replace most uses of optional<T>.
* Deep integration with lightweight future-promise (i.e. async
monadic programming) also in this library.
* Comprehensive unit testing and validation suite.
* Mirrors noexcept of type R.
* Type R can have no default constructor, move nor copy.
* Works inside a STL container, and type R can be a STL container.
* No comparison operations nor hashing is provided, deliberately to
keep things simple.
Documentation page:
https://ci.nedprod.com/view/Boost%20Thread-Expected-Permit/job/Boost.S
pinlock%20Test%20Linux%20GCC%204.8/doxygen/classboost_1_1spinlock_1_1l
ightweight__futures_1_1monad.html
Source code:
https://github.com/ned14/boost.spinlock/blob/master/include/boost/spin
lock/monad.hpp
Online wandbox compiler:
http://melpon.org/wandbox/permlink/cnZM5KRNpjErXrPH
Any opinions or thoughts gratefully received, particularly if I have
the exception safety semantics right (i.e. a throw during move or
copy leaves the monad empty). Do you also like the polymorphic bind()
and map() which does different things depending on the parameter type
your callable takes? Do you like that by simply changing the
callable's type to a rvalue reference you can move state, or is this
being too clever?
This monad will become the basis of lightweight future-promise which
is essentially a "split monad" with the setter interface and getter
interface potentially in different system threads. The then(), bind()
and map() work as here but only are triggered at the point of the
value being changed. This effectively makes the lightweight future a
"lazy continued monad".
That lightweight future-promise will then enter AFIO to replace its
async_io_op type hopefully in time for the peer review this time next
month.
Niall
-- ned Productions Limited Consulting http://www.nedproductions.biz/ http://ie.linkedin.com/in/nialldouglas/
Boost list run by bdawes at acm.org, gregod at cs.rpi.edu, cpdaniel at pacbell.net, john at johnmaddock.co.uk