Skip to content

Maybe#

A container that optionally holds a value.

See Option and Maybe.

A Maybe has two possible states, Just or Nothing. Just means a value is present, and Nothing means the value is absent. The reason that we don't use Some-None or Nil nomencalture is to avoid confusion with the built-in None.

This module also provides a some constructor, which converts an Optional[T] value to a Maybe[T] value.

Rationale#

Python has a built-in None object that represents the absence of a value, and a corresponding type hint Optional[T]. However, an Optional[T] is semantically different from a Maybe[T] object. An Optional[T] is a union of T and None, not a single object. To use a union, you must explicitly check its type before every use.

# Usage 1
if optional is None:
    ...
else:
    do_something(optional)

# Usage 2
do_something(optional) if optional is not None else ...

# Usage 3
optional is not None and do_something(optional)

On the other hand, a Maybe[T] object represents the absence of a value as a state. Operations defined on Maybe[T] objects behave differently depending on the state of the object, but they are always available regardless of the state.

j: Maybe[int] = just(42)
j.map(do_something) #(1)!

n: Maybe[int] = nothing()
n.map(do_something) #(2)!
  1. do_something is called because j is Just.
  2. do_something is not called because n is Nothing.

Warning

Notice that None is not equal to Nothing. Use Maybe.is_nothing() to check if a Maybe object is Nothing.

nothing = Maybe[int].Nothing()

assert nothing.is_nothing()
assert nothing is not None
assert nothing != None

Implementation#

Maybe implements the following interfaces:

Interface Methods
Functor map
Applicative pure, apply
Monad bind(and_then)

The Maybe class is a generic class that holds a value of type T inside _val field. A boolean flag _has_value is used to indicate whether the value is present or absent. If _has_value is False, reading from _val is an undefined behavior.

Maybe and its methods do not support inherit-based subclassing.

Maybe's APIs are based on the Rust Option, and the comparison table is provided below.

Reference Option Counterpart
and and_
and_then
as_deref
as_deref_mut
as_mut
as_mut_slice
as_pin_mut
as_pin_ref
as_ref
as_slice
cloned
copied
expect
filter
flatten
get_or_insert
get_or_insert_default
get_or_insert_with
insert
inspect tap
is_none is_nothing
is_some is_just
is_some_and is_just_and
iter
iter_mut
map
map_or
map_or_else
ok_or
ok_or_else
or or_
or_else
replace
take
take_if
transpose
unwrap
unwrap_or
unwrap_or_default
unwrap_or_else
unwrap_unchecked
unzip
xor
zip
zip_with

just #

Constructs a Just value.

j: Maybe[int] = just(42)
j = just[int](42)

assert j.is_just()
assert j.unwrap() == 42

Warning

Calling just(None) will return a Just(None) value, not a Nothing value. If you want to map None to Nothing, use some instead.

nothing #

Constructs a Nothing value. Notice that this is not a literal.

n: Maybe[int] = nothing()
n = nothing[int]()

assert n.is_nothing()

some #

Converts an Optional[T] to a Maybe[T] value.

something: Maybe[int] = some(42)
assert something.is_just()
assert something.unwrap() == 42

something = some[int](None)
assert something.is_nothing()

Maybe #

A container that optionally holds a value. See module-level documentation for more information.

__and__ = and_ class-attribute instance-attribute #

Alias of and_.

__bool__() #

Equivalent to is_just.

__eq__(other) #

If both values are Just, compare the inner values. Otherwise, return True if both are Nothing.

Warning

Notice that None is not equal to Nothing.

__len__() #

Return 1 if the value is Just, otherwise 0.

__or__ = or_ class-attribute instance-attribute #

Alias of or_.

and_(other) #

If the value is Just, return a shallow copy of the other Maybe. Otherwise, return a new Nothing of type Maybe[U].

j1 = just[int](42)
j2 = just[bool](True)

assert j1.and_(j2).unwrap() == True
assert (j1 & j2).unwrap() == True
assert (j1 & nothing[bool]()).is_nothing()
assert (nothing[int]() & j2).is_nothing()

and_then(f) #

If the value is Just, apply a function that maps the inner value to a Maybe[U] value. Otherwise, return Nothing of type Maybe[U].

j = just[int](114514)

assert j.and_then(lambda x: some(x + 1805296)).unwrap() == 1919810

apply(f) #

Implementation of Applicative.apply. Applies the callable wrapped within a Maybe to the inner value, if both are Just.

bind(f) #

Implementation of Monad.bind. Alias of and_then.

duplicate(m) classmethod #

Create a shallow copy of a Maybe value.

expect(message) #

Unwrap the inner value, if any. Otherwise, raise a ValueError with a custom message.

Raises:

Type Description
ValueError

If the value is a Nothing.

filter(p) #

If the value is Just and satisfies the predicate, return the value. Otherwise, return Nothing.

j = just[int](42)
assert j.filter(lambda x: x > 0).unwrap() == 42
assert j.filter(lambda x: x < 0).is_nothing()

flatten() #

flatten() -> Maybe[T]
flatten() -> Maybe[T]

Flatten a nested Maybe value for one level. Unlike Option::flatten, this method does not require the inner value to be a Maybe. If it's not a nested Maybe, this is a no-op.

j = just[Maybe[int]](just(42))
assert j.flatten() == just(42)

j2 = just[Maybe[Maybe[int]]](just(just(42)))
assert j2.flatten() == just(just(42))
assert j2.flatten().flatten() == just(42)

get_or_insert(val) #

Get the inner value, if any. Otherwise, insert the new value and return the value.

j = just[int](42)
assert j.get_or_insert(114514) == 42

n = nothing[int]()
assert n.get_or_insert(114514) == 114514

