Module Tablecloth.Option Functions for working with optional values.
Option represents a value which may not be present.
It is a variant containing the (Some 'a) and None constructors
type 'a t =
| Some of 'a
| NoneMany other languages use null or nil to represent something similar.
Option values are very common and they are used in a number of ways:
Initial values Optional function arguments Optional record fields Return values for functions that are not defined over their entire input range (partial functions). Return value for otherwise reporting simple errors, where None is returned on error. Lots of functions in Standard return options, one you have one you can work with the value it might contain by:
Pattern matching Using map or andThen (or their operators in Infix) Unwrapping it using unwrap , or its operator (|?) Converting a None into an exception usingunwrapUnsafe If the function you are writing can fail in a variety of ways, use a Result instead to better communicate with the caller.
If a function only fails in unexpected, unrecoverable ways, maybe you want raise exception.
val some : 'a -> 'a optionA function version of the Some constructor.
In most situations you just want to use the Some constructor directly.
However OCaml doesn't support piping to variant constructors.
Note that when using the Reason syntax you can use fast pipe (->) with variant constructors, so you don't need this function.
See the Reason docs for more.
Examples
String.reverse("desserts") |> Option.some = Some "desserts" val and_ : 'a t -> 'a t -> 'a t Returns None if the first argument is None, otherwise return the second argument.
Unlike the built in && operator, the and_ function does not short-circuit.
When you call and_, both arguments are evaluated before being passed to the function.
Examples
Option.and_ (Some 11) (Some 22) = Some 22Option.and_ None (Some 22) = NoneOption.and_ (Some 11) None = NoneOption.and_ None None = Noneval or_ : 'a t -> 'a t -> 'a t Return the first argument if it isSome , otherwise return the second.
Unlike the built in || operator, the or_ function does not short-circuit. When you call or_, both arguments are evaluated before being passed to the function.
Examples
Option.or_ (Some 11) (Some 22) = Some 11Option.or_ None (Some 22) = Some 22Option.or_ (Some 11) None = Some 11Option.or_ None None = Noneval orElse : 'a t -> 'a t -> 'a t Return the second argument if it isSome , otherwise return the first.
Like or_ but in reverse. Useful when using the |> operator
Examples
Option.orElse (Some 11) (Some 22) = Some 22Option.orElse None (Some 22) = Some 22Option.orElse (Some 11) None = Some 11Option.orElse None None = Noneval or_else : 'a t -> 'a t -> 'a t val both : 'a t -> 'b t -> ('a * 'b ) t Transform two options into an option of a Tuple.
Returns None if either of the aguments is None.
Examples
Option.both (Some 3004) (Some "Ant") = Some (3004, "Ant")Option.both (Some 3004) None = NoneOption.both None (Some "Ant") = NoneOption.both None None = Noneval flatten : 'a t t -> 'a t Flatten two optional layers into a single optional layer.
Examples
Option.flatten (Some (Some 4)) = Some 4Option.flatten (Some None) = NoneOption.flatten (None) = Noneval map : 'a t -> f :('a -> 'b ) -> 'b t Transform the value inside an option.
Leaves None untouched.
See (>>|) for an operator version of this function.
Examples
Option.map ~f:(fun x -> x * x) (Some 9) = Some 81Option.map ~f:Int.toString (Some 9) = Some "9"Option.map ~f:(fun x -> x * x) None = Noneval map2 : 'a t -> 'b t -> f :('a -> 'b -> 'c ) -> 'c t Combine two Option s
If both options are Some returns, as Some the result of running f on both values.
If either value is None, returns None
Examples
Option.map2 (Some 3) (Some 4) ~f:Int.add = Some 7Option.map2 (Some 3) (Some 4) ~f:Tuple.make = Some (3, 4)Option.map2 (Some 3) None ~f:Int.add = NoneOption.map2 None (Some 4) ~f:Int.add = Noneval andThen : 'a t -> f :('a -> 'b t ) -> 'b t Chain together many computations that may not return a value.
It is helpful to see its definition:
let andThen t ~f =
match t with
| Some x -> f x
| None -> NoneThis means we only continue with the callback if we have a value.
For example, say you need to parse some user input as a month:
let toValidMonth (month: int) : (int option) =
if (1 <= month && month <= 12) then
Some month
else
None
in
let userInput = "5" in
Int.fromString userInput
|> Option.andThen ~f:toValidMonthIf String.toInt produces None (because the userInput was not an integer) this entire chain of operations will short-circuit and result in None. If toValidMonth results in None, again the chain of computations will result in None.
See (>>=) for an operator version of this function.
Examples
Option.andThen (Some [1, 2, 3]) ~f:List.head = Some 1Option.andThen (Some []) ~f:List.head = Noneval and_then : 'a t -> f :('a -> 'b t ) -> 'b t val unwrap : 'a t -> default :'a -> 'a Unwrap an option('a) returning default if called with None.
This comes in handy when paired with functions like Map.get or List.head which return an Option .
See (|?) for an operator version of this function.
Note This can be overused! Many cases are better handled using pattern matching, map or andThen .
Examples
Option.unwrap ~default:99 (Some 42) = 42Option.unwrap ~default:99 None = 99Option.unwrap ~default:"unknown" (Map.get Map.String.empty "Tom") = "unknown"val unwrapUnsafe : 'a t -> 'a Unwrap an option('a) returning the enclosed 'a.
Note in most situations it is better to use pattern matching, unwrap , map or andThen . Can you structure your code slightly differently to avoid potentially raising an exception?
Exceptions
Raises an Invalid_argument exception if called with None
Examples
List.head [1;2;3] |> Option.unwrapUnsafe = 1List.head [] |> Option.unwrapUnsafeval unwrap_unsafe : 'a t -> 'a val isSome : 'a t -> boolCheck if an Option is a Some.
In most situtations you should just use pattern matching instead.
Examples
Option.isSome (Some 3004) = trueOption.isSome None = falseval is_some : 'a t -> boolval isNone : 'a t -> boolCheck if an Option is a None.
In most situtations you should just use pattern matching instead.
Examples
Option.isNone (Some 3004) = falseOption.isNone None = trueval is_none : 'a t -> boolval tap : 'a t -> f :('a -> unit) -> unitRun a function against a value, if it is present.
val toArray : 'a t -> 'a arrayConvert an option to a Array .
None is represented as an empty list and Some is represented as a list of one element.
Examples
Option.toArray (Some 3004) = [|3004|]Option.toArray (None) = [||]val to_array : 'a t -> 'a arrayval toList : 'a t -> 'a listConvert an option to a List .
None is represented as an empty list and Some is represented as a list of one element.
Examples
Option.toList (Some 3004) = [3004]Option.toList (None) = []val to_list : 'a t -> 'a list Compareval equal : ('a -> 'a -> bool) -> 'a t -> 'a t -> boolTest two optional values for equality using the provided function
Examples
Option.equal Int.equal (Some 1) (Some 1) = trueOption.equal Int.equal (Some 1) (Some 3) = falseOption.equal Int.equal (Some 1) None = falseOption.equal Int.equal None None = trueval compare : ('a -> 'a -> int) -> 'a t -> 'a t -> intCompare two optional values using the provided function.
A None is "less" than a Some
Examples
Option.compare Int.compare (Some 1) (Some 3) = -1Option.compare Int.compare (Some 1) None = 1Option.compare Int.compare None None = 0 OperatorsFor code that works extensively with Option s these operators can make things significantly more concise at the expense of placing a greater cognitive burden on future readers.
let nameToAge = Map.String.fromArray [|
("Ant", 1);
("Bat", 5);
("Cat", 19);
|] in
let catAge = Map.get nameToAge "Cat" |? 8 in
(* 19 *)
Option.(
Map.get nameToAge "Ant" >>= (fun antAge ->
Map.get nameToAge "Bat" >>| (fun batAge ->
Int.absolute(batAge - antAge)
)
)
)
(* Some (4) *)val (|?) : 'a t -> 'a -> 'a The operator version of get
Examples
Some 3004 |? 8 = 3004None |? 8 = 8val (>>|) : 'a t -> ('a -> 'b ) -> 'b t The operator version of map
Examples
Some "desserts" >>| String.reverse = Some "stressed"None >>| String.reverse = Noneval (>>=) : 'a t -> ('a -> 'b t ) -> 'b t The operator version of andThen
Examples
Some [1, 2, 3] >>= List.head = Some 1Some [] >>= List.head = None