Skip to content

Commit 4613aaf

Browse files
committed
feat: add CCFun.with_return
1 parent 4294dc7 commit 4613aaf

File tree

3 files changed

+35
-0
lines changed

3 files changed

+35
-0
lines changed

src/core/CCFun.ml

+7
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,13 @@ let rec iterate n f x =
6161
else
6262
iterate (n - 1) f (f x)
6363

64+
let[@inline] with_return (type ret) f : ret =
65+
let exception E of ret in
66+
let return x = raise_notrace (E x) in
67+
match f return with
68+
| res -> res
69+
| exception E res -> res
70+
6471
module Infix = struct
6572
(* default implem for some operators *)
6673
let ( %> ) = compose

src/core/CCFun.mli

+16
Original file line numberDiff line numberDiff line change
@@ -78,6 +78,22 @@ val iterate : int -> ('a -> 'a) -> 'a -> 'a
7878
[x], [iterate 1 f x] is [f x], [iterate 2 f x] is [f (f x)], etc.
7979
@since 2.1 *)
8080

81+
val with_return : (('ret -> 'a) -> 'ret) -> 'ret
82+
(** [with_return f] is [f return], where [return] is a function
83+
that can be invoked to exit the scope early.
84+
85+
For example:
86+
{[
87+
let find_array arr x =
88+
let@ return = with_return in
89+
for i = 0 to Array.length arr-1 do
90+
if arr.(i) = x then return i;
91+
done;
92+
-1
93+
]}
94+
95+
@since NEXT_RELEASE *)
96+
8197
(** {2 Infix}
8298
8399
Infix operators. *)

tests/core/t_fun.ml

+12
Original file line numberDiff line numberDiff line change
@@ -21,3 +21,15 @@ true
2121
t @@ fun () -> CCFun.((succ %> string_of_int) 2 = "3");;
2222
t @@ fun () -> CCFun.((( * ) 3 % succ) 5 = 18);;
2323
t @@ fun () -> CCFun.(succ @@ ( * ) 2 @@ pred @@ 3 = 5)
24+
25+
let find_array arr x =
26+
let@ return = with_return in
27+
for i = 0 to Array.length arr - 1 do
28+
if arr.(i) = x then return i
29+
done;
30+
-1
31+
;;
32+
33+
eq 1 @@ find_array [| "a"; "b"; "c" |] "b";;
34+
eq 2 @@ find_array [| "a"; "b"; "c" |] "c";;
35+
eq (-1) @@ find_array [| "a"; "b"; "c" |] "hello"

0 commit comments

Comments
 (0)