1+++ 2title = "Home" 3+++ 4 5# Outcome 2.1 library 6 7{{% boost-copyright %}} 8 9Outcome is a set of tools for reporting and handling function failures in contexts where *directly* using C++ exception handling is unsuitable. Such contexts include: 10 11 - there are programs, or parts thereof, that are compiled with exceptions disabled; 12 13 - there are parts of program that have a lot of branches depending on types of failures, 14 where if-statements are cleaner than try-catch blocks; 15 16 - there is a hard requirement that the failure path of execution should not cost more than the successful path of execution; 17 18 - there are situations, like in the [`filesystem`](http://www.boost.org/doc/libs/release/libs/filesystem/doc/index.htm) library, where whether the failure should be handled remotely 19 (using a C++ exception throw), or locally cannot be made inside the function and needs to be decided by the caller, 20 and in the latter case throwing a C++ exception is not desirable for the aforementioned reasons; 21 22 - there are parts of the program/framework that themselves implement exception handling and prefer 23 to not use exceptions to propagate failure reports across threads, tasks, fibers, etc; 24 25 - one needs to propagate exceptions through layers that do not implement exception throw safety; 26 27 - there is an external requirement (such as a company-wide policy) that failure handling paths are explicitly indicated in the code. 28 29 - where interoperation with C code, without having to resort to C++ exception wrapper shims, is important. 30 31Outcome addresses failure handling through returning a special type from functions, which is able to store either a successfully computed value (or `void`), or the information about failure. Outcome also comes with a set of idioms for dealing with such types. 32 33Particular care has been taken to ensure that Outcome has the lowest possible impact on build times, 34thus making it suitable for use in the global headers of really large codebases. Storage layout is 35guaranteed and is C-compatible for `result<T, E>`[^1], thus making Outcome based code long term ABI-stable. 36 37{{% notice note %}} 38Ben Craig's work on [P1886 *Error speed benchmarking*](https://wg21.link/P1886) has led to 39a [`better_optimisation`](https://github.com/ned14/outcome/tree/better_optimisation) branch intended 40to be merged end of 2020 as Outcome v2.2.0, after twelve months of testing. This branch has a number 41of major and breaking changes to Outcome v2: 42 431. A new trait `is_move_bitcopying<T>` is added, which opts types into a library-based emulation of 44[P1029 *move = bitcopies*](https://wg21.link/P1029). [Experimental `std::error`](https://wg21.link/P1028) is opted in by default. 45If this trait is true for your `T` or `E` type, Outcome will track moved-from status for your type, 46and will only call your type's destructor if it was not moved from. If your compiler's optimiser is 47sufficiently able to fold code, this improves codegen quality for Experimental Outcome very considerably, 48approaching the same gains as P1029 types would have. Note that the empirical performance difference 49will likely be nil, but the codegen does look much more elegant. 50 512. If for `basic_result<T, E>` both `T` and `E` are trivially copyable, union-based rather than 52struct-based storage will be used. This significantly improves performance in synthetic benchmarks 53which do nothing in deep call stacks of function calls except create and return `result<T, E>`, and 54makes Outcome return competitive results to alternative error handling choices, improving comparative 55optics. It is not expected that the performance difference will be detectable empirically in real 56world code. It is expected that the build time impact of union storage won't be noticeable, as 57union storage for trivially copyable types is much easier than for non-TC types. 58 593. The compile time requirement for `E` types to have a default constructor is removed. 60 614. `BOOST_OUTCOME_TRY(var, expr)` no longer always declares `var` as `auto &&var`, but simply uses it 62as is. This allows `TRY` to initialise or assign. You can use the macro `OUTCOME21_TRY` if you 63want the pre-Outcome v2.2 behaviour. You may find the regular expression `_TRY\(([^(]*?),(.*?)\);` => 64`_TRY(auto &&\1,\2);` of use to you when upgrading code. 65{{% /notice %}} 66 67## Sample usage 68 69The main workhorse in the Outcome library is `result<T>`: it represents either a successfully computed value of type `T`, or a `std::error_code`/`boost::system::error_code`[^2] representing the reason for failure. You use it in the function's return type: 70 71{{% snippet "intro_example.cpp" "signature" %}} 72 73It is possible to inspect the state manually: 74 75{{% snippet "intro_example.cpp" "inspect" %}} 76 77Or, if this function is called in another function that also returns `result<T>`, you can use a dedicated control statement: 78 79{{% snippet "intro_example.cpp" "implementation" %}} 80 81`BOOST_OUTCOME_TRY` is a control statement. If the returned `result<T>` object contains an error information, the enclosing function is immediately returned with `result<U>` containing the same failure information; otherwise an automatic object of type `T` 82is available in scope. 83 84{{% notice note %}} 85This library joined [the Boost C++ libraries](https://www.boost.org/doc/libs/develop/libs/outcome/doc/html/index.html) in the 1.70 release (Spring 2019). [It can be grafted into much older Boost releases if desired](https://github.com/boostorg/outcome). 86{{% /notice %}} 87 88[^1]: If you choose a C-compatible `T` and `E` type. 89 90[^2]: `result<T>` defaults to `std::error_code` for Standalone Outcome, and to `boost::system::error_code` for Boost.Outcome. You can mandate a choice using `std_result<T>` or `boost_result<T>`. 91