Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/mpusz/opt
Class template designed to express optionality without having to sacrifice memory to store additional bool flag
https://github.com/mpusz/opt
Last synced: about 2 months ago
JSON representation
Class template designed to express optionality without having to sacrifice memory to store additional bool flag
- Host: GitHub
- URL: https://github.com/mpusz/opt
- Owner: mpusz
- License: mit
- Created: 2017-04-03T19:04:01.000Z (over 7 years ago)
- Default Branch: master
- Last Pushed: 2023-09-19T08:23:44.000Z (about 1 year ago)
- Last Synced: 2024-10-04T16:27:43.921Z (2 months ago)
- Language: C++
- Homepage:
- Size: 63.5 KB
- Stars: 12
- Watchers: 6
- Forks: 1
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- AwesomeCppGameDev - opt
README
[![GitHub license](https://img.shields.io/badge/license-MIT-blue.svg?maxAge=3600)](https://raw.githubusercontent.com/mpusz/opt/master/LICENSE)
[![Travis CI](https://img.shields.io/travis/mpusz/opt/master.svg?label=Travis%20CI)](https://travis-ci.org/mpusz/opt)
[![AppVeyor](https://img.shields.io/appveyor/ci/mpusz/opt/master.svg?label=AppVeyor)](https://ci.appveyor.com/project/mpusz/opt)
[![Codecov](https://img.shields.io/codecov/c/github/mpusz/opt/master.svg)](https://codecov.io/github/mpusz/opt?branch=master)# `mp::opt`
`mp::opt` is a class template designed to express optionality.
It has interface similar to `std::optional` ([see here](http://en.cppreference.com/w/cpp/utility/optional))
but it does not use additional storage space for boolean flag needed to keep information if
the value is set or not. On the other hand type `T` should be a type that does not use all
values from its underlying representation thus leaving a possibly to use one of them as a _Null_
value that express emptiness of type `T`.## Dependencies
`mp::opt` library depends on C++14 constexpr features and `std::optional` existence so
a fairly modern compiler is needed.## Installation
`mp::opt` is a header-only library. Its implementation was split to:
- `opt.h` that contains the most important implementation and is intended to be included by the user with
`#include ` in the code
- `opt_bits.h` that contains less important implementation details and is already included by `opt.h` header file## Usage
Here is a simple example of how to use `mp::opt` for some artificial type `price`:
```cpp
using namespace mp;
using price = int;
using opt_price = opt>;
static_assert(sizeof(opt_price) == sizeof(int));opt_price o1;
assert(static_cast(o1) == false);
assert(o1.has_value() == false);
assert(o1.value_or(123) == 123);constexpr opt_price o2{99};
assert(static_cast(o2) == true);
assert(o2.has_value() == true);
assert(*o2 == 99);
assert(o2.value() == 99);
assert(o2.value_or(123) == 99);opt_price o3{o2};
assert(o3.has_value() == true);
assert(*o3 == 99);o3 = o1;
assert(o3.has_value() == false);o3 = o2;
assert(o3.has_value() == true);
assert(*o3 == 99);o3 = nullopt;
assert(o3.has_value() == false);
```### `Policy` basic interface
`mp::opt` uses `Policy` class to provide all necessary information needed for proper class operation. The
simplest interface of `Policy` class for type `my_type` that uses `my_type_null_value` as _Null_ value contains
only one member function:
```cpp
struct my_opt_policy {
static constexpr my_type null_value() noexcept { return my_type_null_value; }
};
mp::opt my_opt_val;
```
- `null_value()` member function returns (by value or reference) special value of type `my_type` that should be
used by `mp::opt` as a special _Null_ value to mark emptiness.`mp::opt` by default uses `mp::opt_default_policy` that can be specialized for user's type `T` so above
example may be refactored in the following way:
```cpp
namespace mp {
template<>
struct opt_default_policy {
static constexpr my_type null_value() noexcept { return my_type_null_value; }
};
}
mp::opt my_opt_val;
```The latter solution is suggested for user's types that will always have the same special value used as _Null_ while
the former should be used for builtin types like 'int' where the type can express different quantities thus having
different _Null_ special values (e.g. age, price, month, etc).### `Policy` types provided with the library
Beside already mentioned `mp::opt_default_policy` that can be specialized by the user for his/her type `my_type`,
the library provides 2 additional policy types:
- `mp::opt_null_value_policy`
may be used to provide _Null_ value for integral type right in the class type definition. For example:
```cpp
mp::opt> opt_int;
```- `mp::opt_null_type_policy` may be used for types which values cannot be used as template argument
(e.g. `float`). That policy type assumes that `NullType::null_value` will represent _Null_ value. For example:
```cpp
struct my_null_float { static constexpr float null_value = 0.0f; };
mp::opt> opt_float;
```### What if `my_type` does not provide equality comparison?
`mp::opt` needs to compare contained value with special _Null_ value provided by the _Policy_ type. By default
it tries to use `operator==()`. If the equality comparison is not provided for `my_type` than either:
- `operator==()` should be provided, or
- additional `has_value(const T& value)` member function should be provided in the `Policy` type and it should return
`false` if `value == null_value()` or `true` otherwise:
```cpp
struct my_opt_policy {
static constexpr my_type null_value() noexcept { return my_type_null_value; }
static constexpr bool has_value(const T& value) noexcept { return value != null_value(); }
};
mp::opt my_opt_val;
```### What if `my_type` prevents me from setting value out of allowed range?
Sometimes `my_type` has some non-trivial logic or validation in its contructors or assignment operators that prevents
one to set special out-of-range value as _Null_ value for the type. In such case it is possible to extend `Policy`
class with following additional `storage_type` public member type. Such `storage_type` type:
- should be of the same size as `my_type` type
- should be constructible and assignable in the same way as `my_type`
- when storing not _Null_ value should have exactly the same memory representation as if the value was stored by
`my_type` type
- should be used in `null_value()` and `has_value()` member functions instead of `my_type`.Below is an example of how `opt` could be implemented:
```cpp
namepace mp {
template<>
struct opt_default_policy {
private:
union storage {
bool value;
std::int8_t null_value = -1;
storage() = default;
constexpr storage(bool v) noexcept : value{v} {}
};
public:
using storage_type = storage;
static constexpr storage_type null_value() noexcept { return storage_type{}; }
static constexpr bool has_value(storage_type s) noexcept { return s.null_value != -1; }
};
}
```Here is another a bit more complicated example for some (not too smart ;-) ) `weekday` class:
```cpp
class weekday {
public:
using underlying_type = std::int8_t;
constexpr explicit weekday(underlying_type v) : value_{v}
{
if(v < 0 || v > 6) throw std::out_of_range{"weekday value outside of allowed range"};
}
constexpr weekday& operator=(underlying_type v)
{
if(v < 0 || v > 6) throw std::out_of_range{"weekday value outside of allowed range"};
value_ = v;
return *this;
}
constexpr underlying_type get() const noexcept { return value_; }
private:
underlying_type value_; // 0 - 6
};
constexpr bool operator==(weekday lhs, weekday rhs) noexcept { return lhs.get() == rhs.get(); }
constexpr bool operator==(weekday::underlying_type lhs, weekday rhs) noexcept { return lhs == rhs.get(); }
constexpr bool operator==(weekday lhs, weekday::underlying_type rhs) noexcept { return lhs.get() == rhs; }namespace mp {
template<>
struct opt_default_policy {
union storage_type {
weekday value;
weekday::underlying_type null_value = std::numeric_limits::max();
constexpr storage_type() noexcept {};
constexpr storage_type(weekday v) : value{v} {}
constexpr storage_type(weekday::underlying_type v) : value{v} {}
storage_type& operator=(weekday v)
{
value = v;
return *this;
}
};
public:
static storage_type null_value() noexcept { return storage_type{}; }
static bool has_value(storage_type value) noexcept
{
return value.null_value != std::numeric_limits::max();
}
};
}
```