get_or_insert_with(d) #

Get the inner value, if any. Otherwise, call a function to get a new value and return the value.

j = just[int](42)
assert j.get_or_insert_with(lambda: 114514) == 42

n = nothing[int]()
assert n.get_or_insert_with(lambda: 114514) == 114514

insert(val) #

Insert a value and returns it.

is_just() #

Check if the value is a Just.

is_just_and(p) #

Check if the value is a Just and satisfies the predicate.

is_nothing() #

Check if the value is a Nothing.

Note

A Just(Nothing) value of type Maybe[Maybe[T]] or a Just(None) value of type Maybe[Optional[T]] are not Nothings.

just(val) classmethod #

Construct a Just value.

Warning

Prefer using just instead, unless in performance-critical code.

map(f) #

Apply a function that maps the inner value to a new value, if any. Otherwise, return Nothing.

See Option::map.

j = just[int](42)
assert j.map(lambda x: x + 1) == just(43)

n = nothing[int]()
assert n.map(lambda x: x + 1) == nothing()

map_or(default, f) #

Map the inner value using a function, or use the default value if absent.

Tip

If the default value is an expensive expression, use map_or_else instead.

```python j = justint n = nothingint

assert j.map_or(0, lambda x: x + 1) == 43 assert n.map_or(0, lambda x: x + 1) == 0 ```z

map_or_else(d, f) #

Map the inner value using a function, or use a lazy default value if absent.

j = just[int](42)
n = nothing[int]()

assert j.map_or_else(lambda: 0, lambda x: x + 1) == 43
assert n.map_or_else(lambda: 0, lambda x: x + 1) == 0

nothing() classmethod #

Construct a Nothing value.

Warning

Prefer using Nothing instead, unless in performance-critical code.

or_(other) #

If the value is Just, return a shallow copy of itself. Otherwise, return the other Maybe's shallow copy.

Notice that the right-hand side should be a Maybe object with the same inner type, although this is not enforced at runtime. And also this method does not short-circuit.

j1 = just[int](42)
j2 = just[int](114514)

assert (j1 | j2).unwrap() == 42
assert (nothing[int]() | j2).unwrap() == 114514
assert (j1 | nothing[int]()).unwrap() == 42

or_else(f) #

Return a shallow copy of the Maybe if it contains a value, otherwise call a function to get a result.

pure(x) classmethod #

Implementation of Applicative.pure, which is equivalent to Maybe.just.

replace(val) #

Replace the inner value with a new value, returning the old value. After replacement, self will always have a value.

j = just[int](42)
old = j.replace(114514)
assert j == just(114514)
assert old == just(42)

n = nothing[int]()
old = n.replace(1919810)
assert n == just(1919810)
assert old == nothing()

some(val) classmethod #

Convert an Optional[T] value to a Maybe value.

Warning

Prefer using some function instead, unless in performance-critical code.

some = Maybe.some(42)
assert some.is_just()

none = Maybe.some(None)
assert none.is_nothing()

take() #

Take the inner value out and leave no value in place.

j = just[int](42)
out = j.take()
assert j.is_nothing()
assert out == just(42)

take_if(p) #

Take the inner value out if it satisfies the predicate, and leave no value in place. Otherwise, take out nothing.

j = just[int](42)
out = j.take_if(lambda x: x > 0)
assert j.is_nothing()
assert out == just(42)

j = just[int](42)
out = j.take_if(lambda x: x < 0)
assert j.is_just()
assert out == nothing()

tap(f) #

Call a function with the inner value, if any, and return the Maybe itself. Unlike Option::inspect, this method does not require the function to return None.

j = just[int](42)
j.tap(print)  # prints 42

unwrap() #

Unwrap the inner value, if any. Otherwise, raise a ValueError.

j = just[int](42)
n = nothing[int]()

assert j.unwrap() == 42

try: n.unwrap()
except ValueError: pass
else: assert False

Raises:

Type Description
ValueError

If the value is a Nothing.

unwrap_or(default) #

Unwrap the inner value, or return a default value if absent.

j = just[int](42)
n = nothing[int]()
assert j.unwrap_or(0) == 42
assert n.unwrap_or(0) == 0

unwrap_or_else(f) #

Unwrap the inner value, or return a value computed by a function if absent.

unwrap_unchecked() #

Return the inner value without checking if it is a Just or Nothing.

xor(other) #

If only one side has a value, return that side. Otherwise, return a Nothing.

j1 = just[int](42)
j2 = just[int](114514)
assert j1.xor(j2).is_nothing()

zip(*others) #

zip(*others: *tuple[]) -> Maybe[tuple[T,]]
zip(*others: *tuple[Maybe[T1],]) -> Maybe[tuple[T, T1]]
zip(*others: *tuple[Maybe[T1], Maybe[T2]]) -> Maybe[tuple[T, T1, T2]]
zip(*others: *tuple[Maybe[T1], Maybe[T2], Maybe[T3]]) -> Maybe[tuple[T, T1, T2, T3]]
zip(*others: *tuple[Maybe[T1], Maybe[T2], Maybe[T3], Maybe[T4]]) -> Maybe[tuple[T, T1, T2, T3, T4]]

Combine multiple Maybe values into a single Maybe value containing a tuple of them. If all values are Just, return a Just value with a tuple of inner values. Otherwise, return a Nothing. Calling this method with no arguments is equivalent to calling map with a tuple constructor.

Typing

The type checker only supports zipping up to 5-tuple.

j1 = just[int](42)
j2 = just[int](114514)
assert j1.zip(j2).unwrap() == (42, 114514)

n = nothing[int]()
assert j1.zip(n).is_nothing()